From b82b6c30c368ae47c0d8ad557d61a57eaf80d21c Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" Date: Wed, 9 Oct 2024 00:39:20 +0000 Subject: [PATCH] Added chart versions: btp/chronicle: - 0.1.27 jfrog/artifactory-ha: - 107.90.14 jfrog/artifactory-jcr: - 107.90.14 kasten/k10: - 7.0.11 new-relic/nri-bundle: - 5.0.94 stackstate/stackstate-k8s-agent: - 1.0.98 --- assets/jfrog/artifactory-ha-107.90.14.tgz | Bin 0 -> 218393 bytes assets/jfrog/artifactory-jcr-107.90.14.tgz | Bin 0 -> 459461 bytes assets/kasten/k10-7.0.1101.tgz | Bin 0 -> 223612 bytes assets/new-relic/nri-bundle-5.0.94.tgz | Bin 0 -> 353607 bytes .../stackstate-k8s-agent-1.0.98.tgz | Bin 0 -> 34903 bytes .../artifactory-ha/107.90.14/.helmignore | 24 + .../artifactory-ha/107.90.14/CHANGELOG.md | 1466 +++ .../jfrog/artifactory-ha/107.90.14/Chart.lock | 6 + .../jfrog/artifactory-ha/107.90.14/Chart.yaml | 30 + charts/jfrog/artifactory-ha/107.90.14/LICENSE | 201 + .../jfrog/artifactory-ha/107.90.14/README.md | 69 + .../artifactory-ha/107.90.14/app-readme.md | 16 + .../107.90.14/charts/postgresql/.helmignore | 21 + .../107.90.14/charts/postgresql/Chart.lock | 6 + .../107.90.14/charts/postgresql/Chart.yaml | 29 + .../107.90.14/charts/postgresql/README.md | 770 ++ .../postgresql/charts/common/.helmignore | 22 + .../postgresql/charts/common/Chart.yaml | 23 + .../charts/postgresql/charts/common/README.md | 322 + .../charts/common/templates/_affinities.tpl | 94 + .../charts/common/templates/_capabilities.tpl | 95 + .../charts/common/templates/_errors.tpl | 23 + .../charts/common/templates/_images.tpl | 47 + .../charts/common/templates/_ingress.tpl | 42 + .../charts/common/templates/_labels.tpl | 18 + .../charts/common/templates/_names.tpl | 32 + .../charts/common/templates/_secrets.tpl | 129 + .../charts/common/templates/_storage.tpl | 23 + .../charts/common/templates/_tplvalues.tpl | 13 + .../charts/common/templates/_utils.tpl | 62 + .../charts/common/templates/_warnings.tpl | 14 + .../templates/validations/_cassandra.tpl | 72 + .../common/templates/validations/_mariadb.tpl | 103 + .../common/templates/validations/_mongodb.tpl | 108 + .../templates/validations/_postgresql.tpl | 131 + .../common/templates/validations/_redis.tpl | 72 + .../templates/validations/_validations.tpl | 46 + .../postgresql/charts/common/values.yaml | 3 + .../postgresql/ci/commonAnnotations.yaml | 3 + .../charts/postgresql/ci/default-values.yaml | 1 + .../ci/shmvolume-disabled-values.yaml | 2 + .../charts/postgresql/files/README.md | 1 + .../charts/postgresql/files/conf.d/README.md | 4 + .../docker-entrypoint-initdb.d/README.md | 3 + .../charts/postgresql/templates/NOTES.txt | 59 + .../charts/postgresql/templates/_helpers.tpl | 337 + .../postgresql/templates/configmap.yaml | 31 + .../templates/extended-config-configmap.yaml | 26 + .../postgresql/templates/extra-list.yaml | 4 + .../templates/initialization-configmap.yaml | 25 + .../templates/metrics-configmap.yaml | 14 + .../postgresql/templates/metrics-svc.yaml | 26 + .../postgresql/templates/networkpolicy.yaml | 39 + .../templates/podsecuritypolicy.yaml | 38 + .../postgresql/templates/prometheusrule.yaml | 23 + .../charts/postgresql/templates/role.yaml | 20 + .../postgresql/templates/rolebinding.yaml | 20 + .../charts/postgresql/templates/secrets.yaml | 24 + .../postgresql/templates/serviceaccount.yaml | 12 + .../postgresql/templates/servicemonitor.yaml | 33 + .../templates/statefulset-readreplicas.yaml | 411 + .../postgresql/templates/statefulset.yaml | 609 ++ .../postgresql/templates/svc-headless.yaml | 28 + .../charts/postgresql/templates/svc-read.yaml | 43 + .../charts/postgresql/templates/svc.yaml | 41 + .../charts/postgresql/values.schema.json | 103 + .../107.90.14/charts/postgresql/values.yaml | 824 ++ .../107.90.14/ci/access-tls-values.yaml | 34 + .../107.90.14/ci/default-values.yaml | 32 + .../107.90.14/ci/global-values.yaml | 255 + .../107.90.14/ci/large-values.yaml | 85 + .../107.90.14/ci/loggers-values.yaml | 43 + .../107.90.14/ci/medium-values.yaml | 85 + .../ci/migration-disabled-values.yaml | 31 + .../107.90.14/ci/nginx-autoreload-values.yaml | 53 + .../ci/rtsplit-access-tls-values.yaml | 106 + .../107.90.14/ci/rtsplit-values.yaml | 155 + .../107.90.14/ci/small-values.yaml | 87 + .../107.90.14/ci/test-values.yaml | 85 + .../107.90.14/files/binarystore.xml | 439 + .../107.90.14/files/installer-info.json | 32 + .../artifactory-ha/107.90.14/files/migrate.sh | 4311 ++++++++ .../107.90.14/files/migrationHelmInfo.yaml | 27 + .../107.90.14/files/migrationStatus.sh | 44 + .../files/nginx-artifactory-conf.yaml | 98 + .../107.90.14/files/nginx-main-conf.yaml | 83 + .../107.90.14/files/system.yaml | 163 + .../107.90.14/logo/artifactory-logo.png | Bin 0 -> 82419 bytes .../artifactory-ha/107.90.14/questions.yml | 424 + .../artifactory-2xlarge-extra-config.yaml | 44 + .../107.90.14/sizing/artifactory-2xlarge.yaml | 127 + .../artifactory-large-extra-config.yaml | 44 + .../107.90.14/sizing/artifactory-large.yaml | 127 + .../artifactory-medium-extra-config.yaml | 45 + .../107.90.14/sizing/artifactory-medium.yaml | 127 + .../artifactory-small-extra-config.yaml | 43 + .../107.90.14/sizing/artifactory-small.yaml | 127 + .../artifactory-xlarge-extra-config.yaml | 42 + .../107.90.14/sizing/artifactory-xlarge.yaml | 127 + .../artifactory-xsmall-extra-config.yaml | 43 + .../107.90.14/sizing/artifactory-xsmall.yaml | 127 + .../107.90.14/templates/NOTES.txt | 149 + .../107.90.14/templates/_helpers.tpl | 563 + .../templates/_system-yaml-render.tpl | 5 + .../templates/additional-resources.yaml | 3 + .../templates/admin-bootstrap-creds.yaml | 15 + .../templates/artifactory-access-config.yaml | 15 + .../artifactory-binarystore-secret.yaml | 18 + .../templates/artifactory-configmaps.yaml | 13 + .../templates/artifactory-custom-secrets.yaml | 19 + .../artifactory-database-secrets.yaml | 24 + .../artifactory-gcp-credentials-secret.yaml | 16 + .../templates/artifactory-installer-info.yaml | 16 + .../templates/artifactory-license-secret.yaml | 16 + .../artifactory-migration-scripts.yaml | 18 + .../templates/artifactory-networkpolicy.yaml | 34 + .../templates/artifactory-nfs-pvc.yaml | 101 + .../templates/artifactory-node-pdb.yaml | 26 + .../artifactory-node-statefulset.yaml | 1588 +++ .../templates/artifactory-primary-pdb.yaml | 24 + .../artifactory-primary-service.yaml | 57 + .../artifactory-primary-statefulset.yaml | 1717 ++++ .../templates/artifactory-priority-class.yaml | 9 + .../107.90.14/templates/artifactory-role.yaml | 14 + .../templates/artifactory-rolebinding.yaml | 19 + .../templates/artifactory-secrets.yaml | 30 + .../templates/artifactory-service.yaml | 72 + .../templates/artifactory-serviceaccount.yaml | 17 + .../templates/artifactory-storage-pvc.yaml | 27 + .../templates/artifactory-system-yaml.yaml | 16 + .../templates/artifactory-unified-secret.yaml | 96 + .../templates/filebeat-configmap.yaml | 15 + .../107.90.14/templates/ingress.yaml | 106 + .../107.90.14/templates/logger-configmap.yaml | 63 + .../templates/nginx-artifactory-conf.yaml | 18 + .../templates/nginx-certificate-secret.yaml | 14 + .../107.90.14/templates/nginx-conf.yaml | 18 + .../107.90.14/templates/nginx-deployment.yaml | 221 + .../107.90.14/templates/nginx-pdb.yaml | 23 + .../107.90.14/templates/nginx-pvc.yaml | 26 + .../templates/nginx-scripts-conf.yaml | 52 + .../107.90.14/templates/nginx-service.yaml | 94 + .../artifactory-ha/107.90.14/values.yaml | 1834 ++++ .../artifactory-jcr/107.90.14/CHANGELOG.md | 206 + .../artifactory-jcr/107.90.14/Chart.yaml | 30 + .../jfrog/artifactory-jcr/107.90.14/LICENSE | 201 + .../jfrog/artifactory-jcr/107.90.14/README.md | 125 + .../artifactory-jcr/107.90.14/app-readme.md | 18 + .../107.90.14/charts/artifactory/.helmignore | 24 + .../107.90.14/charts/artifactory/CHANGELOG.md | 1365 +++ .../107.90.14/charts/artifactory/Chart.lock | 6 + .../107.90.14/charts/artifactory/Chart.yaml | 24 + .../107.90.14/charts/artifactory/LICENSE | 201 + .../107.90.14/charts/artifactory/README.md | 59 + .../artifactory/charts/postgresql/.helmignore | 21 + .../artifactory/charts/postgresql/Chart.lock | 6 + .../artifactory/charts/postgresql/Chart.yaml | 29 + .../artifactory/charts/postgresql/README.md | 770 ++ .../postgresql/charts/common/.helmignore | 22 + .../postgresql/charts/common/Chart.yaml | 23 + .../charts/postgresql/charts/common/README.md | 322 + .../charts/common/templates/_affinities.tpl | 94 + .../charts/common/templates/_capabilities.tpl | 95 + .../charts/common/templates/_errors.tpl | 23 + .../charts/common/templates/_images.tpl | 47 + .../charts/common/templates/_ingress.tpl | 42 + .../charts/common/templates/_labels.tpl | 18 + .../charts/common/templates/_names.tpl | 32 + .../charts/common/templates/_secrets.tpl | 129 + .../charts/common/templates/_storage.tpl | 23 + .../charts/common/templates/_tplvalues.tpl | 13 + .../charts/common/templates/_utils.tpl | 62 + .../charts/common/templates/_warnings.tpl | 14 + .../templates/validations/_cassandra.tpl | 72 + .../common/templates/validations/_mariadb.tpl | 103 + .../common/templates/validations/_mongodb.tpl | 108 + .../templates/validations/_postgresql.tpl | 131 + .../common/templates/validations/_redis.tpl | 72 + .../templates/validations/_validations.tpl | 46 + .../postgresql/charts/common/values.yaml | 3 + .../postgresql/ci/commonAnnotations.yaml | 3 + .../charts/postgresql/ci/default-values.yaml | 1 + .../ci/shmvolume-disabled-values.yaml | 2 + .../charts/postgresql/files/README.md | 1 + .../charts/postgresql/files/conf.d/README.md | 4 + .../docker-entrypoint-initdb.d/README.md | 3 + .../charts/postgresql/templates/NOTES.txt | 59 + .../charts/postgresql/templates/_helpers.tpl | 337 + .../postgresql/templates/configmap.yaml | 31 + .../templates/extended-config-configmap.yaml | 26 + .../postgresql/templates/extra-list.yaml | 4 + .../templates/initialization-configmap.yaml | 25 + .../templates/metrics-configmap.yaml | 14 + .../postgresql/templates/metrics-svc.yaml | 26 + .../postgresql/templates/networkpolicy.yaml | 39 + .../templates/podsecuritypolicy.yaml | 38 + .../postgresql/templates/prometheusrule.yaml | 23 + .../charts/postgresql/templates/role.yaml | 20 + .../postgresql/templates/rolebinding.yaml | 20 + .../charts/postgresql/templates/secrets.yaml | 24 + .../postgresql/templates/serviceaccount.yaml | 12 + .../postgresql/templates/servicemonitor.yaml | 33 + .../templates/statefulset-readreplicas.yaml | 411 + .../postgresql/templates/statefulset.yaml | 609 ++ .../postgresql/templates/svc-headless.yaml | 28 + .../charts/postgresql/templates/svc-read.yaml | 43 + .../charts/postgresql/templates/svc.yaml | 41 + .../charts/postgresql/values.schema.json | 103 + .../artifactory/charts/postgresql/values.yaml | 824 ++ .../artifactory/ci/access-tls-values.yaml | 24 + .../charts/artifactory/ci/default-values.yaml | 21 + .../artifactory/ci/derby-test-values.yaml | 19 + .../charts/artifactory/ci/global-values.yaml | 247 + .../charts/artifactory/ci/large-values.yaml | 82 + .../charts/artifactory/ci/loggers-values.yaml | 43 + .../charts/artifactory/ci/medium-values.yaml | 82 + .../ci/migration-disabled-values.yaml | 21 + .../ci/nginx-autoreload-values.yaml | 42 + .../ci/rtsplit-values-access-tls-values.yaml | 96 + .../charts/artifactory/ci/rtsplit-values.yaml | 151 + .../charts/artifactory/ci/small-values.yaml | 82 + .../charts/artifactory/ci/test-values.yaml | 84 + .../charts/artifactory/files/binarystore.xml | 426 + .../artifactory/files/installer-info.json | 32 + .../charts/artifactory/files/migrate.sh | 4311 ++++++++ .../artifactory/files/migrationHelmInfo.yaml | 27 + .../artifactory/files/migrationStatus.sh | 44 + .../files/nginx-artifactory-conf.yaml | 98 + .../artifactory/files/nginx-main-conf.yaml | 83 + .../charts/artifactory/files/system.yaml | 156 + .../artifactory/logo/artifactory-logo.png | Bin 0 -> 82419 bytes .../artifactory-2xlarge-extra-config.yaml | 41 + .../sizing/artifactory-2xlarge.yaml | 126 + .../artifactory-large-extra-config.yaml | 41 + .../artifactory/sizing/artifactory-large.yaml | 126 + .../artifactory-medium-extra-config.yaml | 41 + .../sizing/artifactory-medium.yaml | 126 + .../artifactory-small-extra-config.yaml | 41 + .../artifactory/sizing/artifactory-small.yaml | 124 + .../artifactory-xlarge-extra-config.yaml | 41 + .../sizing/artifactory-xlarge.yaml | 126 + .../artifactory-xsmall-extra-config.yaml | 42 + .../sizing/artifactory-xsmall.yaml | 125 + .../charts/artifactory/templates/NOTES.txt | 106 + .../charts/artifactory/templates/_helpers.tpl | 528 + .../templates/_system-yaml-render.tpl | 5 + .../templates/additional-resources.yaml | 3 + .../templates/admin-bootstrap-creds.yaml | 15 + .../templates/artifactory-access-config.yaml | 15 + .../artifactory-binarystore-secret.yaml | 18 + .../templates/artifactory-configmaps.yaml | 13 + .../templates/artifactory-custom-secrets.yaml | 19 + .../artifactory-database-secrets.yaml | 24 + .../artifactory-gcp-credentials-secret.yaml | 16 + .../templates/artifactory-hpa.yaml | 29 + .../templates/artifactory-installer-info.yaml | 16 + .../templates/artifactory-license-secret.yaml | 16 + .../artifactory-migration-scripts.yaml | 18 + .../templates/artifactory-networkpolicy.yaml | 34 + .../templates/artifactory-nfs-pvc.yaml | 101 + .../templates/artifactory-pdb.yaml | 24 + .../templates/artifactory-priority-class.yaml | 9 + .../templates/artifactory-role.yaml | 14 + .../templates/artifactory-rolebinding.yaml | 19 + .../templates/artifactory-secrets.yaml | 30 + .../templates/artifactory-service.yaml | 63 + .../templates/artifactory-serviceaccount.yaml | 17 + .../templates/artifactory-statefulset.yaml | 1633 +++ .../templates/artifactory-system-yaml.yaml | 16 + .../templates/artifactory-unified-secret.yaml | 94 + .../templates/filebeat-configmap.yaml | 15 + .../charts/artifactory/templates/ingress.yaml | 109 + .../templates/logger-configmap.yaml | 63 + .../templates/nginx-artifactory-conf.yaml | 18 + .../templates/nginx-certificate-secret.yaml | 14 + .../artifactory/templates/nginx-conf.yaml | 18 + .../templates/nginx-deployment.yaml | 223 + .../artifactory/templates/nginx-pdb.yaml | 23 + .../artifactory/templates/nginx-pvc.yaml | 26 + .../templates/nginx-scripts-conf.yaml | 52 + .../artifactory/templates/nginx-service.yaml | 88 + .../107.90.14/charts/artifactory/values.yaml | 1749 ++++ .../107.90.14/ci/default-values.yaml | 7 + .../107.90.14/logo/jcr-logo.png | Bin 0 -> 77047 bytes .../artifactory-jcr/107.90.14/questions.yml | 271 + .../107.90.14/templates/NOTES.txt | 1 + .../artifactory-jcr/107.90.14/values.yaml | 75 + charts/kasten/k10/7.0.1101/Chart.lock | 9 + charts/kasten/k10/7.0.1101/Chart.yaml | 25 + charts/kasten/k10/7.0.1101/README.md | 342 + charts/kasten/k10/7.0.1101/app-readme.md | 5 + .../k10/7.0.1101/charts/grafana/.helmignore | 23 + .../k10/7.0.1101/charts/grafana/Chart.yaml | 35 + .../k10/7.0.1101/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 | 172 + .../charts/grafana/templates/_helpers.tpl | 276 + .../charts/grafana/templates/_pod.tpl | 1320 +++ .../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.1101/charts/grafana/values.yaml | 1386 +++ .../7.0.1101/charts/prometheus/.helmignore | 23 + .../k10/7.0.1101/charts/prometheus/Chart.yaml | 53 + .../k10/7.0.1101/charts/prometheus/OWNERS | 6 + .../k10/7.0.1101/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 | 251 + .../templates/tests/test-connection.yaml | 20 + .../charts/alertmanager/values.schema.json | 923 ++ .../charts/alertmanager/values.yaml | 379 + .../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 | 313 + .../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 | 522 + .../prometheus-node-exporter/.helmignore | 21 + .../prometheus-node-exporter/Chart.yaml | 25 + .../charts/prometheus-node-exporter/README.md | 96 + .../ci/port-values.yaml | 3 + .../templates/NOTES.txt | 29 + .../templates/_helpers.tpl | 202 + .../templates/clusterrole.yaml | 19 + .../templates/clusterrolebinding.yaml | 20 + .../templates/daemonset.yaml | 311 + .../templates/endpoints.yaml | 18 + .../templates/extra-manifests.yaml | 4 + .../templates/networkpolicy.yaml | 23 + .../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 | 35 + .../templates/serviceaccount.yaml | 18 + .../templates/servicemonitor.yaml | 61 + .../templates/verticalpodautoscaler.yaml | 40 + .../prometheus-node-exporter/values.yaml | 533 + .../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.1101/charts/prometheus/values.yaml | 1309 +++ charts/kasten/k10/7.0.1101/config.json | 0 charts/kasten/k10/7.0.1101/eula.txt | 459 + charts/kasten/k10/7.0.1101/files/favicon.png | Bin 0 -> 1802 bytes .../kasten/k10/7.0.1101/files/kasten-logo.svg | 24 + charts/kasten/k10/7.0.1101/files/styles.css | 113 + .../grafana/dashboards/default/default.json | 5855 +++++++++++ charts/kasten/k10/7.0.1101/license | 1 + charts/kasten/k10/7.0.1101/questions.yaml | 295 + .../kasten/k10/7.0.1101/templates/NOTES.txt | 106 + .../k10/7.0.1101/templates/_definitions.tpl | 238 + .../k10/7.0.1101/templates/_grafana.tpl | 18 + .../k10/7.0.1101/templates/_helpers.tpl | 1524 +++ .../k10/7.0.1101/templates/_k10_container.tpl | 1121 ++ .../k10/7.0.1101/templates/_k10_image_tag.tpl | 1 + .../k10/7.0.1101/templates/_k10_metering.tpl | 356 + .../7.0.1101/templates/_k10_serviceimage.tpl | 50 + .../k10/7.0.1101/templates/_k10_template.tpl | 234 + .../k10/7.0.1101/templates/_prometheus.tpl | 29 + .../templates/aggregatedaudit-policy.yaml | 34 + .../7.0.1101/templates/api-tls-secrets.yaml | 13 + .../k10/7.0.1101/templates/apiservice.yaml | 25 + .../k10/7.0.1101/templates/daemonsets.yaml | 26 + .../k10/7.0.1101/templates/deployments.yaml | 31 + .../templates/fluentbit-configmap.yaml | 34 + .../templates/frontend-nginx-configmap.yaml | 50 + .../k10/7.0.1101/templates/gateway-ext.yaml | 36 + .../k10/7.0.1101/templates/gateway.yaml | 253 + .../k10/7.0.1101/templates/grafana-scc.yaml | 45 + .../k10/7.0.1101/templates/ingress.yaml | 73 + .../k10/7.0.1101/templates/k10-config.yaml | 271 + .../k10/7.0.1101/templates/k10-eula.yaml | 21 + .../k10/7.0.1101/templates/k10-scc.yaml | 46 + .../7.0.1101/templates/kopia-tls-certs.yaml | 33 + .../k10/7.0.1101/templates/license.yaml | 25 + charts/kasten/k10/7.0.1101/templates/mc.yaml | 6 + .../7.0.1101/templates/mutatingwebhook.yaml | 51 + .../k10/7.0.1101/templates/networkpolicy.yaml | 282 + .../templates/ocp-ca-cert-extract-hook.yaml | 195 + .../templates/prometheus-configmap.yaml | 97 + .../7.0.1101/templates/prometheus-scc.yaml | 41 + .../templates/prometheus-service.yaml | 46 + .../kasten/k10/7.0.1101/templates/rbac.yaml | 381 + .../k10/7.0.1101/templates/rhmarketplace.tpl | 8 + .../kasten/k10/7.0.1101/templates/route.yaml | 36 + .../k10/7.0.1101/templates/secrets.yaml | 257 + .../7.0.1101/templates/secure_deployment.tpl | 19 + .../7.0.1101/templates/serviceaccount.yaml | 44 + .../k10/7.0.1101/templates/v0services.yaml | 200 + .../templates/workloadIdentityFederation.tpl | 10 + .../grafana/values/grafana_values.tpl | 275 + .../{charts}/values/prometheus_values.tpl | 183 + charts/kasten/k10/7.0.1101/triallicense | 1 + charts/kasten/k10/7.0.1101/values.schema.json | 2874 ++++++ charts/kasten/k10/7.0.1101/values.yaml | 545 + .../new-relic/nri-bundle/5.0.94/.helmignore | 22 + charts/new-relic/nri-bundle/5.0.94/Chart.lock | 39 + charts/new-relic/nri-bundle/5.0.94/Chart.yaml | 85 + charts/new-relic/nri-bundle/5.0.94/README.md | 200 + .../nri-bundle/5.0.94/README.md.gotmpl | 166 + .../new-relic/nri-bundle/5.0.94/app-readme.md | 5 + .../charts/k8s-agents-operator/.helmignore | 23 + .../charts/k8s-agents-operator/Chart.yaml | 16 + .../charts/k8s-agents-operator/README.md | 204 + .../k8s-agents-operator/README.md.gotmpl | 162 + .../k8s-agents-operator/templates/NOTES.txt | 36 + .../templates/_helpers.tpl | 121 + .../templates/certmanager.yaml | 30 + .../templates/deployment.yaml | 91 + .../templates/instrumentation-crd.yaml | 1150 +++ .../templates/leader-election-rbac.yaml | 51 + .../templates/manager-rbac.yaml | 77 + .../templates/newrelic_license_secret.yaml | 14 + .../templates/proxy-rbac.yaml | 35 + .../templates/reader-rbac.yaml | 11 + .../templates/service.yaml | 15 + .../templates/webhook-configuration.yaml | 134 + .../templates/webhook-service.yaml | 14 + .../tests/cert_manager_test.yaml | 85 + .../tests/webhook_ssl_test.yaml | 176 + .../charts/k8s-agents-operator/values.yaml | 93 + .../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 | 279 + .../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 | 49 + .../templates/serviceaccount.yaml | 15 + .../templates/servicemonitor.yaml | 114 + .../templates/stsdiscovery-role.yaml | 26 + .../templates/stsdiscovery-rolebinding.yaml | 17 + .../templates/verticalpodautoscaler.yaml | 44 + .../charts/kube-state-metrics/values.yaml | 441 + .../newrelic-infra-operator/.helmignore | 1 + .../charts/newrelic-infra-operator/Chart.lock | 6 + .../charts/newrelic-infra-operator/Chart.yaml | 35 + .../charts/newrelic-infra-operator/README.md | 114 + .../newrelic-infra-operator/README.md.gotmpl | 77 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-values.yaml | 39 + .../templates/NOTES.txt | 4 + .../templates/_helpers.tpl | 136 + .../job-patch/clusterrole.yaml | 27 + .../job-patch/clusterrolebinding.yaml | 20 + .../job-patch/job-createSecret.yaml | 57 + .../job-patch/job-patchWebhook.yaml | 57 + .../admission-webhooks/job-patch/psp.yaml | 50 + .../admission-webhooks/job-patch/role.yaml | 21 + .../job-patch/rolebinding.yaml | 21 + .../job-patch/serviceaccount.yaml | 14 + .../mutatingWebhookConfiguration.yaml | 32 + .../templates/cert-manager.yaml | 52 + .../templates/clusterrole.yaml | 39 + .../templates/clusterrolebinding.yaml | 26 + .../templates/configmap.yaml | 9 + .../templates/deployment.yaml | 92 + .../templates/secret.yaml | 2 + .../templates/service.yaml | 13 + .../templates/serviceaccount.yaml | 13 + .../tests/deployment_test.yaml | 32 + .../tests/job_patch_psp_test.yaml | 23 + .../tests/job_serviceaccount_test.yaml | 64 + .../tests/rbac_test.yaml | 41 + .../newrelic-infra-operator/values.yaml | 222 + .../newrelic-infrastructure/.helmignore | 1 + .../charts/newrelic-infrastructure/Chart.lock | 6 + .../charts/newrelic-infrastructure/Chart.yaml | 26 + .../charts/newrelic-infrastructure/README.md | 220 + .../newrelic-infrastructure/README.md.gotmpl | 137 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../test-cplane-kind-deployment-values.yaml | 135 + .../ci/test-values.yaml | 134 + .../templates/NOTES.txt | 131 + .../templates/_helpers.tpl | 118 + .../templates/_helpers_compatibility.tpl | 202 + .../templates/clusterrole.yaml | 35 + .../templates/clusterrolebinding.yaml | 16 + .../controlplane/_affinity_helper.tpl | 11 + .../controlplane/_agent-config_helper.tpl | 20 + .../templates/controlplane/_host_network.tpl | 22 + .../templates/controlplane/_naming.tpl | 16 + .../templates/controlplane/_rbac.tpl | 40 + .../controlplane/_tolerations_helper.tpl | 11 + .../controlplane/agent-configmap.yaml | 18 + .../templates/controlplane/clusterrole.yaml | 47 + .../controlplane/clusterrolebinding.yaml | 16 + .../templates/controlplane/daemonset.yaml | 205 + .../templates/controlplane/rolebinding.yaml | 21 + .../controlplane/scraper-configmap.yaml | 36 + .../controlplane/serviceaccount.yaml | 13 + .../templates/ksm/_affinity_helper.tpl | 14 + .../templates/ksm/_agent-config_helper.tpl | 20 + .../templates/ksm/_host_network.tpl | 22 + .../templates/ksm/_naming.tpl | 8 + .../templates/ksm/_tolerations_helper.tpl | 11 + .../templates/ksm/agent-configmap.yaml | 18 + .../templates/ksm/deployment.yaml | 192 + .../templates/ksm/scraper-configmap.yaml | 15 + .../templates/kubelet/_affinity_helper.tpl | 33 + .../kubelet/_agent-config_helper.tpl | 31 + .../templates/kubelet/_host_network.tpl | 22 + .../templates/kubelet/_naming.tpl | 12 + .../kubelet/_security_context_helper.tpl | 32 + .../templates/kubelet/_tolerations_helper.tpl | 11 + .../templates/kubelet/agent-configmap.yaml | 18 + .../templates/kubelet/daemonset.yaml | 265 + .../kubelet/integrations-configmap.yaml | 72 + .../templates/kubelet/scraper-configmap.yaml | 18 + .../templates/podsecuritypolicy.yaml | 26 + .../templates/secret.yaml | 2 + .../templates/serviceaccount.yaml | 13 + .../newrelic-infrastructure/values.yaml | 602 ++ .../newrelic-k8s-metrics-adapter/.helmignore | 25 + .../newrelic-k8s-metrics-adapter/Chart.lock | 6 + .../newrelic-k8s-metrics-adapter/Chart.yaml | 25 + .../newrelic-k8s-metrics-adapter/README.md | 139 + .../README.md.gotmpl | 107 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-values.yaml | 14 + .../templates/_helpers.tpl | 57 + .../templates/adapter-clusterrolebinding.yaml | 14 + .../templates/adapter-rolebinding.yaml | 15 + .../templates/apiservice/apiservice.yaml | 19 + .../apiservice/job-patch/clusterrole.yaml | 26 + .../job-patch/clusterrolebinding.yaml | 19 + .../job-patch/job-createSecret.yaml | 55 + .../job-patch/job-patchAPIService.yaml | 53 + .../templates/apiservice/job-patch/psp.yaml | 49 + .../templates/apiservice/job-patch/role.yaml | 20 + .../apiservice/job-patch/rolebinding.yaml | 20 + .../apiservice/job-patch/serviceaccount.yaml | 18 + .../templates/configmap.yaml | 19 + .../templates/deployment.yaml | 113 + .../templates/hpa-clusterrole.yaml | 15 + .../templates/hpa-clusterrolebinding.yaml | 14 + .../templates/secret.yaml | 10 + .../templates/service.yaml | 13 + .../templates/serviceaccount.yaml | 13 + .../tests/apiservice_test.yaml | 22 + .../tests/common_extra_naming_test.yaml | 27 + .../tests/configmap_test.yaml | 104 + .../tests/deployment_test.yaml | 99 + .../tests/hpa_clusterrolebinding_test.yaml | 18 + .../job_patch_cluster_rolebinding_test.yaml | 22 + .../tests/job_patch_clusterrole_test.yaml | 20 + .../tests/job_patch_common_test.yaml | 27 + .../job_patch_job_createsecret_test.yaml | 47 + .../job_patch_job_patchapiservice_test.yaml | 56 + .../tests/job_serviceaccount_test.yaml | 79 + .../tests/rbac_test.yaml | 50 + .../newrelic-k8s-metrics-adapter/values.yaml | 156 + .../5.0.94/charts/newrelic-logging/Chart.lock | 6 + .../5.0.94/charts/newrelic-logging/Chart.yaml | 20 + .../5.0.94/charts/newrelic-logging/README.md | 268 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 663 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 56 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-enable-windows-values.yaml | 2 + .../ci/test-lowdatamode-values.yaml | 1 + .../ci/test-override-global-lowdatamode.yaml | 3 + .../ci/test-staging-values.yaml | 1 + .../ci/test-with-empty-global.yaml | 1 + .../ci/test-with-empty-values.yaml | 0 ...and-plugin-metrics-dashboard-template.json | 2237 ++++ .../newrelic-logging/templates/NOTES.txt | 18 + .../newrelic-logging/templates/_helpers.tpl | 215 + .../templates/clusterrole.yaml | 23 + .../templates/clusterrolebinding.yaml | 15 + .../newrelic-logging/templates/configmap.yaml | 38 + .../templates/daemonset-windows.yaml | 174 + .../newrelic-logging/templates/daemonset.yaml | 211 + .../templates/persistentvolume.yaml | 57 + .../templates/podsecuritypolicy.yaml | 24 + .../newrelic-logging/templates/secret.yaml | 12 + .../templates/serviceaccount.yaml | 17 + .../tests/cri_parser_test.yaml | 37 + .../tests/dns_config_test.yaml | 62 + .../tests/endpoint_region_selection_test.yaml | 128 + .../fluentbit_k8logging_exclude_test.yaml | 45 + .../tests/fluentbit_persistence_test.yaml | 317 + .../tests/fluentbit_pod_label_test.yaml | 48 + .../tests/fluentbit_sendmetrics_test.yaml | 74 + .../tests/host_network_test.yaml | 46 + .../newrelic-logging/tests/images_test.yaml | 96 + .../tests/linux_volume_mount_test.yaml | 37 + .../newrelic-logging/tests/rbac_test.yaml | 48 + .../charts/newrelic-logging/values.yaml | 361 + .../5.0.94/charts/newrelic-pixie/Chart.yaml | 18 + .../5.0.94/charts/newrelic-pixie/README.md | 166 + .../charts/newrelic-pixie/ci/test-values.yaml | 5 + .../charts/newrelic-pixie/templates/NOTES.txt | 27 + .../newrelic-pixie/templates/_helpers.tpl | 172 + .../newrelic-pixie/templates/configmap.yaml | 12 + .../charts/newrelic-pixie/templates/job.yaml | 164 + .../newrelic-pixie/templates/secret.yaml | 20 + .../newrelic-pixie/tests/configmap.yaml | 44 + .../charts/newrelic-pixie/tests/jobs.yaml | 138 + .../5.0.94/charts/newrelic-pixie/values.yaml | 70 + .../newrelic-prometheus-agent/.helmignore | 23 + .../newrelic-prometheus-agent/Chart.lock | 6 + .../newrelic-prometheus-agent/Chart.yaml | 22 + .../newrelic-prometheus-agent/README.md | 244 + .../README.md.gotmpl | 209 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-values.yaml | 6 + .../static/lowdatamodedefaults.yaml | 6 + .../static/metrictyperelabeldefaults.yaml | 17 + .../templates/_helpers.tpl | 165 + .../templates/clusterrole.yaml | 24 + .../templates/clusterrolebinding.yaml | 16 + .../templates/configmap.yaml | 31 + .../templates/secret.yaml | 2 + .../templates/serviceaccount.yaml | 13 + .../templates/statefulset.yaml | 157 + .../tests/configmap_test.yaml | 572 ++ .../tests/configurator_image_test.yaml | 57 + .../tests/integration_filters_test.yaml | 119 + .../tests/lowdatamode_configmap_test.yaml | 138 + .../newrelic-prometheus-agent/values.yaml | 473 + .../5.0.94/charts/nri-kube-events/Chart.lock | 6 + .../5.0.94/charts/nri-kube-events/Chart.yaml | 26 + .../5.0.94/charts/nri-kube-events/README.md | 79 + .../charts/nri-kube-events/README.md.gotmpl | 43 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-bare-minimum-values.yaml | 3 + .../ci/test-custom-attributes-as-map.yaml | 12 + .../ci/test-custom-attributes-as-string.yaml | 11 + .../nri-kube-events/ci/test-values.yaml | 60 + .../nri-kube-events/templates/NOTES.txt | 3 + .../nri-kube-events/templates/_helpers.tpl | 45 + .../templates/_helpers_compatibility.tpl | 262 + .../templates/agent-configmap.yaml | 12 + .../templates/clusterrole.yaml | 42 + .../templates/clusterrolebinding.yaml | 16 + .../nri-kube-events/templates/configmap.yaml | 23 + .../nri-kube-events/templates/deployment.yaml | 124 + .../nri-kube-events/templates/secret.yaml | 2 + .../templates/serviceaccount.yaml | 11 + .../tests/agent_configmap_test.yaml | 46 + .../nri-kube-events/tests/configmap_test.yaml | 139 + .../tests/deployment_test.yaml | 104 + .../nri-kube-events/tests/images_test.yaml | 168 + .../tests/security_context_test.yaml | 77 + .../5.0.94/charts/nri-kube-events/values.yaml | 135 + .../charts/nri-metadata-injection/.helmignore | 1 + .../charts/nri-metadata-injection/Chart.lock | 6 + .../charts/nri-metadata-injection/Chart.yaml | 25 + .../charts/nri-metadata-injection/README.md | 68 + .../nri-metadata-injection/README.md.gotmpl | 41 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-values.yaml | 5 + .../templates/NOTES.txt | 23 + .../templates/_helpers.tpl | 72 + .../job-patch/clusterrole.yaml | 27 + .../job-patch/clusterrolebinding.yaml | 20 + .../job-patch/job-createSecret.yaml | 61 + .../job-patch/job-patchWebhook.yaml | 61 + .../admission-webhooks/job-patch/psp.yaml | 50 + .../admission-webhooks/job-patch/role.yaml | 21 + .../job-patch/rolebinding.yaml | 21 + .../job-patch/serviceaccount.yaml | 14 + .../mutatingWebhookConfiguration.yaml | 36 + .../templates/cert-manager.yaml | 53 + .../templates/deployment.yaml | 85 + .../templates/service.yaml | 13 + .../tests/cluster_test.yaml | 39 + .../tests/job_serviceaccount_test.yaml | 59 + .../tests/rbac_test.yaml | 38 + .../tests/volume_mounts_test.yaml | 30 + .../charts/nri-metadata-injection/values.yaml | 102 + .../5.0.94/charts/nri-prometheus/.helmignore | 22 + .../5.0.94/charts/nri-prometheus/Chart.lock | 6 + .../5.0.94/charts/nri-prometheus/Chart.yaml | 29 + .../5.0.94/charts/nri-prometheus/README.md | 116 + .../charts/nri-prometheus/README.md.gotmpl | 83 + .../charts/common-library/.helmignore | 23 + .../charts/common-library/Chart.yaml | 17 + .../charts/common-library/DEVELOPERS.md | 747 ++ .../charts/common-library/README.md | 106 + .../common-library/templates/_affinity.tpl | 10 + .../templates/_agent-config.tpl | 26 + .../common-library/templates/_cluster.tpl | 15 + .../templates/_custom-attributes.tpl | 17 + .../common-library/templates/_dnsconfig.tpl | 10 + .../common-library/templates/_fedramp.tpl | 25 + .../common-library/templates/_hostnetwork.tpl | 39 + .../common-library/templates/_images.tpl | 94 + .../common-library/templates/_insights.tpl | 56 + .../templates/_insights_secret.yaml.tpl | 21 + .../common-library/templates/_labels.tpl | 54 + .../common-library/templates/_license.tpl | 68 + .../templates/_license_secret.yaml.tpl | 21 + .../templates/_low-data-mode.tpl | 26 + .../common-library/templates/_naming.tpl | 73 + .../templates/_nodeselector.tpl | 10 + .../templates/_priority-class-name.tpl | 10 + .../common-library/templates/_privileged.tpl | 28 + .../common-library/templates/_proxy.tpl | 10 + .../common-library/templates/_region.tpl | 74 + .../templates/_security-context.tpl | 23 + .../templates/_serviceaccount.tpl | 90 + .../common-library/templates/_staging.tpl | 39 + .../common-library/templates/_tolerations.tpl | 10 + .../common-library/templates/_userkey.tpl | 56 + .../templates/_userkey_secret.yaml.tpl | 21 + .../common-library/templates/_verbose-log.tpl | 54 + .../charts/common-library/values.yaml | 1 + .../ci/test-lowdatamode-values.yaml | 9 + .../ci/test-override-global-lowdatamode.yaml | 10 + .../charts/nri-prometheus/ci/test-values.yaml | 104 + .../static/lowdatamodedefaults.yaml | 10 + .../nri-prometheus/templates/_helpers.tpl | 15 + .../nri-prometheus/templates/clusterrole.yaml | 23 + .../templates/clusterrolebinding.yaml | 16 + .../nri-prometheus/templates/configmap.yaml | 21 + .../nri-prometheus/templates/deployment.yaml | 98 + .../nri-prometheus/templates/secret.yaml | 2 + .../templates/serviceaccount.yaml | 13 + .../nri-prometheus/tests/configmap_test.yaml | 86 + .../nri-prometheus/tests/deployment_test.yaml | 82 + .../nri-prometheus/tests/labels_test.yaml | 32 + .../5.0.94/charts/nri-prometheus/values.yaml | 251 + .../charts/pixie-operator-chart/Chart.yaml | 4 + .../pixie-operator-chart/crds/olm_crd.yaml | 9045 +++++++++++++++++ .../pixie-operator-chart/crds/vizier_crd.yaml | 347 + .../templates/00_olm.yaml | 232 + .../templates/01_px_olm.yaml | 13 + .../templates/02_catalog.yaml | 37 + .../templates/03_subscription.yaml | 11 + .../templates/04_vizier.yaml | 100 + .../templates/deleter.yaml | 25 + .../templates/deleter_role.yaml | 77 + .../charts/pixie-operator-chart/values.yaml | 75 + .../nri-bundle/5.0.94/ci/test-values.yaml | 21 + .../nri-bundle/5.0.94/questions.yaml | 113 + .../new-relic/nri-bundle/5.0.94/values.yaml | 169 + .../stackstate-k8s-agent/1.0.98/.helmignore | 26 + .../stackstate-k8s-agent/1.0.98/Chart.lock | 6 + .../stackstate-k8s-agent/1.0.98/Chart.yaml | 25 + .../stackstate-k8s-agent/1.0.98/README.md | 263 + .../1.0.98/README.md.gotmpl | 45 + .../stackstate-k8s-agent/1.0.98/Releasing.md | 15 + .../stackstate-k8s-agent/1.0.98/app-readme.md | 5 + .../charts/http-header-injector/.helmignore | 25 + .../charts/http-header-injector/Chart.yaml | 15 + .../charts/http-header-injector/README.md | 56 + .../http-header-injector/Readme.md.gotpl | 26 + .../templates/_defines.tpl | 131 + .../cert-hook-clusterrolbinding.yaml | 24 + .../templates/cert-hook-clusterrole.yaml | 26 + .../templates/cert-hook-config.yaml | 158 + .../templates/cert-hook-job-delete.yaml | 39 + .../templates/cert-hook-job-setup.yaml | 39 + .../templates/cert-hook-serviceaccount.yaml | 18 + .../templates/pull-secret.yaml | 32 + .../templates/webhook-cert-secret.yaml | 18 + .../templates/webhook-certificate.yaml | 23 + .../templates/webhook-config.yaml | 128 + .../templates/webhook-deployment.yaml | 61 + .../webhook-mutatingwebhookconfiguration.yaml | 54 + .../templates/webhook-service.yaml | 20 + .../charts/http-header-injector/values.yaml | 110 + .../stackstate-k8s-agent/1.0.98/questions.yml | 184 + .../_cluster-agent-kube-state-metrics.yaml | 62 + .../1.0.98/templates/_container-agent.yaml | 191 + .../templates/_container-process-agent.yaml | 160 + .../1.0.98/templates/_helpers.tpl | 219 + .../checks-agent-clusterrolebinding.yaml | 21 + .../templates/checks-agent-configmap.yaml | 17 + .../templates/checks-agent-deployment.yaml | 185 + .../checks-agent-poddisruptionbudget.yaml | 23 + .../checks-agent-serviceaccount.yaml | 16 + .../templates/cluster-agent-clusterrole.yaml | 152 + .../cluster-agent-clusterrolebinding.yaml | 19 + .../templates/cluster-agent-configmap.yaml | 31 + .../templates/cluster-agent-deployment.yaml | 169 + .../cluster-agent-poddisruptionbudget.yaml | 21 + .../1.0.98/templates/cluster-agent-role.yaml | 21 + .../templates/cluster-agent-rolebinding.yaml | 18 + .../templates/cluster-agent-service.yaml | 21 + .../cluster-agent-serviceaccount.yaml | 14 + .../templates/logs-agent-clusterrole.yaml | 23 + .../logs-agent-clusterrolebinding.yaml | 21 + .../templates/logs-agent-configmap.yaml | 63 + .../templates/logs-agent-daemonset.yaml | 91 + .../templates/logs-agent-serviceaccount.yaml | 16 + .../templates/node-agent-clusterrole.yaml | 21 + .../node-agent-clusterrolebinding.yaml | 19 + .../templates/node-agent-configmap.yaml | 17 + .../templates/node-agent-daemonset.yaml | 110 + .../templates/node-agent-podautoscaler.yaml | 39 + .../1.0.98/templates/node-agent-scc.yaml | 60 + .../1.0.98/templates/node-agent-service.yaml | 28 + .../templates/node-agent-serviceaccount.yaml | 14 + .../templates/openshift-logging-secret.yaml | 22 + .../1.0.98/templates/pull-secret.yaml | 39 + .../1.0.98/templates/secret.yaml | 27 + .../test/clusteragent_resources_test.go | 145 + .../1.0.98/test/clustername_test.go | 54 + .../values/clustercheck_ksm_custom_url.yaml | 7 + .../values/clustercheck_ksm_no_override.yaml | 5 + .../values/clustercheck_ksm_override.yaml | 26 + .../clustercheck_no_ksm_custom_url.yaml | 7 + .../clustercheck_service_port_override.yaml | 4 + .../test/values/disable-all-resource.yaml | 17 + .../test/values/http-header-injector.yaml | 8 + .../1.0.98/test/values/minimal.yaml | 7 + .../1.0.98/values.schema.json | 78 + .../stackstate-k8s-agent/1.0.98/values.yaml | 616 ++ index.yaml | 217 +- 1144 files changed, 127962 insertions(+), 1 deletion(-) create mode 100644 assets/jfrog/artifactory-ha-107.90.14.tgz create mode 100644 assets/jfrog/artifactory-jcr-107.90.14.tgz create mode 100644 assets/kasten/k10-7.0.1101.tgz create mode 100644 assets/new-relic/nri-bundle-5.0.94.tgz create mode 100644 assets/stackstate/stackstate-k8s-agent-1.0.98.tgz create mode 100644 charts/jfrog/artifactory-ha/107.90.14/.helmignore create mode 100644 charts/jfrog/artifactory-ha/107.90.14/CHANGELOG.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/Chart.lock create mode 100644 charts/jfrog/artifactory-ha/107.90.14/Chart.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/LICENSE create mode 100644 charts/jfrog/artifactory-ha/107.90.14/README.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/app-readme.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/.helmignore create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.lock create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/README.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/.helmignore create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/Chart.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/README.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_affinities.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_capabilities.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_errors.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_images.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_ingress.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_labels.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_names.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_secrets.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_storage.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_tplvalues.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_utils.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_warnings.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_cassandra.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mariadb.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mongodb.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_postgresql.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_redis.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_validations.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/commonAnnotations.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/default-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/shmvolume-disabled-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/README.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/conf.d/README.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/docker-entrypoint-initdb.d/README.md create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/NOTES.txt create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/_helpers.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/configmap.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extended-config-configmap.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extra-list.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/initialization-configmap.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-configmap.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-svc.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/networkpolicy.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/podsecuritypolicy.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/prometheusrule.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/role.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/rolebinding.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/secrets.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/serviceaccount.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/servicemonitor.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset-readreplicas.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-headless.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-read.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.schema.json create mode 100644 charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/access-tls-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/default-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/global-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/large-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/loggers-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/medium-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/migration-disabled-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/nginx-autoreload-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-access-tls-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/small-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/ci/test-values.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/binarystore.xml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/installer-info.json create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/migrate.sh create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/migrationHelmInfo.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/migrationStatus.sh create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/nginx-artifactory-conf.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/nginx-main-conf.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/files/system.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/logo/artifactory-logo.png create mode 100644 charts/jfrog/artifactory-ha/107.90.14/questions.yml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge-extra-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large-extra-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium-extra-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small-extra-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge-extra-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall-extra-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/NOTES.txt create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/_helpers.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/_system-yaml-render.tpl create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/additional-resources.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/admin-bootstrap-creds.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-access-config.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-binarystore-secret.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-configmaps.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-custom-secrets.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-database-secrets.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-gcp-credentials-secret.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-installer-info.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-license-secret.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-migration-scripts.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-networkpolicy.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-nfs-pvc.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-node-pdb.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-node-statefulset.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-primary-pdb.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-primary-service.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-primary-statefulset.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-priority-class.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-role.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-rolebinding.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-secrets.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-service.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-serviceaccount.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-storage-pvc.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-system-yaml.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-unified-secret.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/filebeat-configmap.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/ingress.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/logger-configmap.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-artifactory-conf.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-certificate-secret.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-conf.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-deployment.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-pdb.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-pvc.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-scripts-conf.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/templates/nginx-service.yaml create mode 100644 charts/jfrog/artifactory-ha/107.90.14/values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/CHANGELOG.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/Chart.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/LICENSE create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/app-readme.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/.helmignore create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/CHANGELOG.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.lock create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/LICENSE create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/.helmignore create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.lock create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/.helmignore create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/Chart.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_affinities.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_capabilities.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_errors.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_images.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_ingress.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_labels.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_names.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_secrets.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_storage.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_tplvalues.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_utils.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_warnings.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_cassandra.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mariadb.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mongodb.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_postgresql.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_redis.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_validations.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/commonAnnotations.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/default-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/shmvolume-disabled-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/conf.d/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/docker-entrypoint-initdb.d/README.md create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/NOTES.txt create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/_helpers.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/configmap.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extended-config-configmap.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extra-list.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/initialization-configmap.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-configmap.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-svc.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/networkpolicy.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/podsecuritypolicy.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/prometheusrule.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/role.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/rolebinding.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/secrets.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/serviceaccount.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/servicemonitor.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset-readreplicas.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-headless.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-read.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.schema.json create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/access-tls-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/default-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/derby-test-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/global-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/large-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/loggers-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/medium-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/migration-disabled-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/nginx-autoreload-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values-access-tls-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/small-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/test-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/binarystore.xml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/installer-info.json create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/migrate.sh create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/migrationHelmInfo.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/migrationStatus.sh create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/nginx-artifactory-conf.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/nginx-main-conf.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/system.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/logo/artifactory-logo.png create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge-extra-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large-extra-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium-extra-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small-extra-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge-extra-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall-extra-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/NOTES.txt create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_helpers.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_system-yaml-render.tpl create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/additional-resources.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/admin-bootstrap-creds.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-access-config.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-binarystore-secret.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-configmaps.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-custom-secrets.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-database-secrets.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-gcp-credentials-secret.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-hpa.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-installer-info.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-license-secret.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-migration-scripts.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-networkpolicy.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-nfs-pvc.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-pdb.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-priority-class.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-role.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-rolebinding.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-secrets.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-service.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-serviceaccount.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-statefulset.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-system-yaml.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-unified-secret.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/filebeat-configmap.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/ingress.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/logger-configmap.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-artifactory-conf.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-certificate-secret.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-conf.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-deployment.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-pdb.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-pvc.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-scripts-conf.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-service.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/ci/default-values.yaml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/logo/jcr-logo.png create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/questions.yml create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/templates/NOTES.txt create mode 100644 charts/jfrog/artifactory-jcr/107.90.14/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/Chart.lock create mode 100644 charts/kasten/k10/7.0.1101/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/README.md create mode 100644 charts/kasten/k10/7.0.1101/app-readme.md create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/.helmignore create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/README.md create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/default-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-affinity-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-dashboard-json-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-dashboard-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-extraconfigmapmounts-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-image-renderer-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-nondefault-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-persistence.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/dashboards/custom-dashboard.json create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/_config.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/_pod.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/configSecret.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/configmap-dashboard-provider.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/dashboards-json-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/deployment.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/headless-service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/hpa.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-deployment.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-hpa.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-network-policy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/poddisruptionbudget.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/podsecuritypolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/pvc.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/role.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/secret-env.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/secret.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/statefulset.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-podsecuritypolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-role.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/grafana/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/.helmignore create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/OWNERS create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/README.md create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/.helmignore create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/README.md create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/services.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/statefulset.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.schema.json create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/.helmignore create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/README.md create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/role.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/.helmignore create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/README.md create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/.helmignore create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/README.md create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/clusterrole.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/clusterrolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/cm.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/deploy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/extra-manifests.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/headless-svc.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/network-policy.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/pdb.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/psp.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/pvc.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/rolebinding.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/service.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/sts.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/templates/vpa.yaml create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/values.schema.json create mode 100644 charts/kasten/k10/7.0.1101/charts/prometheus/values.yaml create mode 100644 charts/kasten/k10/7.0.1101/config.json create mode 100644 charts/kasten/k10/7.0.1101/eula.txt create mode 100644 charts/kasten/k10/7.0.1101/files/favicon.png create mode 100644 charts/kasten/k10/7.0.1101/files/kasten-logo.svg create mode 100644 charts/kasten/k10/7.0.1101/files/styles.css create mode 100644 charts/kasten/k10/7.0.1101/grafana/dashboards/default/default.json create mode 100644 charts/kasten/k10/7.0.1101/license create mode 100644 charts/kasten/k10/7.0.1101/questions.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/NOTES.txt create mode 100644 charts/kasten/k10/7.0.1101/templates/_definitions.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_grafana.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_helpers.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_k10_container.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_k10_image_tag.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_k10_metering.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_k10_serviceimage.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_k10_template.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/_prometheus.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/aggregatedaudit-policy.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/api-tls-secrets.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/apiservice.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/daemonsets.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/deployments.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/fluentbit-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/frontend-nginx-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/gateway-ext.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/gateway.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/grafana-scc.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/ingress.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/k10-config.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/k10-eula.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/k10-scc.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/kopia-tls-certs.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/license.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/mc.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/mutatingwebhook.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/networkpolicy.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/ocp-ca-cert-extract-hook.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/prometheus-configmap.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/prometheus-scc.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/prometheus-service.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/rbac.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/rhmarketplace.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/route.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/secrets.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/secure_deployment.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/serviceaccount.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/v0services.yaml create mode 100644 charts/kasten/k10/7.0.1101/templates/workloadIdentityFederation.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/{values}/grafana/values/grafana_values.tpl create mode 100644 charts/kasten/k10/7.0.1101/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl create mode 100644 charts/kasten/k10/7.0.1101/triallicense create mode 100644 charts/kasten/k10/7.0.1101/values.schema.json create mode 100644 charts/kasten/k10/7.0.1101/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/app-readme.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/certmanager.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/instrumentation-crd.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/leader-election-rbac.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/manager-rbac.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/newrelic_license_secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/proxy-rbac.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/reader-rbac.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/service.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-configuration.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-service.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/cert_manager_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/webhook_ssl_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/crs-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/extra-manifests.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/kubeconfig-secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/networkpolicy.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/pdb.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/podsecuritypolicy.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/psp-clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/rbac-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/role.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/service.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/servicemonitor.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/stsdiscovery-role.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-createSecret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-patchWebhook.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/psp.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/role.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/mutatingWebhookConfiguration.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/cert-manager.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/service.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/deployment_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_patch_psp_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_serviceaccount_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/rbac_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-cplane-kind-deployment-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers_compatibility.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_affinity_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_agent-config_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_host_network.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_rbac.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_tolerations_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/agent-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/daemonset.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/scraper-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_affinity_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_agent-config_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_host_network.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_tolerations_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/agent-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/scraper-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_affinity_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_agent-config_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_host_network.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_security_context_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_tolerations_helper.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/agent-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/daemonset.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/integrations-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/scraper-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/podsecuritypolicy.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/apiservice.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-createSecret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-patchAPIService.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/psp.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/role.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/service.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/apiservice_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/common_extra_naming_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/configmap_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/deployment_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/hpa_clusterrolebinding_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_cluster_rolebinding_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_clusterrole_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_common_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_createsecret_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_patchapiservice_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_serviceaccount_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/rbac_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-enable-windows-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-lowdatamode-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-override-global-lowdatamode.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-staging-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-global.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/fluent-bit-and-plugin-metrics-dashboard-template.json create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset-windows.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/persistentvolume.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/podsecuritypolicy.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/cri_parser_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/dns_config_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/endpoint_region_selection_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_k8logging_exclude_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_persistence_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_pod_label_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_sendmetrics_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/host_network_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/images_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/linux_volume_mount_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/rbac_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/job.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/jobs.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/lowdatamodedefaults.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/metrictyperelabeldefaults.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/statefulset.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configmap_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configurator_image_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/integration_filters_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/lowdatamode_configmap_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-bare-minimum-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-map.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-string.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers_compatibility.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/agent-configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/agent_configmap_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/configmap_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/deployment_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/images_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/security_context_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/NOTES.txt create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-createSecret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-patchWebhook.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/psp.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/role.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/rolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/mutatingWebhookConfiguration.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/cert-manager.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/service.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/cluster_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/job_serviceaccount_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/rbac_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/volume_mounts_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.lock create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md.gotmpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/.helmignore create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/DEVELOPERS.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/README.md create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_affinity.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_agent-config.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_cluster.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_custom-attributes.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_dnsconfig.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_fedramp.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_hostnetwork.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_images.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_labels.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_low-data-mode.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_naming.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_nodeselector.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_priority-class-name.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_privileged.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_proxy.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_region.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_security-context.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_serviceaccount.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_staging.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_tolerations.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey_secret.yaml.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_verbose-log.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-lowdatamode-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-override-global-lowdatamode.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/static/lowdatamodedefaults.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/_helpers.tpl create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrole.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrolebinding.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/configmap.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/deployment.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/secret.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/serviceaccount.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/configmap_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/deployment_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/labels_test.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/Chart.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/olm_crd.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/vizier_crd.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/00_olm.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/01_px_olm.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/02_catalog.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/03_subscription.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/04_vizier.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter_role.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/ci/test-values.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/questions.yaml create mode 100644 charts/new-relic/nri-bundle/5.0.94/values.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/.helmignore create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.lock create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/README.md create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/README.md.gotmpl create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/Releasing.md create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/app-readme.md create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/.helmignore create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Chart.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/README.md create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Readme.md.gotpl create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/_defines.tpl create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrolbinding.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrole.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-config.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-delete.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-setup.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-serviceaccount.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/pull-secret.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-cert-secret.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-certificate.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-config.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-deployment.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-mutatingwebhookconfiguration.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-service.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/values.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/questions.yml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_cluster-agent-kube-state-metrics.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-agent.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-process-agent.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_helpers.tpl create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-clusterrolebinding.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-configmap.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-deployment.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-poddisruptionbudget.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-serviceaccount.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrole.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrolebinding.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-configmap.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-deployment.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-poddisruptionbudget.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-role.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-rolebinding.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-service.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-serviceaccount.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrole.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrolebinding.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-configmap.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-daemonset.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-serviceaccount.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrole.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrolebinding.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-configmap.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-daemonset.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-podautoscaler.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-scc.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-service.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-serviceaccount.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/openshift-logging-secret.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/pull-secret.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/templates/secret.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/clusteragent_resources_test.go create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/clustername_test.go create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_custom_url.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_no_override.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_override.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_no_ksm_custom_url.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_service_port_override.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/disable-all-resource.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/http-header-injector.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/minimal.yaml create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/values.schema.json create mode 100644 charts/stackstate/stackstate-k8s-agent/1.0.98/values.yaml diff --git a/assets/jfrog/artifactory-ha-107.90.14.tgz b/assets/jfrog/artifactory-ha-107.90.14.tgz new file mode 100644 index 0000000000000000000000000000000000000000..2ce24d332685a816d87f8d237bda01dbfcba6625 GIT binary patch literal 218393 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Ð}kNowNMrHpLcJE`8q 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 z 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�X3ivE60{o< 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�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 ZNo~Khx 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
vCk0Fgd@YM{H6^Ml)L3T#ABG@oq`z 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

5KxV{+O3J-g9D9 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�q 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

WW~rKr4&pR1Fbfw`D{|%I7chhC2@BO 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

lTR}WLb8T`746ePn|uH!%zJ0?7phF{8$?bYXS?d z7kZa6oen$Huog2|#{bzg-Z{N}S%!Y1B(3a(J{eaWJ#QZS#3Es0caHklf5qoxiDQ&L z4MpvwvqDs@AJ##m@FOGDEjI}2CGRk9-I;TO^kozCB|oQq^$m}EoVq3#X8#R~{@=Id z8}>?NF>ZLN>qq)#1lQ?{v1zw9a1z7sV0oTdZkrmQ3wtOj7Tm!17%mS@R9hV&6GJ(X zgoQ!vUO0kE3}w=tm;!H`Cx?6G*CXe1)8uBoNS67{Ye`yNWp~NDb|O}&(J*dyjXpF6 zU6OnW)AoQ5Fm7$=X^h*^+k-OkDJ>##W5$z+aEWysPY?%I%t!gDQBmbtJ#Q?}rddd= z4Y+5$Rp40z2Ak^%&ZkP%Yq7_)r-Gny;NjSXVNt>C%hPGw;okM}X=O zD?XpcrdW~wt3|-ye1T;nkd=4tk4qG2D%bZMnd=}kT8@r=3OVV=kH_i>=?pdJvZ9)m zB2G=pxuFs*&$Seadd_xIKcDFLS0oz;i65y`EG+D{-udxO>d!96U|cGIisrewwgmmJ zZ4BbKSlCHba7v~WPLG)*)>wFFsXVd{y0~-CU~v$JrKwuHe0+$`V|%fyXx#rkB!WPO z9(znV-zQA8CMX#km(e}V2r#SFhZM11ev>3V&N6aY*PSE7D40<)qSB&V-}H%V63Q3U zqWm{9`tTW*HY0}|*`tahk!8-aP_2w3F<8n?HzH&olTg0S@h71?Pd!Is5q%pH8trok zHD79exNJEtd`h3GCZTcxTSTwLgE4a0XtvJ>!BhT9sQl6HPw%K-ezeq%d@sL{!;tuS ztaS*L(L;J^`EdN4YK>m-<5L7dx zY7IsDpCIE+fb_-Cj7q-+1cOw@YeXpj;X|i{KJjFb)_?ryg3kx}&*c9)|CxOB(f*&wF#l(= zJbFk7Z8OuTFpPpmDy_)uq$TC3Pu2-NcVZqHK78_rs*UwsME~@OL!(dXwq3r(@eo5{(r5XJz#l;>o@Y#X3?1z&q z$tcaSOBReZH{%5L@*s<5IuCM+tl8Y>O8k0;4lLcD+sMlJ%No$Xevx8vu+JLkSt|u+ z8p;Qk{Xj2+wTW;Ztws2Pedqs?WRBYR|JUpYg}r|$8(yxSMDV0BE4m)lPpqryj*-Pc>6k%5b^eOHIncX<;Nc%`>v{DqX|@uEVD6nSaW_@! zAVye)MRyMUP*Ar_4eZ>qSSEL>tLerZI zwJD2P-BSWHgtOAT!`A#ufz%=lqpnL(O8#fb!Jm>7YqgP3=T>If(;CjY!v)SglpKroJ#Ye4XUgpKR1;b!x~2G>&b}N_ti_ z>@X!bu$#7I+P7v}w~cM}d>^V1SgKI4q&{kz<}saYG-b8u%XI9|EOutK8qBmG8f!M6 z1iDAMSTr#*tNk56b@oBPYL$wbxfJIc2>NXmXxnDbGw)H(Cf`z&MCK*NV8)?GQbeki z!^yjoed63+=DUSD2>5;2oPOGze!-{qoFyD_a=K;s4)_h7K2(URUBc)rp!D_Yjj{Zl z>Z@`|hl6UtVBTf_#lsYTc<;vPc6zMT!3Tbp%IoILqaX(&eV0d6jWCfOA7eUzex7_I zIucrU_%Za;*Lau_2Ol9IU5A<;&b*vE;4qrHj4w!^b-ML6h>{Qe)=PNM%7D%m3ZsgKU8T8~yYl*GDsVZZhJA0Lsy zF=1TReZ`mCtI0mGQk;pPLl$FFgCX0T_FD(C&Y|#T6#V-@h0#gL& zD*2e`6pVd4JTSXvSZFwfJG>~7e}0rD+v@(Ef&MRYa-zOCDz0{Tp3{R|!1e>@rrP_-auk4)mOfSOLpvPPNWraKyxIaYJQTYdi6>7*&q-0c9e8H5tFW`FPN$U+V3!i4lpaMd zJm1I+dcqt>xnT#DAZ^JTmOn_Q1LJx1KB*~fD>@}a$T+LJ2)hr~t&6?z{GRE|W@)sY z(N`8X`RwBhMHJALD5|FjtV@79xbq41jUP#uGqf5;3?4JJ8@=$+eEa_WjZ`4e*LC}Q z@|(X#U<^xl&j5G5kM+;n){JNz=zxn5IvM|B|9`U09C|Yoqs73~p&X1bfwEF%E<2Qm zoxz=k@m|^@^3K$trP)FDk@a5-L{V}}k!Jjl$Hkj#%)>;PXg`SrfL3X@X*)*@S|;y; zyXCuXN4cRL%XO~`^qgLJ^YF$4zoqc~-z5Sup?2&S=LxM5h zcIxenvcNjm!r_DN#HqjaCtOE<5Vqz?ubGfOYcU{qakTKWLkr=V?Ycj|YZTLV&my+8 zs^ma3Zj-J#@XW`crkIp+$0I0$|Jcr?X?;a^PwJ-#7Zw%%o30M>M$*q56_!YL-d@k^ zE?lt|DPih&Vb&tgU=Un|Kk@aVKRRkT$a5*3!z)msGZ!UYgU(W8j8ntj#_HXBDY3)e zqa^20+phi}h-vaP>F){WVoBnqcKWW)2!=qujJDO3nqz`wlj^(nav#f9-$d{sTn-9~ zFTr6Ofonq-5qjh^d1MA#q<~I;u`kgLeq9R?N@`)%Wt+Z(F!%N^(NJ@QA$D1|Pz_HF z+b;Iq+Q%oTl8yeFi%PU3b>Hk9Rodlm_{YOR-Y`YLKEJi8O=JFnm!~z$jHl34Dax`< z5YL9TiR++e*i0WHmt#w%813&h*Ow{xFpj{)`m@Qu-1nf>m!(y!DkAD&hRP|pAzeWX zI*hC9%h=vzjHt~obG&J8wPxWf&z|D^7Q}SB)gOU8*9Ymd+_C$fB>e*|a(LV^Jd-Lo zplU zv71a8?G+q1H*E@PIexh)uXd_-YAnw}L{rP&aspH}mzSKbeQfaG*mSB|wz_TetM6dD znW!4Pnk-w7RZVijOWcEwU#6nacOu9$g054l3I9Ab%l+`<+{wBuAfYi|J1oN45mEJ5&iLe0aJmNG{3sYlm5T)AQzy)9|4F#HgLZZ%z@)So z)`8&C(n&+{{TKts0fKb}F40Qget`bfvf8-zPqX2Bd z2mKlM5_9w;3FvU7dG?uhC5Jl6{d;N^K;3l9+m@TpieDNE-yUAPhFpcv|Y37;MBZ&6oe)HV}yYy>Z~FG*B?~* z7zw&atV6CQNIRy{4NOD!5(9DJFbvC-eR#5g<@`y&>ciO3hKkaUzj=G$eYp>mv7l@h z4Bb$a2j_9GRr6-<9{|ozg5)v^FB<-Uj0jKJF13B?2W&m);~rH+w+)v8;4|c(-(zD_ zQjv={vi;M#aiSRp#e0mOrt&(GVSmjbFU5~|6?&X)Rwb!cs zNB4UPR((;-^oa?!T#J7KQ;RoD3i z!Tyk!``oIIz++`|W`hucJibGVRKtb88-X&yoFYj=+WUgIjx>p?H3S8AJ|qqDvfMpZ zA@+V|vMdI9%zQF73VdcPZqlux4yx5i`DP@~?d2LZ>%x?qif~ic@&PkxZLajQ?jXG^ z>Oi?Ou`NxrHc_-4Pr1<;oe{}LuR7moLM|YZ#^+UAwdn1sVDS>L=^mv13K^S;j+@&s z$rarKyXjZ!1e_)Zr)*Wm8T{2<+guIpZZu$pGtYCP+A0=-<_z2EP!r9dtnj3_R;`T*7a*Uu})ldo4xUVxPvitL*} zZd4h+3oGk=c(ScOjjPW`OS*oKAZw_KfY+|T8NZ<0S}*J%;Wd&*hWpu_i%rYdhxeu7 zX+IzaAnS6S4^Ta|l^yutxl-e|*0zbX`SPJDbb6K7|G|8Bv(D>Ps5%)ei?AB-Ar15+ z+sqB}ZV#DW1;5rm9DxsQcN@P0?@*0NK3uRF^YY<dnB-O~-Drp!c^o=S%ma?4go>6l6XO9-ecnXAfrpzA5(pwff%>FbiR);F|sA_VP|U zc=TRn4Zxky)gstQrmD)f8nu7_yw)Fg1iCe@UpkB1olkqFz5Rxe!hhj^AuP6Xcb221>ldVpF_JY8dbl_a0Gz`y+|Q4$jP6)9 zcX^#&1azDFO(HggnP+1}N1G##_g{=UBY{mr1!ldR@;?_2j}vt{7y`$z8;SOYINzwk|qb(DLYmHnjM(!|)WNXFkFTJ2GB4Mb_Mm_n`^{*Gnpf7c9Pj{w z0L)sB@?-f&1bn3sb3eU!x_hu0iF^gX{@vV@yVKdt*|^H74YGA!FEgUZ30VI?^%M~+ zY0N6z(Ax_$wsv2zACl`_I@~D=^jYvzT5CSq>)*U;3h;Ejyv^INH@<{*+(Mfg)PFkN zlYJc%@-sej$`9zzvF^+6Tr z_T zqbDomVx8+GUZTMGE#P|i#nUtMqr3Y_lVGph%bpw`BB1SWf=0DY`-xNd-+ryj{?S}L zA+c-GhR1b5bYn8F$JNHop@cOG`-|It18>(GtIOw|$nFiH&h_)yjkVkP&ai~e+qL?Q zS!VWs?V>L;u~VYiU4p{)CXWV=koU-8ArfQf1))w*?s=S8yiinBr{3N?8laH-!l7I2 zM{`HYZPRjdp0G%68Pd+Wz;3dCh zisgikZF974qS*^1zsu-hZd5KEkO+9#*OL1hZSS%*FF8C8n1_kCKB($+sS#o z62RNT=*ni~TAR;2ct+r2S@p}*xBLC8g)N~C5wWYOsgvQ+1jlMVLD2a}Y38Tit@nSo zY|{XHu?tia@LD!PD0q**4wl*Tz@^t1;dgmKx;N)B#qv$NQ@~A3zw*G9J&*AXr{UPZpc(t|hg+Jn*F4@i z81WE?!AiAnRB)H=!DWl#EN^6gK!rN=DSX%h745FpFC%<0`YK(JC8fjC zQF$Mh{~rKwK##vP9-)XxOAqWrsW$7rKY-!tV=hL(YT2`F1-Zl#`#y4gmUvtbl^ThA$>e67CjxP2Qr(@A68c!3_o0$f-ElM^SfIiYPs9LmG)R2k;URJ3 zC>bM{1)TDj1ny+r3`b`8))7u(Hpa0WXg&C1l=Liuw$Fj_F|(=04TbEQM1#af*QDkTl61|E`LXeuV?ylSPDl@-)H>*i{d9CW1==?y{sBNFopzUEEWC7iz_ zlYOs*kagqGKxF|1!+Uf|CTM>btw!XRghs@JBSXCRNa(YPIKI}(C4}NhNE#@_oL{p@ zK!lrz1{9S|p?3hPgph`n-o}D^jA}T8et9PTUHEExv8WHS=EM3So3y&~Sbf>Y;|}x5 zbRRlxaa;+?DxmE*4^hN?f-vXFn0Tl^fukm7C?t`f$p{A%C~D5FO4=!LTn0m{OU2M1 z)G2(;h%AkpX`NOw*|V)5hcJnd^qcr2|U18Q#`jSFu{?N>qb;{n@<`#bqN1T0yFr zdA+%;*8udQUI{|+nz5lz&<=0|OHhmnjrrIAMb>)jFS6hpChy|JU(q02g z;?sbn9dEP1QC&iU5!v2CBygEWJoI|!7C789Q3s;D+w+D2=s(r1IFC2>ItzZI| zp_-y#j|4b~i-;zGiQ#nt$%MCs#3n!WJq=?1PMpSi7J(5ZcYI3bplyaV4AK7XJVKZ) z`S>VGBD(~dvhK2|at-#;(CrX;>;GC*FW3_DErQpu*-<>gF}lVaxe)=`*u9n;yW-=17|LTC8sxpb1O#%9iVXHrlz`zr{ox4x?7!f#>n;>9C}3 zq6(X7P-`s4?j@4ePwu>PW)8^5>|S$sX0IfqrVHZbE{%vAvuJ{RHl%^UWex){zvAO| zLJwEL%9_!QNDuJ-u3&LkKgKjb&0WbM@O$*jg7g)gn#-y;lXygeSc%*7%Azbck;J1) zA9Lyo2o(~D$J5kV<3c!s!cZ@5FQp)$RnkTvW@P$|`2+_RTLJB$R{$%A``nSq6cr8d z5xc(NNngt}7j$53%&o_~Q9@`=$44>HZ4azra?1i#5I{L`;2HXkLQEsXNr(-9h>^>D z79e+oBkUUIpQ&>pT_M>~+$AxlK1Zt>o3;5!%cZS2vOL=2F>tg2A~%%BG#F~h0S*9{ z1!5s{gn@CfI9v%C6rdzu$m4b- zP)QMkGuB2;@s(oBM{JyZ1>L$2?gpdT$io>hYb*`#fTZ(j{-lxPEasVmf@vbQ30%|w zSiS_VP;iMshH^7Uxas?2J@Nl z)=eyC%tx=~c`{c`Zg-ieimF~k3g+|J3gGHgYx?w3`wnjPxYRlDIZVV3G%}6!JdPn> z_#r$)zSB4eSS)W5i%v^rjgPEi<_x!cTcp#=>snI=+yGmE+~5~4s078F_AVO>i4eA` z%~j;*VfI2(5ELiB5Dx*SXo!+9WRYMc^oo$}eTrBTljobx7>6N=K8d61lLqW~0bjsB zsJ2m09F$pD0+bDD(G*|vZuek^L~*mzF51qWcB%zA>!Rk)A?kJykV~SN4#WYh$R`b^ zMmfv1WoO^IX;=V$!4B`LK$3?>*Ru2;ANs71{Ypq8b%jnUDlFxWk2$wgRMu0pK&wAh z);?oISC3D3iZm3a41L?6R|52T}Lz~T+L7x73q34H9oT`wsHp1cQ)d>3rb;6vb!gHk0~w@VyD1l{RQnG0#xy`7iN+LeJb4B6 z#Kxc;1%!C^y@0&j?Gpsc4j3sGOGYqJHd<1%$>-ej=C90V-XxkK&sJD8-Urf{eQAQM zjNhk*R>4vD=lJq>U@bKF=1|w~>i+Bgs2Ald>zDYrf?bz{(|Yx@k=UkS)g-qDk_A!g z)o zlLe|s4XucY5_o{7q!5cB#%?S%DUX4M1lNM*es4>k;?HHhR&_@>ig|Sy;QxcT2{nxknSR$0zvRH7x*o{X?ze1Vz3~iuhh}{u!HY&BELct|( zpkGT$J%`7nfnX7#E}R5P9hXFY18IB9A%*HL8=G}EBL3KsYcT`rB;;{K@K`V_pNShU z2~Pbj;gG6O1PyexQNm$pe%9-idgXa3Lj-ywf_#EKU=A7c;p79SYS8}$9=f7D!9Kl( z5_AkACj~j&Lt_$0)a8}ZhzZmUdJ-3bUx~V+8T}?VmUN7Vq}tya5U-AZaJJnS183{U zt>;ev#~(eX_Tx4g;7zhsuWy$w$>f?v9&eN!G-d&f8PFd`;ZV9v`o#3~|I{llDmAB8 z62c`MVi8mXO4mTQI2dtCM*kcF281B4)Qj{eQ+R66f(X_G1!N zJYtGxUZj=elKJQ7%ZQQilGru$vy%L&SCZmu#JDZ7?&cP<*Sb7DEDPzZq@2P0!Kfj zJdP$xPV=RD5&J#@awVuWeCeoiDvUY6>@hSVeu#kei>Y$h0DUY2hE^^ot|K`% zj#j~iz-8kx3lQu~SwS|rg99P%sNgijlTYn&Q{A_a)z9@2mZDXk`DO#&u;^akivVXq zPSr&}{ah+Vee4P-QqVUX{K^VC;Ur@xphD?%m`|!;$GYkf6*2{}00}@9BUyS7#^y_sTGgSpaf`lf_X$U`Ojr{ZGB!xK)LVc3kuEJ~lJ z2&T`-O&nnXtCRRd4hw7oghL93Z0R$U3bod=D-!kfZvbXt{&1LC$-iK(VEw4SJ;I^L zicvLa$r?lv*gW!E1AV#66g0{LixaHOSoA2zEeebgx#Sf*#CowyeDh#mMjfe~P;xs@5 ze8nPAcO)?zi!#|BgQW-HW`Lw~KCg~0Rf-6l#Jqv(rP9iZz8t}5M6l-*&W)Z$D@9Tp z+LA7#8Zbb`fR9+>d-f%T8T&o~)DYk%t&>Huka?x~D8pK?kbJ`;dMm_E|Co8rBxVr9 zK0>T7IA=gOMHU3t(h4*o%F$pbW|ic3)tCkyYfKOtlLJBs`Xty?H}(;^34Kfh(PxkHC;@i7pLjzOS9nO=Pk_9j z)S_xc1}P7HX>3&m3+*wNS7XdCc{NG`$KtE1uD&#LLa$#fh0NPEO}&Dq05`GFEuo=Y z7Ei`BXkKCJOEp3r9^spl0NW2|_9`88=uCyj5st}la=%`=EO1E}^XfH=E`5eQcoav> z_esR7X(QDY-9RaQ*j0}jMl;l5zL@lr(1Yua2*e)3`EkVhKwAVZuSqOzFG3uT8mNjx zS{)JW$0PZaMR5ad)@rp<$){K30re4rh-BE`B|e^XiOT|yH_*0tBLpp$#OC3qcsRh+ zPa@LOIbIs5UV}MMhZu1Df!G(^9T6|_NwkhgWmrLF9<%7OTv>iCfr^xZwp5)%)^Y;X zU+5FZw8N{Vm6g&;+Hf0=d!45%Em$s<;D|OXSZ|u~0@bMItvq;Fp-fj$g#ijAnBzhg zl$G=kBg7c-L{!dld#~ji1t`WtbUk96XvMf7#iAH6Y$z3@VY3g_9_J&hz6mZwx}-MH z{-Dj`EMp>$#OC)#cx!FYaSfPsTT6vu{1PnSW53s5(vt?D@lZy2X(ms8&Y1d5>0#5 z{ETr}lKXW>|2B*aY6(U^9LAGf8qMetnA{0^OoO2lFo7G1_=1kvm8xkaF%iu$^e0lTQKAFf zrb{7NOBL|?AQddcaoXgZ4g-YIaYPv`M{_41GnhI=2Z)onw1UEXkw&THvFkwaHYR{1 z-~a0K(WZ8868TbJQ59eHvGqgc#dFYVFLQKF<0fhJm7*pjxY~=@SWJ2_;Y^}sBp9o=TGxWOe~3ByQy%6GbrPN?zL`0!C%=F zQVWLaiXGg#^?fr-Vh-^`z`9jlwyz#A?7hN14qOuH+H1hpvPe6d(^vwyrtwHINT$Bh zS?nLDl~EV`LQ*LetfFz?DVa0^!)Q1%5skAfbTNCCa1uGggnFc^jo^SiRzHgP9JyEx zb}+LVFB+l&9#emULfEB}Fe%;*c?tF%NPmeqA?T-_f;Va0k9ks-#?=W(Ech`&BvTmG z75!e2dm8QmN}6tz%Nk`B>P5Vj&eu3bT&e>41aV`irw16)<%-izszl=vnONmPoRyYM z>C|D4u8Hp(waxzV)s}!)Ws&&4T}ct7M&0CFD`)>$VQfDHqFnG?Ug~vlFf_2F5C>T# zrQm;TOLbx(?8jU67j@Lj89-xkg59VSXGgO%IaSpn6;*zh<+NU**|BQsq)yrMNs17KOx~@|a3f7^p3+|0xS9sCCmoWwF}IVn>!^Hx%P( zH3cKAx5A;k$WyvJW5a~?5UBTPrz6BM+i4eD)?_Ny0#n0Ra1Z6Oh%jDFt2mh|MOq*w zzN;HfN#Ko~)v&H05n9#RsKp>9ip1IGM*?>cBW;)bn62d}Os5MFOGz{dk>%CBs#t4& zmYL^nghSGkssddnA^?&GL%mvM7dlhVx0|EHfd;{{||EL~MFWx6iO#Pu1zEWv4~ zcY)XkD=51a;Av?EnJlK_ppm+u5#w>wr~8Jf1*%ViA-}B^nGP!g-~S(adewF zqz(<76o4Zzt*)dl2~Zp|n>%da;6PUeB=VRwP%xyy4ZKrAhMS`{g~Udb&vrsVuDFtx zYOvD0Ju6E*VA5;^Kl)Y%`B&aZ>4L5C(x!8qhCR)mgk`z&zrZqqbUFDTyatcwk*K^q6nr z5q@2ph9k@@`>J43Vh0DF6PqT0uT6idb3};61IV{ADL?_O5+NYLQI5HxF>4P%O#%l! zrWljz3RcYnx;Tibqn`+b^>=H`LgusK1a-rRUE# zSsd?K{?;s!k#A$&hkE|l1&!nY<*ZM2UlHNO+Ed!lu;SoOiISUpn-l# z@I;Kq5Rd!J|MADF=5HJ*+7M&OW2k|nq1yBga)t5@k{7 zB$2Pn)i-{ClJy2iEt;Rxlr1}G&NbezS!F@aT^k|BfX^L?9W z&;Uu=FJUOTg6l>=|04Jt*|ivk5e+kxGa{l#I7U3iQ4I0OU{bBcluFo+@_Li{)+hQQ_kl*K`c*gZh_98Z=2G(6{?fy3_#aA}i^BN{? zn}c@V9G$h{!c7^NfEEt95%C1mhkY)^X|cNHVQLd`9YdBAL1}z;y;cH46^S!^Ev?AE z$13gycq*}AXyR~`rj0%oSZ&`X-2=t6RwVdF_E?cRQcr9UGbsL}ic@kt#LTz!I5 z27Eeoq&T#i;ZFKExWyF0m4g-)R8%zhOkB;=az_w3A$M^H09V%(*4l)o93?@8kIF(m zfgFsIAyl|N}lNes93YCvGPto=;**e|@)_wP3i~pilqFOX>V{Fzt&mGz`5vtXES5^|OduSwA0k)} z8x=%6DsQvOy(rbZzq@)Q*ed=~Mv)8{IbQ{iz82@tITuGr^nU7SBYb_o))S@*cP(~<%3rtc4& z%$OG$yQUWb0tDO^%ZWo}GlmIw2c>Q%i`|FOD4f~zYWNe<7ssynW-aPE-6Q=&`CC=1yE zVXoF=x-t=}R4V$bRz>{%l1%=gA1bRmFkMNXA@p}Cm_*Q;)CGQQ9v=Lo^zDsZdg|&^ zM`!@XN-r$&JRLxbGRnJuHAE7kju~bq707kE97{jHh zR&D`I8!t(7c9}wEnUmQb?|@wE-b&|#``}knm*0l3)I(q*m6pGJDX)WZ#4{oK5u$m) zFcQ?31z77X3G{tKER%@aqjxQ3#kuf9D zAh2ARmqg{<5fj6v?Xs_R=!tYTwEeYaV71V!YYUh#i3M~fv&@v2WIb8Lq8uFc-b{(z!YrGx`IG>!ou-a%s&YWc>tYtrC`D|Sk zq}sD7AWNAv=WMB^ni`(!95{lLTLnq`T5!&51#B`XJ%n@uVQZo!W%~s01Wejmrpx%6 zJA(a+jf-Hm&-Z0h=k++0u_}mVAE1QiL*|hN+S;h4h66gTKxzx%5BNLSYW@=IjmKb*+ zi=RUS6&qdTC}1gHZ) zYx75ZdF7b(9=h7R3I)s~$4TF(d?Ys-Xhn-ZD@Y)chH?gto|M(KbR0%BaA^oMz#~&O zBpL&OjT;e9a2&){`MNfHdlf>f&xKZB z3@uC4iilzgZX-W-uZzZx+at2%U}W#!^gXVtg5xge`-pLgt64wFmMP&;m5*>lysCL6 zwbsf|dHx{S+i&UpH)8|N+krE-phk)WS8@v~=Kv)?9?$`CC$3M>8^$hq1+}C;Sjb!Q zp3Oyl%tuBa$|5Pc;{d(s^^Wx|Ldn;F0FIFlL&oE7j8khZazoq{YptWz1|Y|nOI=#% zCqp5tOjv@hF}N^6cr_^?VdPT|=JO?0&{`lK9Y)k0IsK&H_lY#z3UNSu^ofN< zeY1i-DvcDJi-J7MC>yc`r2SXWN#|f)AmOuMs8oMaIJ&|<_0TO1Ark~&mrx;1t7YpM zSc-oI8V_P4BaA5|zo?)CdMV9)mG@3jKqPPoRwGayORt5LPByREM(ET}3kmEyK75JHIZ-Apjd z3@)iUGmA^gtj&d;P_L*g@n|f)=p~1!9*fzO7r-ki*Q3obwcUn!r5<$15Qf!K7s|Ij ziOH@#L(*oZaGZhaaQLGaqc){zDb)_Cf>AwW^$o5drkPImRkXEOM?ZSBpX z;l7s^NV$n;U%sG=6@%TbMuu7y>`r9>b zJqNV2Osga3e>}OM-bFpfe#SHw3&1oa>gS3G=9NmvxMuE^r|kqXOpznzL<~Z{cy_~` z2&|qt^;+#&K{>%J`O~m72=U_g0t01}sN6aViXb?sBRrl7M1$I#^ImSlVLY>lzUG+M zhzdp)cr(jDEWC&Jq-Z|mCn8?rk|nV8F`wO8Sp<+RER)XttcE0niwklSMi&<|3MF$F z4oF^+%p%%A^_MU65|{kw4RpxCcA>b?7~crmbr)Ug2xTSBXS|WDU##Lz^sr`E=yOnd87tYL}!t%ucjg+wil3Gb7>IfC|*QdLhZN% z$bf-mC8=OSr}v`Hq%tItm&DTxoXhCug`3by(GEryA3l62D8~ z@@fgifv94!6qRD*kOd@&8(K;#D5OFS<1m!ol|?8e<1vmVGScYu%Kq{Nefsi6=dU-B zRw`q{IUbTSs-U~OPw4J0ceJN&?+-t~mE*lsBRcK~2LX5$uarzHnk_UBY^E?RmE}vM z5ww1r<-24sW1G8zCw2X3MTN^VEEJ1|7t}LcCz4U;te3LQLrep76b&mWS9sw~vxW9z zOL;VDgDqGr99#fC7wXwRwf~9B#x`GeVfB3$&#nlPF=co#@2K z=0xyOCb_#}*sTx)Q;fJh$0h=bnc^_jna&s@&jbaTKbz8Gmql=|(mv|78mK2_*Ay7Y z&@e9`*h&R_)pGit8vCj15L@J&nt7|GP>9P4$~vH7EN22`D5EZ6(iu;61$u|*^Ms9w zV~VBZSjaV5LiSy!d3N!(d(_q{7s$3$R2M7{EsU%Aq7?x>)k7P>FUIz1H4I2B*3O2d zuaTR>erB;rwsa(4gAH~OikaBV)`qX9W~|ki@Yu!_NyRg)-9qq=%)Nd8R-FF!*8G+b z>Thp}QyoqzOIwvkVMt>i_rC%KSuu-+eAZjkYZ~_x_tNG{mhZT(n{BvHa73a-4MUdP zuV@tN7Cx|#8o5%k{>EXmdP)$gB`q=AUKp#+Chonk5t1Y+;b3fQd{?M_fwrIqv> zU>xD1w3$}3wtoKhRmGX@PYN*Mh=D(s6)kyPnHHcvi$|4g0244+SEJT2L#F$JUD>AJD8*=@Q4XsFa2pJ9PR894! zWR6vJ{LhJm1iszapTyKvSh*wOUS14Xbdf-;L~e+L&xW$d7_RkP7H}D!Rsp^Uboj$; zym~bN0g<89qAMxzA|yTC8mKII*#TM=E4d<_&@0-TTzMf(JqMTHHFS6P@$Ud*bc#Eez5nCBAs zIi>v+grFy?CBBcsr0-J~4X_K@10x^KdvW%P-4wBFuFhf^>q0wIOV0#gFL_B_`Y{$? zh@lq03y*&0rlHQILt_;u@km9-sIJJ4X{SUQIA)!#$b5wIO+z)9_aTxjofwJ>PKUvT zkZSB#GWjZac}EREeLW)BYp`gjD<|>jVpeto$S_)EfU)?8Mf6sKhEr|1nxL7zP&d7r z3(Z!=A+1=|rG36^M@Z5XVqc}bCEf*#hB%^6@dYn2&# z_S%>s`#uP#5VR>x%tV3DO#DHZ0n6A zd>4{o%a9w>Vvb0|P)t$io>nsJB&%BMEVjqRi1?upDdN!2aL_7Gkmc~jK0$7t)EpMv6Q|u;l%TlR3B z4A)h0Zd42U=dX@hYF8xcGfpmiHXMqMy-YpS5JErwq#|7Y{Ihjaj9#MhcmlKJq-hgZ zl;<;giMBVl)q{kS3(kEFRtYy^H(kU&M=!H>J>0+Gmo&V%A`u--a&P-I@GiVSG@elJ z<$v*Ymb?_yg|B<=%khLoL-9ohsL5=O?qRQBbL7_h`cjLXkK@N#=K@ek{L^^61u4MI zUyzHPs>rJmrD(FZdXj&O{8f_ZE<+jA3IOPNnTi;e4i%!blzQu^ND2#-lj zB3_^XShw#}0hA^6bJ_F8S9xZtFbPwBnV0mM4kPMSA?Ji_A$88IS2ljAY~@@t7X{g! ziq4V^&yo+gl6pIW=kE%diqIl!JfEuYyf)iA@^=hQxq<{C>G}Lrl9F!ftPJP*B@K}v zKtO3U?*Ku45(t(@!3l>?Lo(;IBmyS^{UsrL)F(1zmg!k8Cb|kLmx*((RX+!8h(3a1J2coHlegx`gKHII`wpA*&KBU*&%BM9^7Ti?GWBy!{7I^=)|hy5(m7CpNgLHF44M!LAovkjK^P4k+4&io`MpvD6f zMnsS#$f+pRxXK(8mYg!8LPI!11Rxa!MOLu3yg0BsP$GMjZ+@QaBz!d=6^Vry7I(R&*G^UEC8Re zODRpIQ6bbFgc72F!In!{tzNxUhY=1(5YSCe;zuh?2mmmOtDHjXleyYfnleGt_oz>J z7aSauhwno#eAo@nEdBB=WMLxmE8%wm?6q@m^#JY2Lu zgsaS$6&x5d#XJl4<4LE}K$nwHEzdXf2qLNh_ave|iKu_T>x#_P9|xbq>nxJ8t&W{u*XN9}IID34zA3cC%WN z6;R`IA2z^p1AjOpV1WI}I01wAGQ5BZ3vfSfAlt^DfFIBZKXC+49KjPu@Wc@qjv$TT z@*tkTj@|NL&cG0>N8t?&v~SHBq=&>p-k_aExLJ}rnCjJH{s8hFsTAsxYf7(R47r7P zJNkz%Z9L-JGAjxjAm)3hBd^*dmXPWW(?9^w!7$!X90}@%N>A2A)DA7>`1cQ^7>Ba)oaAMp^TB-XcRC5Xdqrd6UzD53#>Kc*D%U_(t zjJw#U!Eh#PHKsu)QwhS3rH`J&BbnEJ=lG<7%GCH$RJoR2VC)#2SG zS+^4p7S~E0-k^}3ov`U&n=FAQAjQi!{+vVl%mVcj{ zyI;|z^no-_RRn4tl#DNCfp?}RtFvKP^D_m_yz3S@eLg~_OYr|hKRs(6 z96XbzBg=S_>Zsuz#^OC=<&wr#nCdPh(>G>~B_tj+r_wUd8>Kgl$EfL3%z?BBPEq|b zA-QK)4kOCM=*eh^qIvbe$}kO9OllD{3}ZX%d{2+`>3|ajxR~JnERsx+B&sY@3+9#cr}D1phe*+SYttsBMdmxah#j@?k60an{;$Ej8hSwK}NT%B52a=#dL zGbPYYXIBtf9gzWyf~~M>CaSK6R1kVakWR*{1sMlbrCI!#3bHpQaYS8hS)Vc`4K$WW z3IPnQ+QL>crStkF2m?#L;H!(}bi`fx)+)0ocnu@YNGM64#!Jvr^Z3O{A)a5uss zNu}SK5y%8=2oY)9kdgGD@|at3Bac@+a%B%(!HB_WWyLeErbAGX;|w*klUTi~|MRL= zsn;vD6>(5Ga?Q$2<<=tBsmm3Xz^>V^>cx*`xV-&=?2i~;9k+%C+OALKGZ&(l&MyUd zs=3^s{26F9SQHUA_R}a`O1EJ+Se^lz#+gc2xtgFAv>zZBbLC-i0fMVa1UrpNCe>Ar zC)bUqjdin$dom`iQ^Yxn5i@Qmk~$w7+GLOoj8v8bpoADPSb*#$Az>+vRX8P{A*};Z z9A%c;V$`crc)@cmd|2b}$w#H$bjgq+>Dc%h*~O3caW`mc(D% zTuZf<(T9Tpsg;+-WX%G*S5Rx`hIN$;D^@^9L&ap)m>daA7qp}mz?rLlcYz4ANbj0& z9kD`4^volQe#wht!4Pmxhk-h>mf(x6oaF)Yh-1zc$d^)*>3(!yr7A+^m1jPYX5nAo zw#iSHjV&XXv)I9b=ftKFlRiCtBnMu$)Eo?GAb62!RM4slLofMfEjO$^2t^6r37O}J z@u{w0g#e(9gP1z{iAG%iw#HCpAICn#Sd0mrHD2PAXkDhADvLdSSzc1ai7~(ACh{+d z93`%$;bNU};>R@fiDOzy6VtDx%_~M-ovUf#vJ?Ph#%k+cC1%ogT2Sa|CKaSsQDsNh zunem$fZEG!NoZ49-Yifm4biforcyT*fD$WfAqZ7_iW0A^pnrnzQRC*Lwam zr@3`pM6uS^svQIEO7?bWhsBdYSnV!I=K>r|kTEx2SwX9L+>l3jOanPdqQNyY$~1Qz z&KhJ@F2zmF4^PiP75HjJs=8%y-SJSXk0epCXKZ29i)H(bb+bDnepOiyCtYH~URj_OP_W2A$;z0D`Lzx%j1j?ZM*XUm z*sCOlVAnAn6TXhFiOhIq>uD9I2`g6LpUVEI!ILPZL%J?|1}THZUb!Y2FG|_))Fi2y z$vfvla9)yLhtC`>?EnRLy?tU&I&jHmrLPFmr~>-Jui3L|^jR?W2WhSf6?mIL;-d({ z1bUgWEt5aQs^Fk6ppcSD=uKJ|v>&XitIP)aNrU_O&Je)a_$eL9%DjJw+}K*qE#6w$ z0JX5W)sJ>hQ13v?f_urbB(7 zxtF``Zim1-o zW3)*5CtF*a_vm8Gl=MTgjDWGtpssybDPf=GduA6|&_No$5>wMaf8W{P?G$z{W~qCW zlTNHn4fIxi;3ij#=jS=Qw`9aoA;e$f$b*b5IHsVlod_n&+Ar4-T0y1SLwzE~wqB`k zB99R+I3aU1OlLt>X(Cj9WO0(dQcdpIE$W4wa00o428rJ#ahiu>M@NS6CIKE(7maa1 z2SOx7R}|~c%iNkrkXznPooJ`iC|g$>@~9C~!c|?zIlZKT*Fc(a=;~mrKz&J!vnj>q zCCC&jq+x}-cP~lc$t-n4H&h9n55Y?~%H&qaW)Ff1H>qBQ1moz4omM%G3KTfj?@KE} z+9OfiW(Ndc5>Ka%1_KirBpB(`^yWH07eH7=0v!?e5^hCR{Mi+4A*1u>lbew7G+eE@ znXzN98o{W7_G4Qq)}Ls*kVGYLzZ=FM&$QJUla2ffhL`rNZ}$Y5vR8=RfOnS+7;$QXt(8S~bM0 z?W10+TZwPthqY0w)oR;YTkyA5tL6Vb|6z0cFZHb-YW2;nAGWuj|D{&n-l*08f@+V| z$$6iI$2j^+?ZIv3JNGB~)GO#UiKVR{WCc|jBj9W~;Uq$Lch4ZKw234wXpORzZIlEn zBMw z$`TV$;a6@&i%MWFS(5ojW~CmvgDXE}eA=ghXA5aBYU?QU3Fbt`${<#D87p57s3any z!m9&U{OQxD5}FZ)S}eLI7i+SCYSxq2GIgMOG)RA!tmND!;IGP@#4eV1!wxDxb8&bP zkoW>((yJMH`QIdbrc21g>3tehQ@#(T_-}vHMfk`Rj*_6FeyWb_k{DB;O9LMH9F}Wta7k!R^$8S zi-RH@3`ucf88zof$H^Cq**p_#5+M=Vr!3_bih>pigFrTq(muf)O_)b z6&@JdHcF})I?$#o9F^P5BgZU?%TpT*!o_Ix1gjOrsC~5Cx;Q@S^j?0qYE!NIlB*nv z5!}BNwtuSqTnQ9PcjfdRZt*T%n?`j1IESWy%pZy@jma4BqkoLX1z{fG5>iCKM?nbmHqbGpA% zFtK0L{^G_nGym+1o^an9Nxj3x+|%&x*;;0Yf0rWMn00hHavFl;K*uQx zU`nBU2CL5-KF0+zn!dBj)@j}>S~k}pF@l)v)b zUCAVV{PG2TW;8(Mb*llj4eWH2=+{!p=YB_gGG3Q|PO&eVq=83nbW3ShXh6&Iv{Lj> z5VqAz8hHC0mB%uW^LJ#T>y)K}ro1ML=@FN?s6{o%8yd5ZLOxe@eno+p-5^FAYie&! zuZrbyO??UspaJUBph|)(#7E#k5F>{qgOv8nM?I@rxnH+YPBEhvQoDi6Hm8!o22HzPFhRn>PI_SFsOqQ4)gJBxe_&aN$iy-i{`QowpE??A)c+p;w z3GEB^N)q|yHFYYxN#rAkUk=c}N(ecQvL|tr(>>a*sTX5>0}u6+%)zgwAx#>e9q4Q+ zJJq=?=e$AXCOmC`X`pgCuL>!$tXj(ek3ew0pDA`7F-wK8bCuKiHh8IekongFtTG8J zXDsm1If&3!)A0d))62Cm@?Kv;X)&k6I+V8yZSnTq~OH?i^O^m&p zJn?2#a%M%_x@?{qf+CNCwLFC}qe6ZWa6vhdZgsni#_1)LDiEI)pb{#ySu7B zQ!G4-J84CI$`14tQ#z~suZR;5J*zqns(gL8P+f)Yfh?#drjTk6{Vej z=R-8>Xi7&`Pu$Wk?f=pp;nwbLCcQ4a`P1#w%HJ?)eXiD#z+OOm_ zp}DU)BItWK%v;{v-Ic^+CuV}OEA>~Vf;vaOn~nPPy86A<-cKL(nScLdSq49#jrsRK z8};q&jokgu54G*>r~998@d3Tz6&*J~tu>%N5lq0#u^1M2L>ihYtbSi=-Gus@*mn4b z)l(_WWFFT1pL@+KR{`PM~SbpdV)i{ z4wpbVT2<~d%3_ZOL+}S!tCaQ!D3aC!5{R*kka0NR8I3dXMj^BxAg{2UIk2g*hckTu z>6d0aRgszRml*XaLGiELxd65(1A0k46T_jBOx2@T3n*O6^qGA+Xb9k_vbv0E0e= zzhK!vr2(c?RJm~QTo!3+(F{XPFpQD!7!OG$#&G31ZTohm zl{jC%Ai)-N$dEutQ0$VX?;p^ZMA%2=fASHod|xgATP|CY^)w#y$W*Z;F^!~5!09>t zi4?8$oGI@_RAfa0{)2L~)R?LZ#W_NgoM7W_zG#Z`bK(>tZw&6{7$p`|(?*He9=E3{ z*u7R|CPeLHuhA^`z4cNG<7wK?2ZsyWf;$#<&V#!Iy7Gj^GTsC-o~ zqq5YGlnV}O+j4hp(LuguEqdLSej+dK)I+$zDHpjYO&wXQ5e~ejAC7Q48KbS*+Onz{ zGZP&WCsCloH%2UsC`e02rw_+?)W`0e!GzaomCPCaU1i6L7+Pr_@2gV0@&F=J0>$d05mVt2XfU*TzlK&h z8DEhI9PlH8%KvzY>XrJBPE80CHb5cpDu`O9Sj!DqVG}54#`HCKW=DHYYq@Eeo>OuY zlR)0(KJFyG5sO3!u0A?u9%`y+>Li*ot)mdnn$iV>WCd%lUTugjAY^xQ=6f|H^Gb_R z$y9;CKwmpBA@g=Aj}o{gd6jrWGMB&>l%7cAV;rk!r^kt}V!H4O+7H46qL8e5)D?$6V<4%N^F_<3^2LM>%D=o@Tfk;# z02vc`qnMX?5|X0^o8Ie!(V%*lwHb>j52dQA`tgY zX85vFm8lCz#r}@44x6IFgUzv5krCXaI&;T@HO1BMK@i@?OBou)yuFvLy)h>#Hr+MiA zAkkhFLK4!MwAr267!>h64Cp{0huOu3j17J0l#B*;p%&zHL5M-WTtOc%RgeTEqV5F~ zzm_P^E@QkzF?^du1bfRMpwdOUY`lEwfW!+5i}pnkv3)M6M}nBflZ)9Ek^rt_FEq^% zbw~IjA|B<%GpDj2!1Riha3K&ZE~-O9UcfndAvP}_k_z9Pvl3_2xh8!NU(H-~Git$b zT=abgOG4e4sU|<6U%}@Zwx5%+NHv7CkUr%0vYMLxfG+a8bBc^{FV=0H)-+|PWS&?s zRP$~UMI?w1aR~J-Y^I8;bC?A&4djUdfqwJl6k^a5<>#kR1Lver8QYpVX50ijns3CM zEEb*saxDkLHfdk=Nswtz@VcH?S4d4)&ANd`WJqofA>Ks!U;p}U^?w@|7pwoqI z1L~7suW;hWYis}e<){C1_w#?>m5ojQ6plqL$+iZV`s!|QfXZF_oIub1i$Bw0YXw4m zV&eN*(QDJ-I}mYwI#wPeL?hU%m>3$_Kyy}(O7O9i;h*~xnVVE6fi(GN3m0>xj8>(e zZl#r40NioMNQ?FpUq0n2R;2P`^5hBRVe|+s~)sztx{V*?)bD&kP-%Eh)v|y$30S%Utl5Wh-iHSv-@!ct*fl z;KHBDu^?FYe9{GFmt3~l`=T^z#aH3ehA8q9HHjG{Lg;2nfWKDOCU;K3!pr1b#4?HP zR1mW-sn=>{fyZ(Nx|NYkytfmS-4CkmtQv5Yrr_?bWPW5uAyZ@?eqNw*K`B@&G1LQk zJe5QZft$%xW;GpzY`UMTaEClQI8Bj1YQ{8R-_EYfrSdCpT@PF=p8}bCO#>P?;p=yc4tGDZwcYmcRMsbj;UG{IyKe6%!XTH>8E-AUY;8 zJg0;5u8PZyiAlqw0i{{LzHS_@?94+3HAxnfY;724IG0uaxjgda411kDQ7rPrWdb{7 zvEy9lw42sw=Q;NI$7(KQf$oL&2zJr}k3!%yT;rX|_m?7av+CWC(9LdrCOMc@Wtt?+ zFFB9s&F|b-6TUw}Z|_Y6XThB&gqj$kpUM#NgpbN9ZEpmrJ5kvIHLM9YGSACduhYN% z4JlyAxj|JDyJ_jdz$<0+b1?~;?#ScQA(K^nsgNzx2$A5*rmcaIWt3ZYN}m9l?TK%_ z{ZbT?;%jg*sj;V#(l}WLVHwnP8g85TO$Yy31Ghgts`VP07jS0LZ2qPv?J@|E;7W8m z^?X!I!X67pT?Uy9;_@@=HuRonoxD6MKI)d(I@70Falt>q37cQ|8PUwnd04o#jUz`v zm&9xH(=MYkkIN$ZSK*SLKK~y>I?Qqmi;QoDw8ULULU|NkLL;`En3OGjKfWQ~l68JI zW)_ntVZm`CsDx~&`Wt3u!uB$v3DlN5mWgO zrtsK*-9GBHE}93OR&)2`#c}6or`7GYcBgOx!=qWaXlRG6-kYQ8x(EB+UO89vp!uqG zP}K7o9g=xBv*S@1X2C9hicN6JskSdpn+GSYJO*FBs8hHy^8htNZ)vr3<31Z6kSpTP z#7e)P#_8zw#X;+IdQ852nMob)-`dQX+dDegZFQ#RBXiV2&W^NU$r3ujnT@soi5l8`;>6R`51?$Swn|NTh|LQk}thplemVq3Js<}EQ! zaO93M&S+T-pxNVQ;a~_($%M(BV*T!q%OwqaK0hT99ZaS%>Au@PzUUovFHT#X{k@N0 zz3S(p;6^lz3)>u~OqRE*J`t*tfg!AKw_06DV9zGUps#_Ffluk97efs!qKC(zP zv!&mZ-Jh#a(oB;hOD$%%=hC1>D-<}^Jg=JT4wzUL^y({^&IjHld=(FB!FysvpBT}i zh4?$^MP?xL$5D*TT+Nw1`ftNm+r3(vxt^-@$KkLusC^qYdw!FT$7@^dJeb|KasGqy z+zYng*R$ParSDq35BQW#U;9dN;bFY_f+714XCJMZ`#K&nv$vMx>%Q8WOXhXK93s-U7b!$PBN ziz26HjK(XxfElyB#_#S*`DfZRyEsBu#)&=SFE7z8@|mB7U2)2*nIWhp;d__#$t&z$ zUgOAv@BlG|U;wb`Wf5hVe?WuFY}ohd*SS{RoK5p`m-nkh*#ZjeyYItnKi%FxG1*VI z_l4~B>??W5>wQS#5i0=pbibd&>FIv|Ps4IQ-S2;s`+c*~7N#qaxj|04tn`!RB$Mf6C6$De z=pC_{kaC(erTl0^pQdGb?6famHDkJkKjVz)AQpwQmI)?0f9B?J!Gc(Hh|h4rK>!QB zQ*;W=B-{efN#~##Rx#6Xd}MTT_hfy5sSq*)iFzs%=v2>V9x^sZQa)?;cHWfjN%=Ar zyTtTu8bIWe|_efpNJ#{n& z!jqcjNljC*sD5|Twm+uJ-yE|VgYX|*L(ksZ)N*7r^WlK zp}u)))kiR+n{s{rGibJK-_!q)+vMd&L$0x=kTzs6y%|gQqS=8`YGQ00l?ANly|6A$S za-7RoDA$U4pNDH{=3=(+7+S2SDJ=WjD%chb_CwB-uT`;qquE^y09o-{k(K(k}*y+_gO*rfm4XmxU&%Dr_u#hlJVbpms- zT5z~5(gzfl{4I4k3x@dX&yKIt<9w4DTSABPf7$!r>#T;bPpY0@mZKpl^X7ySS2}!6af+avXuATeY@4`faBsgfWleFeqZ7dSQgXx*@ z43D;Ai|cP{rua#F^Y_)>JXCBi%RvwQc!Gna^){;dlh)=*Ym?tkkEyfKy?)~w8x7`q z^fkIQo3^H6j#k#xEYR*cR5?lekE6&b?{0l+9M#9lN*oRA_hg!*Vg82HGYLrR_Z|C+ z>Yyk6%ai`)N&m72{Y!!#)}nn$arWcrUNSt#!kQNw$FiyvUjb1idaje;E#)3k<5D@X z%QCRbXQOTxgX`jw2<#5tV8h&k_#)ZMFU zN1ACxmeGb-R1J4`#YUx)VNj^Z-y7h{kspNNFDRli5X9Y`S{IA*WGzaQ#~Z=II+iqw zd{()VTQ`@?W-x70{Tw7kvzg;_rTNxV5py#xQkG+w`)NP`qe*h$`QK%Io8^}8ID-z7 zF!mQ;3m>>77~@Gj5=e3j1UOY-(Rd`w8ei?y-uGab)wG&G@UQhCh&C9R`4#|@Z&ln! zb~%MJ?4cO`42F5HSOcXAo}T|o1Z^9vx?u9ESUQ0=5#!YxlW{lX`J^JJdLIH`oiF{X z0q;N=iQuFJ$IoHZ_3=p0Z#%;+WwvR^(E(*Ba8Uws#kic}2mVqz_x)73sNYaNfl28qDN6jClgsT%FyDGAz}h1X;3qxPmnh)kRBaAd84E!irC zsqd!sySe(^RB`MGsEV`RE-7N2^Ce!CER#Gs?_g+Mu#IbUO9ghKBw`1CIEA(I<}5j1 zwK}1?A>|S$>r&6nFJ)V;dZ3m)w!yLW=WOH7CTL6bg4|x9CXucy0EXeX1faYgAd-e;|u*lLZPI%S}%8I)` z$mU~AHcI+_?b{@6y4fbt#x1JR)b?*g!fD_2P=@>C2?;V&0W*@D??3}i4G4C@>**( zK)KSgEFLv@%xFK=a9tdp6}V6DH)o3$2oxE+D>F%zC3$ChOz*lOXvK2x~H=b zM^`6jNgeCu_824ID_m@gw{ig~gDe&+kTmu6%`NP`RFCxAyvG0g8D5n4_dTxPy38;V zOeg8Ob=pqeUtAuXo?e}uetdKC{-S!iT&@4w|>KSM1P$~cUa~RAS^g#mBQ_623pR9 zgp&Q}%2u=R+HP+$^Llu2div_%@ZZg6R(6Af$szRp5p=H?oMdbL#pb)TZ8e8yihbeJ z*JN21U5EV4*THj9yxgkST^P%Cc5wOj;_B$&^1w0^8xxd}Pa|1-fZLl+2-#HLL{zkU z`r%E%9Jx`uKAG4liEP(OM;bO=MWwCV4s)yj|j+wZJPmDrI}aO7XbxxK43*;VZGCIR#%8rHuI zmS7BXSX<9a7@;x#oRyUMo3uYw;`w5jvUJe4yW~#Fx_)K>Us$*0i8hj^13bsKSCg0jtKsMU+O`=9iBBfC7a`v;ha{jZf>@~oOFynM8b*8+ ziL_d(kL2e9txD~+sA0JQOR$XIAqxn_;(FJO5M)TAFhxNIKLPZIMXw-lO%Dqdgq}gV zTq^t4yJB7iv!59ZtsRTH<#(>~Pd!li>wQydmTxX|78kLvtGBHyUlUOw`J6X2OZQKL z^OJ8`n$YMj5;HV2D3bU&$RK^o?)#yn>B0|#1Ie4G6)tp(<%&1FHAYoS?Q>4$I@Si4 z7+apo-&xK3#BMOS;_~_pt?#Ng)ARi9Ydo-`O>4oPHGAHy^-J_SxwF}*PEinF?1Yyc zh}^#KNVO9OrDK@f3SOyTjBA@f6RDY|rAd+)^oR93yA2U&hjX``+wrEEYf^#9%eO-x z;za0h`H%PJ@^4rHQ4=icHdYjGW{w75hbCj4LRvo9=alIbICpp5e}aIpDVKtkflYS7 zZL^WrDjOZ(;2oNg=r*s{L+YYW&)E!lIGzoFEM35nRV`EeCpyJ5Z0>9Qss;_`_i_Ri zxU+DARmZf-lBCj#BtH>#HYy8`Hgi<(7A6{X#wc}$D2>Nw=KFl0F)n7yQg6qyG*hR! zoV52vNJ6b;O1)Le0;^(#z5hyEuQG;?-L$3g_!8&iuQaxr$j)WRw!mZSC`V_LdjfERvflN^ltn^!~{*PSCXd70xMRL6rny*l}Wwu~b zeO5uEr~drMU+1~*_qy&BMvNwgm<-KAA2LL{t{+o|qKNoNnA|SeN3YlG?eFf2e|x=N z{@?9hfBzr--Iu-o&hE?o{k?zm`Y&JZ@B9Pw);w106H^AGfAm)FD?hmJ177?y~O&UoZbsT%WlpOKC1})pIaRcxl7B)_1V2IyD zBo65Sd}_CU1J6La4RGLLm*YRcrUvH+1RjrPsnZob0mov$tl}D&HEFoKb%9QO)ks~W8zABiy9MjXd zMpo2ZifPI*UYgohY_z|qlvn;#hrJ`@1M`&B3JC*pLD;7S0%S z(Ctok*eqJa(ng*VQD(9_m8ZMU>)TawU500yf1I++g1aB=B#+4T(goidzzc zG5=e97fp%X3q23W-N*>Q%)C7>=mjZ@a4e6o{44iJppd+Z5iya3heStEdXB*^s>$KbvTa(^nV6J%T@UT{6G!;@4ei~>3_F( z`nxZm=>KCp7COC(M9mH6sq&_wq3A-Op}Fc8LKNTeF? z)T0`!*t9dUeaE0?N_yr6 z8v1V*TNlGHi&SWx##-N$RWmgZ(C^Z`NBN18GgMZw)9IK$zdlK=#{Vev$?XgUt7!jg z@&E45{%)TC-|s*1|Bv!$L{W$V7E^S=AVcHWzd&rWcoS^#NZM@TekE~?NeN&W(r!Wy zj*?CHUC0&|zFn5KAfaY_`h#6BhsYgdJ?SzF3S&m73w<0+j0H`-<3k_2P`i}rpV>QI z_0pnCiI9xL@)lD@qFcj1A*ov|AgfZtvmJmLWbX8|{*p!4I<+j_e#vB~`0j#5kfDi+ zbxGAP>x8pN`1w^Sk~BtDBQSO~f3i15OAN>GHn0pTB;ILeLpJkUusB8JK)deu9Qfxu z7+oWF=0g`bBkc3tbHdva_XDMH7Iv|)b)O>ln#MDwce0<}yo4%~Bhh$PfV131I@g_D z%*k}GK`)VOtprvZ}C_!b6&jm7>a!coua3)j92N^Qq+j zg*jhm1_L}sl+k5{fV%s?elLIj*W259;{PAzxx2GDWqpu_h^fR{aF9lY&OUFa{LREN48fhx9eBD<4`3pvG3ZQ<}7=#`$ioH~6{yIS(RzX9v32^}em{dM^93EkEMbf3D+7 zLx6mupL(+ZdyQp`1oADd*JL1{d8x*Lpne6vVS9q+W39F>$hPy>ur;W>>|gA1zk047 z?oza(U+PYg7yBs6UfX6~OJ`}PJPS~LFNIJPMpuPz>*&voBg}3ONx;zOY%p~5!_uKU z8^ACnlpXjOQe-Tr@$w=24kJv&FnvgAcDf6n7Knao?MwV}!r1o<35Ud|v-C@L$m*ui z+{gKR4*9GMW*n;$XcL=+Oq}^t*PJm@C^8pjKhGmH!IVX}rV0DRCnM%Br+<8iJ5e%%pr=o@P)MW^e+&eEDAzo7EMT5Vmu} z%8Qnd33vgPEI~1L=Na*_dt1R;q+47T1MpM-Q+pjkdidds1x$p(XP*C`)Q3pDAPEo2&{0BsnFz`M1qL zv=7$Ek)55qx0jb^S7#s2FLS*$$w!)-5ldddnoXw~c}D_%WQyrlv!EWo7v^v8jxWzo z4lnHR%2+OSC#u{brA(x>i#)8fAMkgPng)y5;s$zDp=-wI|0 zrA|)eMHzbps#qjY&Z_jc8$c&R+snw#`9ovm_wTY4HoS$MSf9vFAB5{{Z zi7_2ZjwCx_*11~e;D0f;=tB;E;X$pm@-=9okm*o&n9FyVHnt1~r~x9M#86hsr!uSr zVFh`~Om!$Lm=#8|6*}YtYQVa3?!5Xk^Qw&?kb)dFH zv;2MeNzWzjkWVgsNA(8ZZE)5~UM-Du?@F?$6lT(bk+K<8%y-p!RpiqwyUJIb<&wN9 zO=hN$L!}6kEt5(X#od*p`>5~UNC;BwIGYy4(&c(KTqXfV6QJ*^^0BCCt;))xhTb4I zi)DKA8Cv9%&(oqSR%L6klBtqaTw$g14d?T?sOIe$?ye@MOJ!I2)GqSLCwM7#WJ&W< z+K;_ylSb6-)xYr-`#1e8a{s@qIcS~z=iW{}{%^ncl>g;%p1WfFKf4h?DWF#ME=K0hTcfYc(uf5~QYP zpmoiE$3~$G4DFj($>j}3%Uf=4KH8v-zm75KBK%)~*bRxUJE6D|ZO#hT#edq*`~Ph3 zY`@%l;{P7wS&{!!TS8uvJ3gLab`ne?L=)d}lVwn)`B6&q`bgg4z1h$@n7s~`a4f_0 z6)eVFGUIH`07x6dY)?S-1Xu*eW9)))L}oy>p0HCP@h%?2Dlu(oF+05y_1wNy$t=dh zSL*UM)LP~aI7wz)&4~VxGdbuqy@4+?60jKvj5GJaopjNQw+SV)ub&*3t+t|5&Fg13 zPy%Rv(HnGc-2_$3Ueusk*57KGS-J9M&02IkRku7D1R*q$YMbbZp&8>SgB&+7eZ0$s zQd}=4JN`Bh8G0-{hO%gkk-sAnJJz#}3>5YZK;I`fzzeA0^8GkrcuTR{-i{SxeLOmA zT2v#qXMjQfTBfTp7NxRx1Gg@NO&NK@rDh`!i%Y+v$gl|izibqk8vcK~*X!r?zb|+8 zpZNdBcvhsEdP|h(e%BL+<-76+(=+mLeqp+Y2k;LoMsH~dOA<~-Hf=HOt z3f9L9Sn?W`Z1gUnjZ2Cpb>>E4c&55FERNEbl`a_#BJ(uouXevb9o5tGaL*F-f2NGg z*QNj4>F*Tue>;2IPxSvWo+Z5s@&d8i6I>v6NrVV>Tr%tC$-io7OP}G;J}CvCmtSt^ z%%{?zS65G73>vChR+{+;i9LyfNwO`CS4)BLI~j?!P_D)~tA`wRX+KsB0Z^t}5&d!C zlI}VSAe99cTJW-_I>UUkpcwJQR(^L+HUB3y^{d8!tl|Imdd2)d{oYgj$47bcOq5!2 z&Fbk`kI&VIgAkw43&uZ!!B2KG7TpF`1*IX2a3?LacM*p|6QJ&d(&zE42RN;wjvL^uxdBGPZfxQL9i9p1GH5T-;zq= zQjS-lw(`c;eAFi!J<`=mtUL7;u~h;{r*b)Vdd+E^^ibBS$UrLUxByjoV5r)onI8+z7Z9D~-)8yN%BmW^PW2qJgk#AtD)~eO)uEK4*q1C; z2_l>l$|y)H=KO3jr4s8v{?iO;QgrB^GG_RI@H3gYyGq7v9?pVmhV7!MS1eZW4(+5u zNCv&+uBgnl$q0__P+V2PXE8XpTGa_?d&|oe|?lE$M)pR_3XF$4{H7P4KV)S zc>dOc(OTsi;?x`) zG_b~Ou$orRa@pBvh;D5rkP-3GK*Tw!*=A47=R-Y}{9i~s57Q_PIqn<9-UKZe`CmRY z{9k{2Cm;X6zt?}V|9q4uhrUDN9VILNO00OE`?KLcOSLi$@L93w-;Pf-wXS)~^1?jM zaBwh(*iUsW=9^Lj%VurE&mRMrYd!O>wEYLl%KwDvANQ%l|2j74Dms8V`~SVYy!~H) zyZ_|>`zVja{y!Q)*9mEOtP{05hcJXA>|=%zbq>x>6key!TS!4$ZjJ7|FQOB4&o~z; zH;rU$+lBP3*%~Qf$ktnyEUbd%3(`47u1jWNl3&1qemI4>fGR@TK;BtI#@I*FY6raV zhE}v_jdR4`G2?jFQ4fDV?1{y8Np(TjI6Dj zsT8qFCU^9QwV22n+YaY$d$Y`c@XUtpzE`<1J4GylXLI*-jdu=a`{w<_suvJ7U5#-7 zef%eibWiDvC2u;Q>mo|u5f2rNqO4<0i>DfPA-+~wL*=5LbAAEQR5MK(i9`Y$Lp5!7 zO7Wk$gbn7AveobjxsF4*k2w#t2{z;_(4P0(pl#jjZHt+rxluPQ-;IsGsq*v9lU~Ju z9;0K2ucniByy&)43y`xB%fDSf4`Q~RQAadVY*afPe0MtQO8OrWe{paB;!{Wecl&w! z@BN*p_^*%il*#{Ygdjs%(k?#46dfzFeolO}S+EKgrC`jaBvP^e3RVKV;&(y3AM!e1 z%{+<96)%NVNk->^4WlB(9&%yic#uv<1V$b$gl$NRFsSHVX_?TECpaj67a$0Rg)iWF z>?11aO5TX};k8MevcnK7Ia5;VPcP7Ch62&ni{BJ6*R)+1$*VLm(q-o;8tHPfY)nzE z9i;*RpX#^jZzUJy-^QBSCh=3d{Tp}&+HHUX54)TK2$|R=5dsHiCsHoX&+Dglk`a}* zw?JF!E@(S5sh`>t#Q4`6PEPqBKBlZ~oukZovuwGvq~(}xXO5Ys!pas@Zd35-x7K$@ zrXTF7rT-(o8G^|wY(U-p?@m7cPygl1{*(XLqdYc^v7-IU(e+nyzx?gVJ3l)rZo0;? z6?53qG1LWXWLM`%nHKkMg8ih>gM>5cFm^NG@^YN!ZL5;iTcs>#K|7!;j}D zm;bpsIC^*T{_5=D;^Nm2=SNppuTM^o7phWI_~_(9q^>@Be{y+paH=XFo*y4vo_u(} zxtZ0O_#O=Hw^Z^=c4P%Jm33gdzY<@0X~1OC0A8vMUpz~9_c zga3DX``dZ`?!7YM4oav1G}0%s>!bCrX)&Ny|5J$n-P_&i zKk@&M@z{y1x}qt$3C#2S`tAqDjLi67+4q;^8U;2Kse9q|-toV{Jy1>{RYl9X;9#24 z$HPbn8j=SP6p}%%EH7BNfv0P^r^oU%!T$>rqVbpMRh7tHPn&;tJ0`0Wkcg64G)r*;j1KKSS z{qAGRK$|k!25rOnv(4Wn8zGH+ukRenTt+ij@hka_P9ZIK9Ock&Zst8lVHi?_8%g2% zE-t7Z@)lon01*m(?82Hi+-$zdhF)~_QtLpLP4uFdLxr~_Rs=Lo<%TZXeP z@J{V1@@Ed6cBh0~w+BEqocqx6kdGMZNDt%o0JKMFOd`|~AH`J{Cc&2YMPt`R$U|Ox zB~D&XNfZYToesc#M>O)kIDnvi$^ZWrL7z?A;7*~8IP!mLi!3eOZv7l~VTij^*+fgV z;1Uo0^nBxHf*9zCq|GSmM2LnYpoqT-Ico$u9i|j&Klb>K2qVgo*YmqL^1p!XpZ~iL z?g6~H1~0n*4*+j3FVDJnis)~Dd9kT_`vlt0d5>-I(@&ti-Rrf1bp8F7u-wyA_B6%+ ztAqlr!~eUzy+Z!)ou~ZYkMfuT(CU#Hs>w)MNQlRa4ysAc_d`LFP*|mbF4cvGEjs3s zu^=sx6m_6SdmKpfYh@Hjqo6h5r13o$R5OXGseT(VG1VI#V^jT1B5djgC{}+-;q-|* ziKi+4UnLY&9sb|$7xF*tzI^ikdz9xJ#QzT&KZ|{l!dz7c?{a?c<5NE_&i6L8S?xqLk%@)KZT9aGEP)lL5 zf^9=~>9iPsNCJjFGi`R4dAXF&dNyP=p1}!n!r1qv^-*f4Iq}H|`m(^8@m>1t=iXK< zc)n;GY$rQQiXy%zR(zSPSjy>S@um4VTPjkR!)W?JfyfcLW*jj&LCUXm5db zM})>!)K;1bsPNhP8bo-$KHx}P z+DPJHvL*b=7@i~HU}wCCf!BHl0yGg+2W*598spE%llbTVx9I?mod5|$O%4(WQ3QM( zAgARVU0hu-5~0>Ja7bn|5`bS0FMx+5+HxkCb;bWB_$_DjXVew{*Dt1%F8`nYN#{W~ zZD<7D>o^2s>?7Lxr$cYT)<2yQyl(x|VY9IH&;Pge4EzEkOkxU7j*e-|nNyc|sO6|T z=9Ys!1iP|)M1F65`1Sqq`NemEWGv_Z{P^JL-LW(C*40Pl`S0)V@-v@1|GnL(`2Uac zJOgj!vV*tCp8;V}-)j96oCK6X-xM43dB?q;j? z>=}527&u2E0Vlzjv|7XA&?NySK5B6)12}62pyvTq6Yz$p(b5F7X*_Z;>GJX&xmAt2 zap*yYc)P9E)dfP}Q}Rth#T(P&o=03Neu)2F@j~5AFeZ`a(SS$XSQxJe-T0Llf|?4Q zB2-P$B>`A|kV#VbTZzg<9#P2#3%&&mXSW^oVmK(yfo>)(rDsFWf*jtnXW(Oi%YcdX zr>>M|#b={Q4~0It@5Dia7YRv#0!9I(l(-l& zjn>nt5wk10q2>%Za$Mmgf?{XE0BC;}pkxV7m>>wy4Vb~-Nd)Fn>(FW)s28#W^_u?* zCJ1c!_!@zE-`RF{K<8&L-*$RVZ!?qI1~LvxgaD=h1}z-$kwAt=J_4f%!D|kMYA--Y z062p|41EAyAv+21pMgsPc6sw);~??E`kQ?IFiqf6iHNB@Xg~i1wt2B$+QJ@h zq1E~iiNO@k<*+>ubUI@ax#H(7i6bBuEEW@%t`3xws=@=1u`%(YEjb7;?QCCmRt@2f z(>DQpDeJrz=Os{%6t{Ar37<$=QJ&E)0m9If@DiDA2tvqZ42B|b11X3QKPO&`k!%Gm z5)K`3GUlTPH+;1M!ax|IIj6|}Eue^r?NZX$76^!1t`??AK4?G!J&W1a7in2k!oJ_) zyF)Ng0c8k!BAax{P{f9pmh-Wcz+0_*aF$F+-E$9)5OpIQ3O1_#xldADFZ$fK?mNw& zManKy-l*BS2Se?tI;FW;npJcm|*@T7WCS{xXo2!v#No3}6njB3$sZfGZBHkUaog zscLy>_LGAvEoE7^?Kk7XyX7g4E-bI!`lnwlE9QzDpsN%1i?Iq)#l_;O?TE~X5O5a;p)-#OgmgIJF4nS-Fy+M(;orBetth`{C0 z->fRX2DrjRF<-mryr^05w8}5x|g2ElR(C;Hn-P)E1B_H%uz|<=N{z4T65y4>WhM(T6b~ZM8yR!> z4o{BGsq`*Io{9h?Kpa`%T9iy?`H3`HM%ljC%chGO6KGM}tc%aFCDj;a`cW!ijzBeM zZO_uN?Aq!0W*2pI;vO7IgIfr6ol3Ur9t?$7@8Z)W;L7KtsKBKMo`)qXp8zB;iN_=Y zDhs_~IkfcR$@wCb9rzg1QYJ8g3v!w$eBv|>42k!Gf-Yiim$yT+qdAIh#Tb;?n!3|*z(4X`HHAxRRCyr>ygT?0XW5f0M@|Y2nM2*XQtoO_Y1iBUXmp1lpaEW=Kt;CZ{jvBaDnJ$KH^;dl2JIV(A@*|QCgoUnO+!kBM;!f!YMDg z$5OaX>zFGGf5rEfut|ae9-|zSyaw6cogy#xQB`4RlvqDoV!C#x`#$2 z_u;01OMF`VDZoYXpH*;O;6Jl)C3EKb5YGzbhyC6gyq0w|gURO`lY`b>RSm~o=LmYgMwp=w zfr}a=vLaknvg^G)2y#Y>&B3(_{xabj;{f{jPdhm(gDb6+FF*VQ+S%aO;oR+n5n)I$ z>TO`3Ld(AA;?3N(qW^nvEJPZG?m1-)L;qHqTAuxK=oH4Ww)vv*=&V$%z6UZ9AOtiX z0S`yXHA~?btavC3aN$6*$yUUF4^FZu6;ok`sLF0}F;7w>BB{AF$K;}_j#&+G(3wd_ zoDqO$VMOL)Tw@Z=xYV`+k*FFD$IGXj&(GLWUFf%h)N^1{@Umco^`aLKQ$A(C!YIH& z-r(ULyu}^@?>}4~Uw|7~WDCTGB(0hf_RoB;a{N4kbn(ek@*QS}Q*sj=K?bWRzF2?J z{ub~^@Pct+3MukHZaAMd#UX$U2*0y91oy(tZpjJD!c}2Na}RjYj?2q-1TYEXsEk6g zMm5Wr&NP6`Vo=irGSj%`9t=k@y&aK-RZyjHSq*CH;bL%-DL?ExJAG$yo6u6YtTrfy@T%dBb@lYy&3&ryVn3vvY&2l#KGlpITwutg=717BPXF=WUp z!VjdR{7R@51@ngVPEmXm%yC4dx?m0?EN>w~NU5}7*A{@QO6tK}coe^mNP+3j!)1kN z9FZA-=~T*n;D8GxFX9f)PJl;T3P=Dx9rNDQZKq9PDsqSrmv$pGMiHk&-3ba%#MuT@ zuL?ulo%fyo%kDE2%;zwY-FD{vj!PmWeT4+QX23%XVxKzTFvo}Jsnheor$Cx`HIsB< z)-Hip1RFDDuP5D4{97Oc?+BfNFzn!!laLde`jB!usV;9A;q{^Cp+p5@>;?NUIEE<> zqxz!eR9!9?20_2pC_ep`9YvD1pu*(NGK#d?w=j@%6nH4A+gJDC0tHk!G&MT-Fz}=T zL%0;FG(KQMmzqW-o=m|f7W!r0b>xN-S#BCn z3dE zIu#-hFp-m9Sq2))W(T~3fd@I?y9MxAYkcMHx{kP%+k^rxiDC|QBA$BV0*REgtEdJc z$>d|s*D8j&TKO$ot=M8!`zwk0k=tn{rUx}kY#6VosPI)gzX~oLRXzvSl3Pql!+t!$ zbr(?*t}|Iw5euI{PA#BMf{DB~)`vltUuoZSCc4qJ;(g-2nC>d0a)Dg1(BE8+pOoj( zM-eOP3Yx*CE3Xl#^sz#P&pQ$W}<@C|Ej;b=yITY#I;i{Xk zdvKH#Rl2fLW8t;JRl^wGn=D`=NvNAV$>7m+0#}eYC0t1aw4ml5KGc+16sI~j;7_1gIYWg8%JG}C?zFVnJX zt}I-u9Qs-dO)FdttTk7?S<2*gtyD~$5wFQatqrawb{871hGwm6f$OZ!!AbRFT3)2` zOX}^UZF&;87Ib~hz~$pHa&KK9y(Oenpa)7dql$tlFRSgKE|?B86GrHQkq~@3mL{W> z8bJo>HSOv{jXELm(loNfOVgFXp941i868V(}9842_r-q z33V;HsV8V9lNl){R$T_^3DTxdwhJIz`QaZiysf?vvVu_RMqLD<+Ku`i46pGGoElX~Zf=n2ZH@CJJbPy{1)s-zd*f}l9i^JrIwqN80S%GnIyAfcW2SqN>=2sWT- zqb4|sM9v)Ji4;173P?dgrzG~hByd=AwUCnKD^aqMhTR60xXl|Dp%VJJrD1D;&TJ6L zRY=xJnEGL{$`JmFnFP10()JG^3tFqtY7?lK`9It3~XdRO#2ss7)h1_~3 zkiK9VaUdpFAepUHCr&wn84)%W{qaARncxWd1bQFOPqWDQH7BES}u6$S9aO9Pb|I0;khV~Is5`aN^u-hK!k;bV1~7bet!HvS0B$$1$yLvzdOEs`{AhM z7Ba=&HV|-t*@5^oYr-aJk}$IpyOoQbGW>zUJ8H!tN7BMvBr(xd>Yd#0@y!!zX63_{#SkPOq4z-SYXoX8>-wZ%UAig;W&Tq{NNv+iKw&6F(j1Jvb#3 z>_Q*N*C~C}lQt3ANpm{$B}ANtaD9|>cCQt@wBnybJ+Tsv@JXuD0Tnksm`Q;wyb5#|t$Y>o5xoGkbctY+C`Z^*rE{V|# z7R4@$BNJFTgZj#r5uietGUI3hsw>K=6hY}luzI}L0nva)_&&nx&x2z&3@RvUQ z1x0vly|F6r6ubstzHrsWV|`J=ySt7#@(eL>MU?bKtvJgj+$sRpNPtDWUL^%4rI$ut zR&g*3Xq8ad4BQovYla3HpuJ9rkk(!c05uRq$q6p;edOj4>_!}jSSz;vl<%QzUMkix z1r)JO#bt`n3tIM5zMf)}cG)cK=B~x7N0r`nXqoqNYQ~8wo-r=&h#iXqQ_Num5EV*N zQ5ec_p!c_!C}Hc06??vdQ)RO$O9rDO>?kNWueFt60#`@Bq47u+hyy_Yrl@i=hd>PBXbdXLYNY))yZ20KC5(J5btvf_xR#{3POp1Z%(;Q$Zss=UmTymKX`Y%C4QY9 zTwMJ6;rwW8EgFE#v|BVhkFVk}rD~fC%`ZWHLg94#rG$G?=S+Z@=pq3_N@^gSrO4Tww0qHzqr zqVn5>0|%HL<;}CJG~mWeS7^b-fm6gvx>LKWD9B9|v#6Flv4<|RV{As%QnTu%b=75X z*+ugjs+PjjfTrof3mw56fJM2y2}NrJ=A!g0h0HD)twzsMkWFf~L9`1M-TdoOwsc1A zin;HWvZVvFbNP=?*-VIR;$Duj>Ed6Fu1Nq~71YsA*^C2emG>TtvZZivYV^M7MpH%E zl9_%yx|X7v##C*_9>(zf;@#yLa3kcQfMMuUr}2eZ3Rg@~^lwxhb5O!{jc&ENG=WP% zUW9*B zZ4FQxtZn%eN}!sotti>4TVVsP25Y+jW_Z}SR@72D(P#DU3 z+VT<8LQzZw`%u6bIdDpS8j|b~s%V`{!+ph8sD5m4N&UP1?o36d&kwk)^|fRJ(r}f# zrxfO^#?@jySt~SLL8+gGk>x`HCa{NO+_q$}E8%>caGlBqS4WyiHq^LumE<)PK)z%J zhw5sD@8_X?-*Ff&5w+Vmm%}SM!kG>dF$(NEU@;CqaN>-$a>57(dMQZ54 zMIeuM1gJ4Z#|qc-R!Rt*V#y$gh&1okYoFk0{sD9ZI~69slNbaYFp4?bB*hC!8pYzn z1d(~7iXOg-ijs84cLc|(j%62wG(4`)xuABz8SyTZoxaN3u^~j8%!ba&03~YMKE*Yz6(!?LrfD0P7rqP6Q1gt^TT!}2dQbaON z23|*ShHgl7T^&>L*EDFRy4dHHI}wr*k8%ilMXa2Kt1_mdDBBt9!o!>=os%oQWJJXZ zsI2i6OQF)G^KgwIqh0>LHNjO9Pq73p1}8R~wR|Tub;5@-*WL!Ig>tFU0JHopMhjGLiGLFI;cnmL{d$}B<9I~@RRPI2` zNpu|fGLA@>|G5^p3dqg|SCu32Js7@pUhXXGL|g_}!JXI!SCu32icZ93a20~b+Tf~k zBwi5#Cudw|$S&72eaIRLKH#%VHI<48(cyDa?XG&8L+$+$bpEd6fb>WV%_y9oHSe_ zwdWjKFJ9l08!#5vQvvgD!C5*@>K2o_5ls97?!mBM9g21lxXLrb-Gei2pJ;Yp-Z~g| zXNX1EU64<+!bROFva6i#!G-u{s%%uh5>ieRvv5rrs|Yi=5M0bx(X}l5tC-4bW9_aQx>Quo>fR{J{gy>p3Yh{P<%NvOB# z`p~$smirA;K4Us7HW2BQ7u{N=aBeW2*g6c2#raElKtjr z2pdPgvcV*G@wo*uaUY(3YXBJzCWy)eev88uA$cMD7E{ivY0@Czw2`S=Nhf0R2DF+ z0^Qu_VLNFQ>ZTSCoM=T1V2J^tbfwxuJ?QDm-JP*LV%3q3t zd1nqINzrnUF9T*BX|sb_niilm6F?E#zesX`+}u>eQO@YRbCa=P=;X{2*3q{zFiXOm zSAnTCHnu?GwL5KDFiWDISAn@WeCq|ic|1uKG97NE@i3^PG29~8=}B>Pp9`Ib-yRKCSWeHHNyxKv38b3 zidA6hdcYMKn->D}#fxQ6P)T;Va;|$3MmrB?-E~<7n4u*|Xkj)rY$7UXd5WVtFzeI| zHDDIeX5q|4OkQt}Zp|IAAecY&eyF^(ssNM0XoA>T2H5J^)Zi|-{L6s}!|*I3jJU)W zd*HzHq|=2_42&`Illzi|_o%Qxswq+gQ^`OVwgi&*X=0Fglz0ehJ(s>k&>R!={SkDp zh4pqdm`k2?6cr4{Rg33nMi|ONR0Bx$vmBVRZyZbx)od6L--APQinInyPjL?T2L-b# zVDWt>5|i!JTSFDsg1H1js#5*jgCNoN=@j2}gXAb4*U6SnXoRVWTdBn~{lA~+snSBhaI@TFzGzIior-OweekOj2A%adD$b9Y}oo}Qj0-qJw6urMO?At&q8sN=QZxZk;<$q%o~C0<0OZG zm|>R$flPwEgxp&PzXeO#=KnLt*O;Y_i|N6-2M4Bwinu+}UuAd_S=Z+qhrc%50VOI` zS2+r6xk-#ZK53|mg4y?&1%9lNy|ocT3NiPPaIX^efNITlhi}fIZRSJ+)e$RfJ)_!= zGrO<6XU-4UR4hiO7Ewgp4vA20$^D}Q!9MQ_>4<*i;P@VR1>Nf#7)i% zrCGKN3tX+hla}XMH6U*ytA{L6d@WCX;`1_ZICWsp*EkT z4p2FdWG*@}Zvv__F+bfojJiG^>1pbkZF^Z}&7Ia>>jLgto0pPXfeFyaCX^ zwe+68Nn@D|4ErAO(I?k91+DW(jNR*wjPo$awcp?F^?z*AhkB!t#nHvp+4+Ze$Cq!9 zKVBpYB<}MH;ACo@e)#uEzWgb<#tLv3`o)zM6jx7N1aF)PX47~SQxv%*;AdY>iS4FTA&eMQE1n^^V6oCJI z@!`GNLH?~c>xQ2k*2VZXeFZ7n-vuaeiHAJ7S2vI|aljiCMVM{24nTlzKnNo^LkvY= z3Mm*N6lBf84+(ojn`g#s4crr>$V zCDPrG3accHyu?gOvvC4+Mu3OuwbMEoOG8ZA5SFt;5d+d1GwocEGx62di5i_~JMB;u zVrgfB;gR!|OhT)rUL2gA$lzkiC|Jb{<{P?Q&Qhx?a)x}HdnW0d0OQggWv?VPr!rSmA@qII* zOD?)kCK8mIWQHONNG#p-NuU_)7F22kn8xtHK}7^+M+75~yV=vJQ7Y8`+G zxnwp&fhXl>Of|bh0;4AoB4d@eMEn#GAuo2d)SV<)$^^^hbJfbW#Mdm2B58ij(M%YT z5JmninBgcA<46N-8KO!W6;rll@L({+zewI>0OtMHH4eN%8eFI~Lkx1V$!V7)deVL~ zVY}0V1eDfLk-0+r>j7x@+b!ML^$#@1ZVbSErv<<#$8mm3Xj%58B;n4gkBTK;XL43# zEXPAa+Y<#N2)@APz3nSk$r8obo4uI6lR8cQz2ILnQ~e!6;+Cpb1uB_WtFj$c;jX~ zI~*jpUqX0bc!rDT2DtXd(ERDa`EiKSQcZeh(O^crjmLwN%sU#G(s-qc*dMCO>Nay1z{xe zT8IM&e0-gTl#!HI2W=5nd?_4!ibdXJ7;@wiVaU=wo7}E20%kA}maKe(8K@?8@`Iko zMHfQZB;uPgjgTlX;C0@&uns9nC@mN%MMWuo$ReX+5ZeKG!8yej;8`N$O9o&}q5&v9 zW-Xap`bhO3PCDMm;UFvI1%L^<6!nZo@?yukTM$lwH!{j3!UVew(IA<=DfBPaS`MaeqAHfJx#DSCZYzowr`ZMD3Syu^I#Kg-#yXyX-D{?;Ec3i0c8ppnG zG%|yqKZGbaBtbw2pa-5M7-GKX?Ee3=B~Lsbsgm0jC0!B(@8IWCJVqSD3_yQ(w^!A) zp7(T-d!qc0y`7(abauADkGaOM({JoL3ve1CT`*ny&rnTS{g$nXl9G|9Mh!31pQRh<;NgjPvEy-eioulD7NK9RmJqk)ZB`;W-$C& znygW&RMMfTToTNY$fm`J)V~TX0MoJyeOC;&Pn6#CTN&Qu2Ei>?}5P<^ZR|;JMSPhj9-=VFt zq#m+0oFadw-O7hjD+gI7{wvKdCGSUsBW7v)>kw*0TF#PT=7gM<5^_K)t>m<|j!XmspNbRZt>01l_4> z#J^t(9Z;=XWH%ut-nEs$6M zVVBX8ipHx&7t*qD=2}&@pchlqd|k0`B?>x?acbOFMIG{K-@mTXdLs=vq7nD#wl>@O z=4#6BY_9nYJ2o-cIdt!yP8bgX?kypSSJos{N2%CI>{u8%c4#O!;9%4 zV2qfGE+I#)NQ<;o)Euah40cQA9Z1TXZHnu{jnziwp*iwNh$7nM6|RJWQ;p{;JAh56 z^&b+8WfV3af@;X?V-kV?9K1VK-ns@6Rl^f_#a!q$-cX1}%Wfyc$KuU|ch)wU z<&@j+GJC>QTQ9cCT4BM?8;c`LffVidauY;o68kXnZv_<6ctm5Iq3(aE&#|&~ z%>`1{2wr^^Hmr`s{Q{#+A~;I-moRNp3^k``B5(|5DpR{=m%Fhe7A{qLDpRe5wuqwLw2-R3-Msd8 zvGz{$+B@0W8Fo_dr}7q*)n$3De3QvWT&&G(wHDClKpq35t>SrCCC^03iW4s})dlC< zwRoC*i;#=(T=DKHc@RiKaep0M#wbyg#`KdHKlJkq{E9|XLaxEb^HUXAbBtABxLjFj z=iRonkYE)=T^J(Zkyv=O#QfBcBIz!n&KL`QFpdKuSUI`282ZfRp}RXUyt_+q2xzMz zJ4$+M0|$KhG6Y|~yvR3z8DHA+7Ni}fQD2hJsXXN}D-`xI!wO<~*o$Z~%O$~xvm5^e zCxM^lxe!)b%4o;n5|vwwfmF4KA+Vap3l48i@X(?I*;-8H4-sw|&sAPsq4eP&5*?ki z*^CG=3^%h7WNC332*Q#~LIDFI+9DC4MF3;Y03W;%JY*~tg|(#^f^(7xO1Wh3YjDZd z0XS*^a^{qXPn8tI=TStAL~2Mk zdT+>i&;ydk+S3D0ePbD6Q0W#`R_}zO4~E*#Q8-c}&jH7rj)|!k!$*M!;!xh@XC_CB z+!+^$%z;zVYLaF_iTrPJ$rH>K0u=v=5|xd%Q^!<;$EDpHt+Lbz3JjgsVyWLCHWYHn z6b7CzvVCyYmd4{T*4S-EC=)A3rQ!TrNx_BbUjmxa7K?)$Mre#bgAKjEH(RZf0FcO& z0oLsmRUPC^oGoc-IOui_KN#KE&zoB^!hXi8lM&8^X#+|@$DEXO^RQJ`X^f1J`aPo<9kq z_`f!se{SX-hpvC1iNlbG=B+BmR?Y&_WE*@+QJ04zy0mUMgCDxjQatIQXj3UXW%Q@i)K+lpmS9pj zT;szD*f8QC>A6xS;;LQgCbcQl<3^SyeF<=qnUO)M!`depv#BGZe_{8E@L#{B+ZX(# z_pfwKy~6y}^Q$X9-n%YhZdY~LUYu+u2ikV@AvnA^0bazJeIT46BE%w$ z=IIX7U41(EGma4WH@XGC2o>>}YS+LM!c|0|2~k*&wj-|4J1)f?*;RrLI*^y!Y`9?A z<^%(S3bP!=f$rBfr9zCNSPLr>o?O$8OR>XIIwr}LN}&;;7>fhOB0P#2G6W#mnx3FT zMYv|S$+)$o6A%xb#UmfnsT}k<8V_<}9K4=)+z4s!KfyNhb=L6-z?mOUIwyfz-yEBX zbPmr?Q8CsG4kp&stBy0;9rmXZr7f-2cgtIfudx%SxN`(Co)(AB5Y0gA1v4sTD44(w@EuKul!5m^A zD(7rY&*DJXCMB|5q8Ph@A|D`#;=te_0R&Ut4m8`*w?L-aRL)dO>W`r3bAWYtrBM=b z0u1#><=QQ0cF21Y9NExd$%ZYZggT&r%REwP%RALso+%QUOgXwW2x3WVax^#awk0G6 zoe6jbrAe=5qYKb80?0_10z=;yMNZDBD&5jn%iupG*3sLQ?+qAo;ah|b{PJZul*Eh~ z*)|1;LfMwa_Hs%|y!5_Q%Xej}y^3rq08@AsA{5a9NMY#cp$&fQ{kW@(`NB>13VNPk zdcpI%vkymCCud(WrJy(G9a2V<^S#V_j>>6@4oEycXnUy@9+Ji*WZ7+v-1zoTjMxOpO`Wj{iC;XU-`j( zC(ko*ilmnzrc$z%xf=y4qH&}$#vh(=ghr*Bn*un297$%<5*BpT8L9AVkMrrVKlby_ zacoBiL0TCL)SUl*zqgY;|J%LZ%ih!Ze~d>jvTz?E-a)|df$<6m6%mReJFdAx9^ z$cs69c`^w|^7>dLITZJ-`g(!4rdqm0(agLn@XgRj*-L~OWbX9Tyc$m)7=}(>Pb3+& z)aQmF%4(}++;mLu2NDvWOm2mt>lk7`j!;Jio^eKq!CW(ck|4F`y+L>~Wy1YJ$sp1m ze$iPgotn~#S!!CEPD*A|I-i`zf%&dKm}^>|Mu?7eh&Zl z`}_M(`2R7UY|I-&Um=mq?6w1MkUjd%GjO0YeUDWd+h;IqiLC$JxtXT7NNq zWe1Uo0F;Bf&E>_y*ovbApv`%1b|PZ+CovEMgl-vN|S9V;#$5> zu~+dGyCgUyK8bi=(PRWSdRt(--`@h;dwW};ziI1I&#w{k9{e{r5LT$R@_Id!t&}$s z_>%ekoAK*QX8yHHiDf)(8KI{=072~g*)I_cr`!yc z)Mi9|YgaZKV)q(F#l|Bvju4$*>EwQ5Ozl0|u7;8`T)f}hoakU!NAYZSTN4fI#B=}i zKZN{x-839lX(XtxjkwqJG@}123V?O`pZ&c&{qJq}_nzqgV>~O+|0Kv6A8&GNqJy0E ztB+68n!YlTG57S_hJ3cd!e1jV&RSByOEINL~iI!XIcvVg_tu7fp>24kk*p5gwZa z2TU0}nGsJ@rvYFFyETn(K=CU1YcF}`sW|`H3)hE}|M&VY_lxrX)BVq*JR$K?rJ@YG zddf-H!2ldU2w~k_KL0(JH0}KO{n7FH$@@1~2j_1t2B6bXG_o@Z8T1_vIw=}`P9 z`l-MDL+|s;y&rNF-+eeb9_V{<5ga1}Uk1Wg1Mq4z8jV^2Wa7Qcf@W$HgA5y~M3T3O zeXmh7A=7RcOAcqIUd+&pM7Q$YJDlix5zW=vD^}T)74p%gQ)DYJj&b5d8O>A7Sa^Gc zqXD@4l5HeK+Pozbr?Oet)vamslgf_Ds6h&1Qb*!@Jwl^X5zabtyD$uuU0{l_4J4Us z$Sh-!k{k|7I<*=N*`P#`^20!NpdyX~HI#-8SRRBzi#hZz4VpfPfE|jI=IrcvrM%af z+$X8YUAk6fQy^LE)C0UYouzs=Ws*@8%%?<3rY``QD6zZ}JyqK!zK>j{!oa*s+OZZ^hxbTbT)&Fl zYa}(2{VlM!1zv7}AGbiizXkd`TcE$U1-5%zV5j$6RtcDg-Y1iZO(DZSp*a$>g@Z9E zY8zb=P~sOCM||YVTCSdRiBBT@C$F0g-lc%723aS;w80ExHoYX*NGL%&h^bveb3H~UcB(dG9tIJm$t#<^8S$isQXaAV zu|~7lhYTruzC#j7>uqB(GQ}0WFF7vCsn%ZCfMq13k`)is41PW&fecjZ^4+~xD8hFG ziRFw|ACftWMuei%(;yf`~OI6Qv);q>VE92_2>U!J@^IXt*L*2-8r|MXw(=Jmh3 z+fVwRM|mbb89`slbxHDl5x3E6oI@!Md(gSN1I{mUv&fbmqgn9fOD5;j*B<-RQ+fWA z+&?Z_%>uaY{P*(rKmC_Gy_Zkt|1ln8kwxeS8RdEI7S8OPxv!>W4n!!J3x4QzL=qF|Uk=Xu={0bo?ff^o&G-T? zHyHU`quYU)(zMI%ytCp%^tauzL_&X>m2a1NTJV2e9=?V9sOA59+j;r#<;$o0zejoQ zzI<&6o}RDy80SAtt=T=~Z&r2tOrDzazq6A+|NXt~y(jzsM|pCQcwofv7`h@X047}@ zyC|S&01iUvPElvu>DdYe*Hqs%>KjQ2E#1uo`XOo#I$@)ARZ`jbJ} zBOORO4U%Gf6t)FAUup`TZM-Vys_?7Ij6BV;;Pm^>KeLcr$0H=t;G&MYEiJ%PEV5Rp z_MGhkR7Ohy@lXeSmJ!Yiu$2{DD}=kw&dUl2!Xm7B0zcR_5iZ7xWPX3rtauKNY|C4lj#hn&1EC`wZ zkK{NLR7N<@i?aID`65lR(+X!y3Y;a96C(a6XF?JdC8WfQa4IG3RaauKkgNtv=$%K2 zmKAmySX$KC3d_o>ZGBp7>tN~fQt57hR%y{3G8XGts=adSJc%AYj)tGFX!VzP!u;Q6 z;(pktmH!()dbqduxXS-M8Q$@KxAJU3LZ3luTwMkxY$3X*FR48@D?$I#PI_DDg`1*M zhpTiBieBlLjUm$MKnDz!CI@>25aJYbf-W$p>d+M?T5ygbR|L&$Xvs2ka<9k7I0iRW zh}UG@?Q)i!7|*5B0kje{M|-gzEz;Cv+<_Z$TLo9y*RB%zB27(zw?2g3dI6zkZ3v_N zIEL7QI+c$Oy*tlYMlvZPJ1EeT;{APrCW|z^LccF?sv}0J+EH(7tLH-%)u54HFH1d? z37PEj?dZct?WMA{h2G{8GJ>!P!VAg3+fi{2Ka}ZB_m2+Uglv(mdRV3uV;6*TnvhJ# zoKi?d1-KIrZ}1A!D*=+)2_g+d)K;UInXN74M0eCd^TE$vO31UzAY4=#fL<|IEPG8P zHh#g-`>ieXTJsj7)Kige88{K;-S zPkjClvP$a2x`eB_F0Rp-^uIkA@gI+0*$B@1tEpo{wU_CL1ylgm6Y>F@;Mp1s$u88e z3)h72?<}>goMbDSp^DIfjxj2TL|DN3(v%u*w3Mm^M&`Wt185eMS#kAa1 zCvRWB{AvH#yhVDAj38b;{8=e}w10B)^V{PWe)&>Eyw#wj-SX75^D8Gn})wHfYkocH%N*&!Ujk{6K=t`04#Dg8UVVJ)e2o8 z0flox!Hd#;xRhvqU*Idn<}cX?)m5$mNjx2)i@oUCH)6)J zfxcy9>4(BelzC+q5IcT+Wh+16WQw_tE5|0PF%rq#knW~MUz3!l0ku~^WHW!D4HLb1_!#qT|3S2zb$^GzFBIf(D;tX z+Ev>b!Ks}6QND(a>h57UVid1SM(F!@?~W`LA)Er6S-AvEGpcx~7bL}3#n*=#vjYFy zMh!lgG4tB4FvYivQ0R0j@?Y2+RItAE1+3b-3NNdLAbc=g(1h$Nrh6fYldr)g2<0s= z*aju1y|v}8e-l=`b^-$w8dbw^PWHb-Q{C8b z&5sI#Fgnog_{S_I=z*FL=fyV^@+WNT@&nro$ut}^K$kX^A;QS1VdHt*v> zdG)_`gtxZPn{P%%?$t|^3`T>}p=1`xa+<(yv7l$bPx24}b`%vD=zG_TlV$E2i1e33!22oru*voSE$W?PwgRS$^2 zb9_#OL;ev&Cm!hN!H^mGa>h`?fM0}2yavDh0zIMcFkj+KdXOYTnj<5K!LU+}*k$Gn z>W-W9%G|HGqzQ?ff&O2%K(Bv1d53T+AO^5IDMna;q>@FbV3QyoZb2tH(s7xqaeU?M zqp}^xIfFH=!E_fKBOUS(T1G!R?hnxNF z%yNML{DEA3eWvGId#796YG~LG5b$yAoFVo+)IyccI0EnA9&64#0V4Qze<5d(RX`mE zHoF~j1EVZ$S*Ib$IeUNQHf>tA!W)!&sEc_&Mim>Av^X zUPB$BcLzt#7ysN3jjUmdi&gi|s2ue@27_41hA)K4J!R8@L3)!a)RyJs!yln9)CL$B zS1WyMqQf#M*Pl)nY5VKCsTE~PPP@oOr(M>d;ybq|XKj_v(40?3*^tvg$PsubJI;F3)O5;8<`pBkg(-*K7u? zrXq~$I0s@~com=tC8^P&m7X<7=bG2TB&bb{2zh(@l8UyZO;^$|e)BoblHxekKdsch zwN)q1>Of#1lcrdRHx6%I``{=z^)YPN*H<(<7f2oCF(etgoDoRuOEYa|FeyD3(z4RR zCKMPGM6=&WEQv;0n;CMJ+;32w?&zqyHV2=$$|QcT2J!`Dyg=ajV)(wF_mAHkzWHus za@DB`R}-9xA`?9DOlGFUIn8rTM}GKH=+3J-z8FTsN6}uvK9O3_OI^X%7NjHl@%WX^ z#FF$8YvuzqGA+27s-w{(q}c{lDdD|CPjLX4+aS-~f50IB7P~+ENg#tgK;|WsW(X z6G^z(Q617}onm|R-W*H_mJ3dv<0*j*Qu?FNshJ{p8*w5P!^g5{j^7U?os;K}AUupN zY+`myXiA=gqEWway>G4{Z>+)TTZL>Xt?0gh#uL8SdzrACHIgZvkoYQ2tp&jS2Aui; z=Ts+KQ;VFlR5;QeB=dAKzFiPd_<&|ZZ@z*|*j$GgGTewoJVUt90E0k$zvbxE+=`l( z&P!2YEH;51^!a5ntQIz+5WcQ@M^Pm+!zgR6i=tY7Ap20LgkHR7?$tZC7OW#Fq0|dEfF!tIYBxLq;OV~JDs{v z5rKAuhI1Dpq|IAolBVk5(*TW<5*DvOY!QNI0!fU}a8ER22=-6Ka6`IW&upQ;P?l9u z+uCYy>jO2Y!;_i}jebhTN`gk680-|*WNBGe?|xRVjG^Qvn&f(3ofeM?c3H;FwWyY& zZA$%Z1VFXn*g~GN$tfg9g*iLb!O2EwPi!0y@*X4-ZkZ9A2G=B&yr8&#nAkdYUlshI zh2?Z+j^Cd;?;#xPp6SH_NR^seLr52{Ek3V|*<|Rf|qS97TuX4h1zadS=}y>JJbq}$BtI!I6*!NZKo15agTtVZ+? z&35N3@!It(I9sE&*FROPswKRTMYS8W5S-2+6mtz!J%G~p=3B)sTP_cX#x_VOFBV4C z7VG$p#)VlmWRX?xEH2CH)(x_)cH}C)g@G%Gbxm0i#?`SMH$knVxok2{+%&m89o}c6 z_C07OO*Q>hi%3?2Ff~Pyy}J-(Y}Q00#T6RdC>WlvK4d2GDx~{voMktyXVZYW!TQt~ z4}y?`V^+F~6>PuO7Z+v)D$s`y(XrO5N2-yJ9|s>kR1{UkoLQz^7u1cwNGqZlETvTf z+T2w*JMNYmI6d{CG1A~9LvP-`Ly$>U@etVQg;jQz8kstvNuR(-bj%GaYe~!eTSRNDa*%DyFPz#c`t8 zSlbeW&kOX`>IJNp#w_um2UlTMQOoOF#$Ai=*U!-~8h!~9yU{(B$^c?98W@ozB^;Ol zM{%SJxH>Y`$_XJ29v0)2O?MyeJ>1*fd%U~%aF@>2QBC0N8%Q>gGcvHP;+zg>HYfoM z6t25l21A-np*ZATAF#;)XSS-%2W(+ z47aw`RNwD{TGa=&=1ka_7sFVn9ticuKj_?%>k@1+WW^zINM9dNT!>SgrA zF9aEs=yX^D?_Qnw{dJtjRy(iVB_Px40;nFY#+A*SPcjuwkDQ=jrByGyf-FN0eR@%9 zKTo7O)23GtlLO>`{aZuTw}Fm{?ur%En2`G^#gN=wXL>f)R+34$9w(V*CICM~Ghq*) z*frNVaIE|b9iaIL^$YuFzvuQA2GO@-_MHf4_LzOcRD&!f+@wDOlcDasC{hE0Yyu${ zwT}s=h2*l#xh`f|OmZo9OK1afK|vF@3xEbVO9mwhyMc9Mbfzv<1twYphH~7DNO(qi(%Mt3oVI~hE`SS>}N;)szFr5sMJQzk#fk13~qu0u# zD5t?D1v>-$U`od<0L3XD1_cc8qXPCMQ6n}(2o*C8!vx#lhENaS#t zBxaeS>m_;PWD>1U-VF4ChCXnh@Ak?JrYG?Ap#13QF^*XBf(pLKbp-GPJO_<~bb(mR_8~v{o30tw8*p;?&^po zFU`Rp%@^6gxsdC+bSLB8tpWw0o_3Z7IbdP1 zZ$@*34kzF1Z+c{0&(sXz6wJQTp%**a%Gei@@$-JvZMt+%tVv4=+S+T)>CAkQN}8u+ zP_$Ie;8xi@a7J)#sAAUWB5!KZADEfl1^C_`6;P}~?;yo=?!Yn^kyLRotGuu}vbGyW zP5zXY5wu1s%OjSZv9~p17Y(a>0BaD?UWp|z+2!rOE$fC3A5Jt6S(VKjm|UVjLw`8J^QvpGbl-drx8|=kjtyOtuW)yIiC8T47e^ zay}+F0ei~ZEGDaWw=RdKbrH1t;B}L|SB6^*Hmnf$!Wud@DO?*ZfAzp{daW9%c?TW+ zbWjK~c5M~{jd9F23)G&?W46c=BfnX%V@&7ktvFCeLaYW?w$_x{(NwD1s{flekgkj9MU=xqM!zWTaCC?rGuK|fTj-`_DNf5SzI`u-|R>i}<& zStZB=$J=Kcfe2I>WOfwKRc#KmW{4I&QbM!bLGg?+VV$xHH;r<4M+iNb<_{Hby#Mow z>IKdc%o9{f$q0S-!^<7@8T}`Eo0IHB;`n^qGx&Sx=(HCrPmI0&peRFVUi9KMrBtC2 z`e*Ik*XxaD4bCdVIX8^7GaLb2ojrB37?+*`Y|?#*z!80sS!;UXoTg7C-ROOxH(}X= zN%{syptp$}13@uc;X@T!Nv_^FT6Tx(dtYtp@Ip!p=Ntfn#2GD>Zgx^@oBCXJeRuPtH^6Ge*kJ+vx; zex+WzPJ&KS>&i9Fpq@f_5>-g=P4PU!bNmO(@TCCFDrc!514uXhs)&`aQmZwaj8g-W za=+FwgKFwUt=4d&3b`WGiLr0ciQ-~eQeI*;@`n;Wy(nMRgMCG{jTR`Blr^x4i;R%r z3R?BZAYpTZ*9WZ^@Y$t}A-9F|9P>Hj8wM*#9J6`8X{c8smT3Xh{(($uk*yHnG@n%! z2%wE6$q{E0s&80{5xW1y{nenD6Ul)xLiZDJz_#B?h<0#GkO9R;5F%e>lFms9(h)q- zNh!vt6>bzz2Ra%IA&K3EQ&`Fz$r;Yj5Iqn?5x10GZkIk!WmP4(8x`kbJ|Q_M#Rn3wmSus~4r^iPJg;mqRNH0^oP;*U1%)mNl?#$wh ziNWs+!f6pMM1d)ED7g#r;z7qvdei7flKcR5)fMS3$Lf@{1TU zdy}`>i4^)z9SOwy06ErBLL}jHnh|>!#X=Dt4o5#7=!+5H1`QlMUl3s3^)^?ppy%5# zs4)hsP{J+?u|;7}I8i&$cMvCvC6e@lq*qZXd21ElKcbbn(^8LFdMPNLwPmCJM})ouiw@o#mL(JwjM+8>BP-%U7@9Q5Q=E~a zrSTQOi{XevCJJwirMdN`ZL@CY3ozBqb zM+*L|jO*)0_Qwdp-Rz=JZo*CVQ1PcDxc)>qGzh}l{z5WfeXP_Mpp)U4t+=RZ|Js=crfVx~dvu-71p2;2K-=lAa zURVRJ*lwZ+TWFCdSQ4Y5xX>VB9?ytZA``<%T*Q&toYo+048=~t`Bju; zBGND-v6xJ#b->deXg*l&8JIBXx(Hbpk%#@BUyvl3*el$E+T&jZ^ctcP>I$0Vi)>%~ zCq+@u1*WYor$lbyd|J6$9xk0(C)YKZ6%dokDsDl}^fqfdc zL&zhx%_!JGPB9DO@W@l9+6$A1N63VHAi}?RTt5Y^eBj!kSJ#&s=e&W*_D{2^eYd++ zTw11i)pHZxA09cu#)`<&_A$(}z7Zs9;gw~~&UL9|&Mp8Sg9oRuE zT^TnhK)K>mP%G zSg3BT8;@irx)3;qSSgK?OAE;urxPr(7GxYT;@O5auVeSm}fWX=+#g-v~8OgWB;VP6Yb zqx}Az`dneVAMFL$Rj|!C!Z1`p53DuFhPBXBMfpkopNWxr3E*SH1=rUIHx zPr-R^*BTFsH__ERHHXa>nyPcw2@0h*&|xN|qQ|0ev4xtx$1OlD%f8V1k}@(7d7twU zTIA^YbF}AunJ_*=Pv+jcv@oZY?-U)qAROp>Zf;{1pDG!2+*)2*kNY1!pnDp6L=CR! z<3~l^eq&?gd{sX`enc}Yv}G(|mzfDPW_6KI(41xqbxz)|V!D{mu@cYj+W|oh4J)ws zjUxHw(ecZJ{dX^4pcgN{-T(2`JM`23s~=zPAmiRh-YXJHWqPG%Mx722ogG%@4$5^@ zG*HOarr6HAt06unG2_XNtU3;$y=7N<<*vI;c0JHc&n{Tgb^ofpGb)gGiYRAs#7o zI{Oj2c$D34$CA9ynJ~dqO?ibR$&#GWOcr5D%o24Z2gkao8BlQQwsUn> zKseKJKGkT$7t;@W69eujofA@6R8628+ddY8#S~;P!<~naDUeuL51Rs<-CSzvn+sxR zK=;nf)g>k57$$;d)08-4;TX4sjjy8bsHdqKrtkL~3P*umci*`Rh;wR2`+l$3R^|6o zk%YIRc2Ousr{@@GtdY`2=1}9n;DWg7Y@~x=YW3j``fVX(;hzt)ruGo{up`MFs4!G# zQAsC^@*IG=&BIW>dv${T6ODCD0SPg4A#93Sm^SW+FRDP)n^-PzYBsh5wOj|`Of{nG*{AD1l zOgY^D_X*g6VZ5_m(YJ-COX?$u=wUgQDxH7?%opLgz zEc-L?g^AEx%;Q<;y|r|n%B#FfyMM5ZY`0l1YAT<){od7my(Afk@fqnRDYv*F#uv}WBk=p2k;2wgj0(VvA4M3Rf#NO2uH{Jm)Ou6ibp`wf2kS`RPB4JO^L5 zObJ-Z%8UrEZ7eGiR6zEd?5ZrcL^d+$4~m_2zaS|ymN8x&38niYx~5s`n(9!i@hmsv zX*s5fn&>sPnwryN**1#DsoJAqWIO_Xto!?afFzb5Y7dM)uwxI?f~O*{SGv7cNX{m-hS}kqtX8*|JeTT|9YVQYv!u%4M+DL zZh$%MFlCCK)wn?qnNw5h-f)8$HE_%;SXb7aC_k8I_l8}|&AM&7;cUCNoU^2fonN8b zj(8bl*-~~{=iB~EhYG|(Uj{KbRv{oPN)>!w9&7rhm%c1yJ|df#<1fShX%=k-9?$WQE25YSw|5i)E^JHHAR;1H;v?u;bj9&}Gb-Uc(9m7?_`;OtdW4P`Zt~I;u zFkIe|;OMIJg_upTB$xPV!*aas_^lCh2+Ob<8FzYLa;7E^w9l^Cj9O!e$e`nHB3Wev&@%Z<~G@eoK^aN#%Ih;pCjS3q6;a~ z*Xhw}G$v=sQu_^V$;Rln;!>{VMfF>W=>He}d2f6BLu>!wTSdYWjD+N1wPC>h*n3koZp&#q8VD%st^X~*cALOhDqdBu{#{H)ix`muZF`X zYDMm=C86&I7ON>GTvo4Rz{d+l6O=4;8cq{-O~9QrCuc-&%L+*}L*MW2yMYeD;jQ#d zvVl@^%}ArP(_GB!bP++t-^^dfHLmW{nk|1hC*e}G8ymH4<@=v&5PCq=q~UHnK& zz`cno_r`uPktj{F`Z><%1VX+Sc~8LSFhiJge5H5vYg?#DIX*ytnqj~>{YhU>8=;G# zd3jE=WaMXL(K{L^SYo?-%t#l@u^L`&0F9f^Zbg&zegN-^Oy0dM)1tk!$0kTD#_<~c4s7=YoNl0 zAE()_dExJbt)BaP)T@M8_E6R0ECXT>+doZ7(s>aT;o*w}88)c1&hYq9XLxk7Q{8aX z>a8sh{ss^>cVI>2Tc3|Q4r*5FZ)>*%@-li)bge9|d!9eVr;aK6-r-^G4I zgTIB|=9X7Ajy-;oI`ekFszMC-==gq0|91EM%RcABz51fx)jKN!-4%|06D+$<8+&_u zdru!fhX42W_A38B9X@>a_}_+)pY07FJ%0A|>63rk8$NyZ?CHOuz0GyA`B?~w`M>R5 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 literal 0 HcmV?d00001 diff --git a/assets/new-relic/nri-bundle-5.0.94.tgz b/assets/new-relic/nri-bundle-5.0.94.tgz new file mode 100644 index 0000000000000000000000000000000000000000..0fc7a9e537e04fbc6d59c126b0e9f200b0348a1a GIT binary patch literal 353607 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

~c(z&#Gtzjnc^2u5ncEY|-&(HO7i^Pf{;RkQp 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 Dk#qB#J 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 z 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 Jt`-2)0{}S=!_xo& literal 0 HcmV?d00001 diff --git a/charts/jfrog/artifactory-ha/107.90.14/.helmignore b/charts/jfrog/artifactory-ha/107.90.14/.helmignore new file mode 100644 index 0000000000..b6e97f07fb --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/.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 + +tests/ \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/CHANGELOG.md b/charts/jfrog/artifactory-ha/107.90.14/CHANGELOG.md new file mode 100644 index 0000000000..5edb3eebb0 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/CHANGELOG.md @@ -0,0 +1,1466 @@ +# JFrog Artifactory-ha Chart Changelog +All changes to this chart will be documented in this file. + +## [107.90.14] - July 18, 2024 +* Fixed #adding colon in image registry which breaks deployment [GH-1892](https://github.com/jfrog/charts/pull/1892) +* Added new `nginx.hosts` to use Nginx server_name directive instead of `ingress.hosts` +* Added a deprecation notice of ingress.hosts when `ngnix.enabled` is true +* Added new evidence service +* Corrected database connection values based on sizing +* **IMPORTANT** +* Separate access from artifactory tomcat to run on its own dedicated tomcat + * With this change access will be running in its own dedicated container + * This will give the ability to control resources and java options specific to access + Can be done by passing the following, + `access.javaOpts.other` + `access.resources` + `access.extraEnvironmentVariables` +* Updating the example link for downloading the DB driver +* Added Binary Provider recommendations + +## [107.89.0] - May 30, 2024 +* Fix the indentation of the commented-out sections in the values.yaml file + +## [107.88.0] - May 29, 2024 +* **IMPORTANT** +* Refactored `nginx.artifactoryConf` and `nginx.mainConf` configuration (moved to files/nginx-artifactory-conf.yaml and files/nginx-main-conf.yaml instead of keys in values.yaml) + +## [107.87.0] - May 29, 2024 +* Renamed `.Values.artifactory.openMetrics` to `.Values.artifactory.metrics` +* Align all liveness and readiness probes (Removed hard-coded values) + +## [107.85.0] - May 29, 2024 +* Changed `migration.enabled` to false by default. For 6.x to 7.x migration, this flag needs to be set to `true` + +## [107.84.0] - May 29, 2024 +* Added image section for `initContainers` instead of `initContainerImage` +* Renamed `router.image.imagePullPolicy` to `router.image.pullPolicy` +* Removed loggers.image section +* Added support for `global.verisons.initContainers` to override `initContainers.image.tag` +* Fixed an issue with extraSystemYaml merge +* **IMPORTANT** +* Renamed `artifactory.setSecurityContext` to `artifactory.podSecurityContext` +* Renamed `artifactory.uid` to `artifactory.podSecurityContext.runAsUser` +* Renamed `artifactory.gid` to `artifactory.podSecurityContext.runAsGroup` and `artifactory.podSecurityContext.fsGroup` +* Renamed `artifactory.fsGroupChangePolicy` to `artifactory.podSecurityContext.fsGroupChangePolicy` +* Renamed `artifactory.seLinuxOptions` to `artifactory.podSecurityContext.seLinuxOptions` +* Added flag `allowNonPostgresql` defaults to false +* Update postgresql tag version to `15.6.0-debian-12-r5` +* Added a check if `initContainerImage` exists +* Fixed a wrong imagePullPolicy configuration +* Fixed an issue to generate unified secret to support artifactory fullname [GH-1882](https://github.com/jfrog/charts/issues/1882) +* Fixed an issue template render on loggers [GH-1883](https://github.com/jfrog/charts/issues/1883) +* Override metadata and observability image tag with `global.verisons.artifactory` value +* Fixed resource constraints for "setup" initContainer of nginx deployment [GH-962] (https://github.com/jfrog/charts/issues/962) +* Added .Values.artifactory.unifiedSecretsPrependReleaseName` for unified secret to prepend release name +* Fixed maxCacheSize and cacheProviderDir mix up under azure-blob-storage-v2-direct template in binarystore.xml + +## [107.83.0] - Mar 12, 2024 +* Added image section for `metadata` and `observability` + +## [107.82.0] - Mar 04, 2024 +* Added `disableRouterBypass` flag as experimental feature, to disable the artifactoryPath /artifactory/ and route all traffic through the Router. +* Removed Replicator Service + +## [107.81.0] - Feb 20, 2024 +* **IMPORTANT** +* Refactored systemYaml configuration (moved to files/system.yaml instead of key in values.yaml) +* Added ability to provide `extraSystemYaml` configuration in values.yaml which will merge with the existing system yaml when `systemYamlOverride` is not given [GH-1848](https://github.com/jfrog/charts/pull/1848) +* Added option to modify the new cache configs, maxFileSizeLimit and skipDuringUpload +* Added IPV4/IPV6 Dualstack flag support for Artifactory and nginx service +* Added `singleStackIPv6Cluster` flag, which manages the Nginx configuration to enable listening on IPv6 and proxying +* Fixing broken link for creating additional kubernetes resources. Refer [here](https://github.com/jfrog/log-analytics-prometheus/blob/master/helm/artifactory-ha-values.yaml) +* Refactored installerInfo configuration (moved to files/installer-info.json instead of key in values.yaml) + +## [107.80.0] - Feb 20, 2024 +* Updated README.md to create a namespace using `--create-namespace` as part of helm install + +## [107.79.0] - Feb 20, 2024 +* **IMPORTANT** +* Added `unifiedSecretInstallation` flag which enables single unified secret holding all internal (chart) secrets to `true` by default +* Added support for azure-blob-storage-v2-direct config +* Added option to set Nginx to write access_log to container STDOUT +* **Important change:** +* Update postgresql tag version to `15.2.0-debian-11-r23` +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default bundles PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x/12.x/13.x's postgresql.image.tag, previous postgresql.persistence.size and databaseUpgradeReady=true + +## [107.77.0] - April 22, 2024 +* Removed integration service +* Added recommended postgresql sizing configurations under sizing directory +* Updated artifactory-federation (probes, port, embedded mode) +* **IMPORTANT** +* setSecurityContext has been renamed to podSecurityContext. +* Moved podSecurityContext to values.yaml +* Fixing broken nginx port [GH-1860](https://github.com/jfrog/charts/issues/1860) +* Added nginx.customCommand to use custom commands for the nginx container + +## [107.76.0] - Dec 13, 2023 +* Added connectionTimeout and socketTimeout paramaters under AWSS3 binarystore section +* Reduced nginx startupProbe initialDelaySeconds + +## [107.74.0] - Nov 30, 2023 +* Added recommended sizing configurations under sizing directory, please refer [here](README.md/#apply-sizing-configurations-to-the-chart) +* **IMPORTANT** +* Added min kubeVersion ">= 1.19.0-0" in chart.yaml + +## [107.70.0] - Nov 30, 2023 +* Fixed - StatefulSet pod annotations changed from range to toYaml [GH-1828](https://github.com/jfrog/charts/issues/1828) +* Fixed - Invalid format for awsS3V3 `multiPartLimit,multipartElementSize` in binarystore.xml +* Fixed - Artifactory primary service condition +* Fixed - SecurityContext with runAsGroup in artifactory-ha [GH-1838](https://github.com/jfrog/charts/issues/1838) +* Added support for custom labels in the Nginx pods [GH-1836](https://github.com/jfrog/charts/pull/1836) +* Added podSecurityContext and containerSecurityContext for nginx +* Added support for nginx on openshift, set `podSecurityContext` and `containerSecurityContext` to false +* Renamed nginx internalPort 80,443 to 8080,8443 to support openshift + +## [107.69.0] - Sep 18, 2023 +* Adjust rtfs context +* Fixed - Metadata service does not respect customVolumeMounts for DB CAs [GH-1815](https://github.com/jfrog/charts/issues/1815) + +## [107.68.8] - Sep 18, 2023 +* Reverted - Enabled `unifiedSecretInstallation` by default [GH-1819](https://github.com/jfrog/charts/issues/1819) +* Removed unused `artifactory.javaOpts` from values.yaml +* Removed openshift condition check from NOTES.txt +* Fixed an issue with artifactory node replicaCount [GH-1808](https://github.com/jfrog/charts/issues/1808) + +## [107.68.7] - Aug 28, 2023 +* Enabled `unifiedSecretInstallation` by default +* Removed unused `artifactory.javaOpts` from values.yaml + +## [107.67.0] - Aug 28, 2023 +* Add 'extraJavaOpts' and 'port' values to federation service + +## [107.66.0] - Aug 28, 2023 +* Added federation service container in artifactory +* Add rtfs service to ingress in artifactory + +## [107.64.0] - Aug 28,2023 +* Added support to configure event.webhooks within generated system.yaml +* Fixed an issue to generate ssl certificate should support artifactory-ha fullname +* Added 'multiPartLimit' and 'multipartElementSize' parameters to awsS3V3 binary providers. +* Increased default Artifactory Tomcat acceptCount config to 400 +* Fixed Illegal Strict-Transport-Security header in nginx config + +## [107.63.0] - Aug 28, 2023 +* Added support for Openshift by adding the securityContext in container level. +* **IMPORTANT** +* Disable securityContext in container and pod level to deploy postgres on openshift. +* Fixed support for fsGroup in non openshift environment and runAsGroup in openshift environment. +* Fixed - Helm Template Error when using artifactory.loggers [GH-1791](https://github.com/jfrog/charts/issues/1791) +* Removed the nginx disable condition for openshift +* Fixed jfconnect disabling as micro-service on splitcontainers [GH-1806](https://github.com/jfrog/charts/issues/1806) + +## [107.62.0] - Jun 5, 2023 +* Added support for 'port' and 'useHttp' parameters for s3-storage-v3 binary provider [GH-1767](https://github.com/jfrog/charts/issues/1767) + +## [107.61.0] - May 31, 2023 +* Added new binary provider `google-storage-v2-direct` + +## [107.60.0] - May 31, 2023 +* Enabled `splitServicesToContainers` to true by default +* Updated the recommended values for small, medium and large installations to support the 'splitServicesToContainers' + +## [107.59.0] - May 31, 2023 +* Fixed reference of `terminationGracePeriodSeconds` +* **Breaking change** +* Updated the defaults of replicaCount (Values.artifactory.primary.replicaCount and Values.artifactory.node.replicaCount) to support Cloud-Native High Availability. Refer [Cloud-Native High Availability](https://jfrog.com/help/r/jfrog-installation-setup-documentation/cloud-native-high-availability) +* Updated the values of the recommended resources - values-small, values-medium and values-large according to the Cloud-Native HA support. +* **IMPORTANT** +* In the absence of custom parameters for primary.replicaCount and node.replicaCount on your deployment, it is recommended to specify the current values explicitly to prevent any undesired changes to the deployment structure. +* Please be advised that the configuration for resources allocation (requests, limits, javaOpts, affinity rules, etc) will now be applied solely under Values.artifactory.primary when using the new defaults. +* **Upgrade** +* Upgrade from primary-members to primary-only is recommended, and can be done by deploy the chart with the new values. +* During the upgrade, members pods should be deleted and new primary pods should be created. This might trigger the creation of new PVCs. +* Added Support for Cold Artifact Storage as part of the systemYaml configuration (disabled by default) +* Added new binary provider `s3-storage-v3-archive` +* Fixed jfconnect disabling as micro-service on non-splitcontainers +* Fixed an issue whereby, Artifactory failed to start when using persistence storage type `nfs` due to missing binarystore.xml + + +## [107.58.0] - Mar 23, 2023 +* Updated postgresql multi-arch tag version to `13.10.0-debian-11-r14` +* Removed obselete remove-lost-found initContainer` +* Added env JF_SHARED_NODE_HAENABLED under frontend when running in the container split mode + +## [107.57.0] - Mar 02, 2023 +* Updated initContainerImage and logger image to `ubi9/ubi-minimal:9.1.0.1793` + +## [107.55.0] - Feb 21, 2023 +* Updated initContainerImage and logger image to `ubi9/ubi-minimal:9.1.0.1760` +* Adding a custom preStop to Artifactory router for allowing graceful termination to complete +* Fixed an invalid reference of node selector on artifactory-ha chart + +## [107.53.0] - Jan 20, 2023 +* Updated initContainerImage and logger image to `ubi8/ubi-minimal:8.7.1049` + +## [107.50.0] - Jan 20, 2023 +* Updated postgresql tag version to `13.9.0-debian-11-r11` +* Fixed make lint issue on artifactory-ha chart [GH-1714](https://github.com/jfrog/charts/issues/1714) +* Fixed an issue for capabilities check of ingress +* Updated jfrogUrl text path in migrate.sh file +* Added a note that from 107.46.x chart versions, `copyOnEveryStartup` is not needed for binarystore.xml, it is always copied via initContainers. For more Info, Refer [GH-1723](https://github.com/jfrog/charts/issues/1723) + +## [107.49.0] - Jan 16, 2023 +* Changed logic in wait-for-primary container to use /dev/tcp instead of curl +* Added support for setting `seLinuxOptions` in `securityContext` [GH-1700](https://github.com/jfrog/charts/pull/1700) +* Added option to enable/disable proxy_request_buffering and proxy_buffering_off [GH-1686](https://github.com/jfrog/charts/pull/1686) +* Updated initContainerImage and logger image to `ubi8/ubi-minimal:8.7.1049` + +## [107.48.0] - Oct 27, 2022 +* Updated router version to `7.51.0` + +## [107.47.0] - Sep 29, 2022 +* Updated initContainerImage to `ubi8/ubi-minimal:8.6-941` +* Added support for annotations for artifactory statefulset and nginx deployment [GH-1665](https://github.com/jfrog/charts/pull/1665) +* Updated router version to `7.49.0` + +## [107.46.0] - Sep 14, 2022 +* **IMPORTANT** +* Added support for lifecycle hooks for all containers, changed `artifactory.postStartCommand` to `.Values.artifactory.lifecycle.postStart.exec.command` +* Updated initContainerImage and logger image to `ubi8/ubi-minimal:8.6-902` +* Update nginx configuration to allow websocket requests when using pipelines +* Fixed an issue to allow artifactory to make direct API calls to store instead via jfconnect service when `splitServicesToContainers=true` +* Refactor binarystore.xml configuration (moved to `files/binarystore.xml` instead of key in values.yaml) +* Added new binary providers `s3-storage-v3-direct`, `azure-blob-storage-direct`, `google-storage-v2` +* Deprecated (removed) `aws-s3` binary provider [JetS3t library](https://www.jfrog.com/confluence/display/JFROG/Configuring+the+Filestore#ConfiguringtheFilestore-BinaryProvider) +* Deprecated (removed) `google-storage` binary provider and force persistence storage type `google-storage` to work with `google-storage-v2` only +* Copy binarystore.xml in init Container to fix existing persistence on file system in clear text +* Removed obselete `.Values.artifactory.binarystore.enabled` key +* Removed `newProbes.enabled`, default to new probes +* Added nginx.customCommand using inotifyd to reload nginx's config upon ssl secret or configmap changes [GH-1640](https://github.com/jfrog/charts/pull/1640) + +## [107.43.0] - Aug 25, 2022 +* Added flag `artifactory.replicator.ingress.enabled` to enable/disable ingress for replicator +* Updated initContainerImage and logger image to `ubi8/ubi-minimal:8.6-854` +* Updated router version to `7.45.0` +* Added flag `artifactory.schedulerName` to set for the pods the value of schedulerName field [GH-1606](https://github.com/jfrog/charts/issues/1606) +* Enabled TLS based on access or router in values.yaml + +## [107.42.0] - Aug 25, 2022 +* Enabled database creds secret to use from unified secret +* Updated router version to `7.42.0` +* Added support to truncate (> 63 chars) for unifiedCustomSecretVolumeName + +## [107.41.0] - June 27, 2022 +* Added support for nginx.terminationGracePeriodSeconds [GH-1645](https://github.com/jfrog/charts/issues/1645) +* Fix nginx lifecycle values [GH-1646](https://github.com/jfrog/charts/pull/1646) +* Use an alternate command for `find` to copy custom certificates +* Added support for circle of trust using `circleOfTrustCertificatesSecret` secret name [GH-1623](https://github.com/jfrog/charts/pull/1623) + +## [107.40.0] - Jun 16, 2022 +* Deprecated k8s PodDisruptionBudget api policy/v1beta1 [GH-1618](https://github.com/jfrog/charts/issues/1618) +* Disabled node PodDisruptionBudget, statefulset and artifactory-primary service from artifactory-ha chart when member nodes are 0 +* From artifactory 7.38.x, joinKey can be retrived from Admin > User Management > Settings in UI +* Fixed template name for artifactory-ha database creds [GH-1602](https://github.com/jfrog/charts/pull/1602) +* Allow templating for pod annotations [GH-1634](https://github.com/jfrog/charts/pull/1634) +* Added flags to control enable/disable infra services in splitServicesToContainers + +## [107.39.0] - May 16, 2022 +* Fix default `artifactory.async.corePoolSize` [GH-1612](https://github.com/jfrog/charts/issues/1612) +* Added support of nginx annotations +* Reduce startupProbe `initialDelaySeconds` +* Align all liveness and readiness probes failureThreshold to `5` seconds +* Added new flag `unifiedSecretInstallation` to enables single unified secret holding all the artifactory-ha secrets +* Updated router version to `7.38.0` + +## [107.38.0] - May 04, 2022 +* Added support for `global.nodeSelector` to artifactory and nginx pods +* Updated router version to `7.36.1` +* Added support for custom global probes timeout +* Updated frontend container command +* Added topologySpreadConstraints to artifactory and nginx, and add lifecycle hooks to nginx [GH-1596](https://github.com/jfrog/charts/pull/1596) +* Added support of extraEnvironmentVariables for all infra services containers +* Enabled the consumption (jfconnect) flag by default +* Fix jfconnect disabling on non-splitcontainers + +## [107.37.0] - Mar 08, 2022 +* Added support for customPorts in nginx deployment +* Bugfix - Wrong proxy_pass configurations for /artifactory/ in the default artifactory.conf +* Added signedUrlExpirySeconds option to artifactory.persistence.type aws-S3-V3 +* Updated router version to `7.35.0` +* Added useInstanceCredentials,enableSignedUrlRedirect option to google-storage-v2 +* Changed dependency charts repo to `charts.jfrog.io` + +## [107.36.0] - Mar 03, 2022 +* Remove pdn tracker which starts replicator service +* Added silent option for curl probes +* Added readiness health check for the artifactory container for k8s version < 1.20 +* Fix property file migration issue to system.yaml 6.x to 7.x + +## [107.35.0] - Feb 08, 2022 +* Updated router version to `7.32.1` + +## [107.33.0] - Jan 11, 2022 +* Make default value of anti-affinity to soft +* Readme fixes +* Added support for setting `fsGroupChangePolicy` +* Added nginx customInitContainers, customVolumes, customSidecarContainers [GH-1565](https://github.com/jfrog/charts/pull/1565) +* Updated router version to `7.30.0` + +## [107.32.0] - Dec 23, 2021 +* Updated logger image to `jfrog/ubi-minimal:8.5-204` +* Added default `8091` as `artifactory.tomcat.maintenanceConnector.port` for probes check +* Refactored probes to replace httpGet probes with basic exec + curl +* Refactored `database-creds` secret to create only when database values are passed +* Added new endpoints for probes `/artifactory/api/v1/system/liveness` and `/artifactory/api/v1/system/readiness` +* Enabled `newProbes:true` by default to use these endpoints +* Fix filebeat sidecar spool file permissions +* Updated filebeat sidecar container to `7.16.2` + +## [107.31.0] - Dec 17, 2021 +* Remove integration service feature flag to make it mandatory service +* Update postgresql tag version to `13.4.0-debian-10-r39` +* Refactored `router.requiredServiceTypes` to support platform chart + +## [107.30.0] - Nov 30, 2021 +* Fixed incorrect permission for filebeat.yaml +* Updated healthcheck (liveness/readiness) api for integration service +* Disable readiness health check for the artifactory container when running in the container split mode +* Ability to start replicator on enabling pdn tracker + +## [107.29.0] - Nov 30, 2021 +* Added integration service container in artifactory +* Add support for Ingress Class Name in Ingress Spec [GH-1516](https://github.com/jfrog/charts/pull/1516) +* Fixed chart values to use curl instead of wget [GH-1529](https://github.com/jfrog/charts/issues/1529) +* Updated nginx config to allow websockets when pipelines is enabled +* Moved router.topology.local.requireqservicetypes from system.yaml to router as environment variable +* Added jfconnect in system.yaml +* Updated artifactory container’s health probes to use artifactory api on rt-split +* Updated initContainerImage to `jfrog/ubi-minimal:8.5-204` +* Updated router version to `7.28.2` +* Set Jfconnect enabled to `false` in the artifactory container when running in the container split mode + +## [107.28.0] - Nov 11, 2021 +* Added default values cpu and memeory in initContainers +* Updated router version to `7.26.0` +* Bug fix - jmx port not exposed in artifactory service +* Updated (`rbac.create` and `serviceAccount.create` to false by default) for least privileges +* Fixed incorrect data type for `Values.router.serviceRegistry.insecure` in default values.yaml [GH-1514](https://github.com/jfrog/charts/pull/1514/files) +* **IMPORTANT** +* Changed init-container images from `alpine` to `ubi8/ubi-minimal` +* Added support for AWS License Manager using `.Values.aws.licenseConfigSecretName` + +## [107.27.0] - Oct 6, 2021 +* **Breaking change** +* Aligned probe structure (moved probes variables under config block) +* Added support for new probes(set to false by default) +* Bugfix - Invalid format for `multiPartLimit,multipartElementSize,maxCacheSize` in binarystore.xml [GH-1466](https://github.com/jfrog/charts/issues/1466) +* Added missioncontrol container in artifactory +* Dropped NET_RAW capability for the containers +* Added resources to migration-artifactory init container +* Added resources to all rt split containers +* Updated router version to `7.25.1` +* Added support for Ingress networking.k8s.io/v1/Ingress for k8s >=1.22 [GH-1487](https://github.com/jfrog/charts/pull/1487) +* Added min kubeVersion ">= 1.14.0-0" in chart.yaml +* Update alpine tag version to `3.14.2` +* Update busybox tag version to `1.33.1` +* Update postgresql tag version to `13.4.0-debian-10-r39` + +## [107.26.0] - Aug 20, 2021 +* Added Observability container (only when `splitServicesToContainers` is enabled) +* Added min kubeVersion ">= 1.12.0-0" in chart.yaml + +## [107.25.0] - Aug 13, 2021 +* Updated readme of chart to point to wiki. Refer [Installing Artifactory](https://www.jfrog.com/confluence/display/JFROG/Installing+Artifactory) +* Added startupProbe and livenessProbe for RT-split containers +* Updated router version to 7.24.1 +* Added security hardening fixes +* Enabled startup probes for k8s >= 1.20.x +* Changed network policy to allow all ingress and egress traffic +* Added Observability changes +* Added support for global.versions.router (only when `splitServicesToContainers` is enabled) + +## [107.24.0] - July 27, 2021 +* Support global and product specific tags at the same time +* Added support for artifactory containers split + +## [107.23.0] - July 8, 2021 +* Bug fix - logger sideCar picks up Wrong File in helm +* Allow filebeat metrics configuration in values.yaml + +## [107.22.0] - July 6, 2021 +* Update alpine tag version to `3.14.0` +* Added `nodePort` support to artifactory-service and nginx-service templates +* Removed redundant `terminationGracePeriodSeconds` in statefulset +* Increased `startupProbe.failureThreshold` time + +## [107.21.3] - July 2, 2021 +* Added ability to change sendreasonphrase value in server.xml via system yaml + +## [107.19.3] - May 20, 2021 +* Fix broken support for startupProbe for k8s < 1.18.x +* Removed an extraneous resources block from the prepare-custom-persistent-volume container in the primary statefulset +* Added support for `nameOverride` and `fullnameOverride` in values.yaml + +## [107.18.6] - May 4, 2021 +* Removed `JF_SHARED_NODE_PRIMARY` env to support for Cloud Native HA +* Bumping chart version to align with app version +* Add `securityContext` option on nginx container + +## [5.0.0] - April 22, 2021 +* **Breaking change:** +* Increased default postgresql persistence size to `200Gi` +* Update postgresql tag version to `13.2.0-debian-10-r55` +* Update postgresql chart version to `10.3.18` in chart.yaml - [10.x Upgrade Notes](https://github.com/bitnami/charts/tree/master/bitnami/postgresql#to-1000) +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x/12.x's postgresql.image.tag, previous postgresql.persistence.size and databaseUpgradeReady=true +* **IMPORTANT** +* This chart is only helm v3 compatible +* Fix support for Cloud Native HA +* Fixed filebeat-configmap naming +* Explicitly set ServiceAccount `automountServiceAccountToken` to 'true' +* Update alpine tag version to `3.13.5` + +## [4.13.2] - April 15, 2021 +* Updated Artifactory version to 7.17.9 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.9) + +## [4.13.1] - April 6, 2021 +* Updated Artifactory version to 7.17.6 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.6) +* Update alpine tag version to `3.13.4` + +## [4.13.0] - April 5, 2021 +* **IMPORTANT** +* Added `charts.jfrog.io` as default JFrog Helm repository +* Updated Artifactory version to 7.17.5 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.5) + +## [4.12.2] - Mar 31, 2021 +* Updated Artifactory version to 7.17.4 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.4) + +## [4.12.1] - Mar 30, 2021 +* Updated Artifactory version to 7.17.3 +* Add `timeoutSeconds` to all exec probes - Please refer [here](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes) + +## [4.12.0] - Mar 24, 2021 +* Updated Artifactory version to 7.17.2 +* Optimized startupProbe time + +## [4.11.0] - Mar 18, 2021 +* Add support to startupProbe + +## [4.10.0] - Mar 15, 2021 +* Updated Artifactory version to 7.16.3 + +## [4.9.5] - Mar 09, 2021 +* Added HSTS header to nginx conf + +## [4.9.4] - Mar 9, 2021 +* Removed bintray URL references in the chart + +## [4.9.3] - Mar 04, 2021 +* Updated Artifactory version to 7.15.4 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.15.4) + +## [4.9.2] - Mar 04, 2021 +* Fixed creation of nginx-certificate-secret when Nginx is disabled + +## [4.9.1] - Feb 19, 2021 +* Update busybox tag version to `1.32.1` + +## [4.9.0] - Feb 18, 2021 +* Updated Artifactory version to 7.15.3 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.15.3) +* Add option to specify update strategy for Artifactory statefulset + +## [4.8.1] - Feb 11, 2021 +* Exposed "multiPartLimit" and "multipartElementSize" for the Azure Blob Storage Binary Provider + +## [4.8.0] - Feb 08, 2021 +* Updated Artifactory version to 7.12.8 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.12.8) +* Support for custom certificates using secrets +* **Important:** Switched docker images download from `docker.bintray.io` to `releases-docker.jfrog.io` +* Update alpine tag version to `3.13.1` + +## [4.7.9] - Feb 3, 2021 +* Fix copyOnEveryStartup for HA cluster license + +## [4.7.8] - Jan 25, 2021 +* Add support for hostAliases + +## [4.7.7] - Jan 11, 2021 +* Fix failures when using creds file for configurating google storage + +## [4.7.6] - Jan 11, 2021 +* Updated Artifactory version to 7.12.6 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.12.6) + +## [4.7.5] - Jan 07, 2021 +* Added support for optional tracker dedicated ingress `.Values.artifactory.replicator.trackerIngress.enabled` (defaults to false) + +## [4.7.4] - Jan 04, 2021 +* Fixed gid support for statefulset + +## [4.7.3] - Dec 31, 2020 +* Added gid support for statefulset +* Add setSecurityContext flag to allow securityContext block to be removed from artifactory statefulset + +## [4.7.2] - Dec 29, 2020 +* **Important:** Removed `.Values.metrics` and `.Values.fluentd` (Fluentd and Prometheus integrations) +* Add support for creating additional kubernetes resources - [refer here](https://github.com/jfrog/log-analytics-prometheus/blob/master/artifactory-ha-values.yaml) +* Updated Artifactory version to 7.12.5 + +## [4.7.1] - Dec 21, 2020 +* Updated Artifactory version to 7.12.3 + +## [4.7.0] - Dec 18, 2020 +* Updated Artifactory version to 7.12.2 +* Added `.Values.artifactory.openMetrics.enabled` + +## [4.6.1] - Dec 11, 2020 +* Added configurable `.Values.global.versions.artifactory` in values.yaml + +## [4.6.0] - Dec 10, 2020 +* Update postgresql tag version to `12.5.0-debian-10-r25` +* Fixed `artifactory.persistence.googleStorage.endpoint` from `storage.googleapis.com` to `commondatastorage.googleapis.com` +* Updated chart maintainers email + +## [4.5.5] - Dec 4, 2020 +* **Important:** Renamed `.Values.systemYaml` to `.Values.systemYamlOverride` + +## [4.5.4] - Dec 1, 2020 +* Improve error message returned when attempting helm upgrade command + +## [4.5.3] - Nov 30, 2020 +* Updated Artifactory version to 7.11.5 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.11) + +# [4.5.2] - Nov 23, 2020 +* Updated Artifactory version to 7.11.2 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.11) +* Updated port namings on services and pods to allow for istio protocol discovery +* Change semverCompare checks to support hosted Kubernetes +* Add flag to disable creation of ServiceMonitor when enabling prometheus metrics +* Prevent the PostHook command to be executed if the user did not specify a command in the values file +* Fix issue with tls file generation when nginx.https.enabled is false + +## [4.5.1] - Nov 19, 2020 +* Updated Artifactory version to 7.11.2 +* Bugfix - access.config.import.xml override Access Federation configurations + +## [4.5.0] - Nov 17, 2020 +* Updated Artifactory version to 7.11.1 +* Update alpine tag version to `3.12.1` + +## [4.4.6] - Nov 10, 2020 +* Pass system.yaml via external secret for advanced usecases +* Added support for custom ingress +* Bugfix - stateful set not picking up changes to database secrets + +## [4.4.5] - Nov 9, 2020 +* Updated Artifactory version to 7.10.6 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.10.6) + +## [4.4.4] - Nov 2, 2020 +* Add enablePathStyleAccess property for aws-s3-v3 binary provider template + +## [4.4.3] - Nov 2, 2020 +* Updated Artifactory version to 7.10.5 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.10.5) + +## [4.4.2] - Oct 22, 2020 +* Chown bug fix where Linux capability cannot chown all files causing log line warnings +* Fix Frontend timeout linting issue + +## [4.4.1] - Oct 20, 2020 +* Add flag to disable prepare-custom-persistent-volume init container + +## [4.4.0] - Oct 19, 2020 +* Updated Artifactory version to 7.10.2 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.10.2) + +## [4.3.4] - Oct 19, 2020 +* Add support to specify priorityClassName for nginx deployment + +## [4.3.3] - Oct 15, 2020 +* Fixed issue with node PodDisruptionBudget which also getting applied on the primary +* Fix mandatory masterKey check issue when upgrading from 6.x to 7.x + +## [4.3.2] - Oct 14, 2020 +* Add support to allow more than 1 Primary in Artifactory-ha STS + +## [4.3.1] - Oct 9, 2020 +* Add global support for customInitContainersBegin + +## [4.3.0] - Oct 07, 2020 +* Updated Artifactory version to 7.9.1 +* **Breaking change:** Fix `storageClass` to correct `storageClassName` in values.yaml + +## [4.2.0] - Oct 5, 2020 +* Expose Prometheus metrics via a ServiceMonitor +* Parse log files for metric data with Fluentd + +## [4.1.0] - Sep 30, 2020 +* Updated Artifactory version to 7.9.0 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.9) + +## [4.0.12] - Sep 25, 2020 +* Update to use linux capability CAP_CHOWN instead of root base init container to avoid any use of root containers to pass Redhat security requirements + +## [4.0.11] - Sep 28, 2020 +* Setting chart coordinates in migitation yaml + +## [4.0.10] - Sep 25, 2020 +* Update filebeat version to `7.9.2` + +## [4.0.9] - Sep 24, 2020 +* Fixed broken issue - when setting `waitForDatabase:false` container startup still waits for DB + +## [4.0.8] - Sep 22, 2020 +* Updated readme + +## [4.0.7] - Sep 22, 2020 +* Fix lint issue in migitation yaml + +## [4.0.6] - Sep 22, 2020 +* Fix broken migitation yaml + +## [4.0.5] - Sep 21, 2020 +* Added mitigation yaml for Artifactory - [More info](https://github.com/jfrog/chartcenter/blob/master/docs/securitymitigationspec.md) + +## [4.0.4] - Sep 17, 2020 +* Added configurable session(UI) timeout in frontend microservice + +## [4.0.3] - Sep 17, 2020 +* Fix small typo in README and added proper required text to be shown while postgres upgrades + +## [4.0.2] - Sep 14, 2020 +* Updated Artifactory version to 7.7.8 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.7.8) + +## [4.0.1] - Sep 8, 2020 +* Added support for artifactory pro license (single node) installation. + +## [4.0.0] - Sep 2, 2020 +* **Breaking change:** Changed `imagePullSecrets` value from string to list +* **Breaking change:** Added `image.registry` and changed `image.version` to `image.tag` for docker images +* Added support for global values +* Updated maintainers in chart.yaml +* Update postgresql tag version to `12.3.0-debian-10-r71` +* Update postgresqlsub chart version to `9.3.4` - [9.x Upgrade Notes](https://github.com/bitnami/charts/tree/master/bitnami/postgresql#900) +* **IMPORTANT** +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x's postgresql.image.tag and databaseUpgradeReady=true. + +## [3.1.0] - Aug 13, 2020 +* Updated Artifactory version to 7.7.3 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.7) + +## [3.0.15] - Aug 10, 2020 +* Added enableSignedUrlRedirect for persistent storage type aws-s3-v3. + +## [3.0.14] - Jul 31, 2020 +* Update the README section on Nginx SSL termination to reflect the actual YAML structure. + +## [3.0.13] - Jul 30, 2020 +* Added condition to disable the migration scripts. + +## [3.0.12] - Jul 29, 2020 +* Document Artifactory node affinity. + +## [3.0.11] - Jul 28, 2020 +* Added maxConnections for persistent storage type aws-s3-v3. + +## [3.0.10] - Jul 28, 2020 +Bugfix / support for userPluginSecrets with Artifactory 7 + +## [3.0.9] - Jul 27, 2020 +* Add tpl to external database secrets. +* Modified `scheme` to `artifactory-ha.scheme` + +## [3.0.8] - Jul 23, 2020 +* Added condition to disable the migration init container. + +## [3.0.7] - Jul 21, 2020 +* Updated Artifactory-ha Chart to add node and primary labels to pods and service objects. + +## [3.0.6] - Jul 20, 2020 +* Support custom CA and certificates + +## [3.0.5] - Jul 13, 2020 +* Updated Artifactory version to 7.6.3 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.6.3 +* Fixed Mysql database jar path in `preStartCommand` in README + +## [3.0.4] - Jul 8, 2020 +* Move some postgresql values to where they should be according to the subchart + +## [3.0.3] - Jul 8, 2020 +* Set Artifactory access client connections to the same value as the access threads. + +## [3.0.2] - Jul 6, 2020 +* Updated Artifactory version to 7.6.2 +* **IMPORTANT** +* Added ChartCenter Helm repository in README + +## [3.0.1] - Jul 01, 2020 +* Add dedicated ingress object for Replicator service when enabled + +## [3.0.0] - Jun 30, 2020 +* Update postgresql tag version to `10.13.0-debian-10-r38` +* Update alpine tag version to `3.12` +* Update busybox tag version to `1.31.1` +* **IMPORTANT** +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass postgresql.image.tag=9.6.18-debian-10-r7 and databaseUpgradeReady=true + +## [2.6.0] - Jun 29, 2020 +* Updated Artifactory version to 7.6.1 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.6.1 +* Add tpl for external database secrets + +## [2.5.8] - Jun 25, 2020 +* Stop loading the Nginx stream module because it is now a core module + +## [2.5.7] - Jun 18, 2020 +* Fixes bootstrap configMap issue on member node + +## [2.5.6] - Jun 11, 2020 +* Support list of custom secrets + +## [2.5.5] - Jun 11, 2020 +* NOTES.txt fixed incorrect information + +## [2.5.4] - Jun 12, 2020 +* Updated Artifactory version to 7.5.7 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.5.7 + +## [2.5.3] - Jun 8, 2020 +* Statically setting primary service type to ClusterIP. +* Prevents primary service from being exposed publicly when using LoadBalancer type on cloud providers. + +## [2.5.2] - Jun 8, 2020 +* Readme update - configuring Artifactory with oracledb + +## [2.5.1] - Jun 5, 2020 +* Fixes broken PDB issue upgrading from 6.x to 7.x + +## [2.5.0] - Jun 1, 2020 +* Updated Artifactory version to 7.5.5 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.5 +* Fixes bootstrap configMap permission issue +* Update postgresql tag version to `9.6.18-debian-10-r7` + +## [2.4.10] - May 27, 2020 +* Added Tomcat maxThreads & acceptCount + +## [2.4.9] - May 25, 2020 +* Fixed postgresql README `image` Parameters + +## [2.4.8] - May 24, 2020 +* Fixed typo in README regarding migration timeout + +## [2.4.7] - May 19, 2020 +* Added metadata maxOpenConnections + +## [2.4.6] - May 07, 2020 +* Fix `installerInfo` string format + +## [2.4.5] - Apr 27, 2020 +* Updated Artifactory version to 7.4.3 + +## [2.4.4] - Apr 27, 2020 +* Change customInitContainers order to run before the "migration-ha-artifactory" initContainer + +## [2.4.3] - Apr 24, 2020 +* Fix `artifactory.persistence.awsS3V3.useInstanceCredentials` incorrect conditional logic +* Bump postgresql tag version to `9.6.17-debian-10-r72` in values.yaml + +## [2.4.2] - Apr 16, 2020 +* Custom volume mounts in migration init container. + +## [2.4.1] - Apr 16, 2020 +* Fix broken support for gcpServiceAccount for googleStorage + +## [2.4.0] - Apr 14, 2020 +* Updated Artifactory version to 7.4.1 + +## [2.3.1] - April 13, 2020 +* Update README with helm v3 commands + +## [2.3.0] - April 10, 2020 +* Use dependency charts from `https://charts.bitnami.com/bitnami` +* Bump postgresql chart version to `8.7.3` in requirements.yaml +* Bump postgresql tag version to `9.6.17-debian-10-r21` in values.yaml + +## [2.2.11] - Apr 8, 2020 +* Added recommended ingress annotation to avoid 413 errors + +## [2.2.10] - Apr 8, 2020 +* Moved migration scripts under `files` directory +* Support preStartCommand in migration Init container as `artifactory.migration.preStartCommand` + +## [2.2.9] - Apr 01, 2020 +* Support masterKey and joinKey as secrets + +## [2.2.8] - Apr 01, 2020 +* Ensure that the join key is also copied when provided by an external secret +* Migration container in primary and node statefulset now respects custom versions and the specified node/primary resources + +## [2.2.7] - Apr 01, 2020 +* Added cache-layer in chain definition of Google Cloud Storage template +* Fix readme use to `-hex 32` instead of `-hex 16` + +## [2.2.6] - Mar 31, 2020 +* Change the way the artifactory `command:` is set so it will properly pass a SIGTERM to java + +## [2.2.5] - Mar 31, 2020 +* Removed duplicate `artifactory-license` volume from primary node + +## [2.2.4] - Mar 31, 2020 +* Restore `artifactory-license` volume for the primary node + +## [2.2.3] - Mar 29, 2020 +* Add Nginx log options: stderr as logfile and log level + +## [2.2.2] - Mar 30, 2020 +* Apply initContainers.resources to `copy-system-yaml`, `prepare-custom-persistent-volume`, and `migration-artifactory-ha` containers +* Use the same defaulting mechanism used for the artifactory version used elsewhere in the chart +* Removed duplicate `artifactory-license` volume that prevented using an external secret + +## [2.2.1] - Mar 29, 2020 +* Fix loggers sidecars configurations to support new file system layout and new log names + +## [2.2.0] - Mar 29, 2020 +* Fix broken admin user bootstrap configuration +* **Breaking change:** renamed `artifactory.accessAdmin` to `artifactory.admin` + +## [2.1.3] - Mar 24, 2020 +* Use `postgresqlExtendedConf` for setting custom PostgreSQL configuration (instead of `postgresqlConfiguration`) + +## [2.1.2] - Mar 21, 2020 +* Support for SSL offload in Nginx service(LoadBalancer) layer. Introduced `nginx.service.ssloffload` field with boolean type. + +## [2.1.1] - Mar 23, 2020 +* Moved installer info to values.yaml so it is fully customizable + +## [2.1.0] - Mar 23, 2020 +* Updated Artifactory version to 7.3.2 + +## [2.0.36] - Mar 20, 2020 +* Add support GCP credentials.json authentication + +## [2.0.35] - Mar 20, 2020 +* Add support for masterKey trim during 6.x to 7.x migration if 6.x masterKey is 32 hex (64 characters) + +## [2.0.34] - Mar 19, 2020 +* Add support for NFS directories `haBackupDir` and `haDataDir` + +## [2.0.33] - Mar 18, 2020 +* Increased Nginx proxy_buffers size + +## [2.0.32] - Mar 17, 2020 +* Changed all single quotes to double quotes in values files +* useInstanceCredentials variable was declared in S3 settings but not used in chart. Now it is being used. + +## [2.0.31] - Mar 17, 2020 +* Fix rendering of Service Account annotations + +## [2.0.30] - Mar 16, 2020 +* Add Unsupported message from 6.18 to 7.2.x (migration) + +## [2.0.29] - Mar 11, 2020 +* Upgrade Docs update + +## [2.0.28] - Mar 11, 2020 +* Unified charts public release + +## [2.0.27] - Mar 8, 2020 +* Add an optional wait for primary node to be ready with a proper test for http status + +## [2.0.23] - Mar 6, 2020 +* Fix path to `/artifactory_bootstrap` +* Add support for controlling the name of the ingress and allow to set more than one cname + +## [2.0.22] - Mar 4, 2020 +* Add support for disabling `consoleLog` in `system.yaml` file + +## [2.0.21] - Feb 28, 2020 +* Add support to process `valueFrom` for extraEnvironmentVariables + +## [2.0.20] - Feb 26, 2020 +* Store join key to secret + +## [2.0.19] - Feb 26, 2020 +* Updated Artifactory version to 7.2.1 + +## [2.0.12] - Feb 07, 2020 +* Remove protection flag `databaseUpgradeReady` which was added to check internal postgres upgrade + +## [2.0.0] - Feb 07, 2020 +* Updated Artifactory version to 7.0.0 + +## [1.4.10] - Feb 13, 2020 +* Add support for SSH authentication to Artifactory + +## [1.4.9] - Feb 10, 2020 +* Fix custom DB password indention + +## [1.4.8] - Feb 9, 2020 +* Add support for `tpl` in the `postStartCommand` + +## [1.4.7] - Feb 4, 2020 +* Support customisable Nginx kind + +## [1.4.6] - Feb 2, 2020 +* Add a comment stating that it is recommended to use an external PostgreSQL with a static password for production installations + +## [1.4.5] - Feb 2, 2020 +* Add support for primary or member node specific preStartCommand + +## [1.4.4] - Jan 30, 2020 +* Add the option to configure resources for the logger containers + +## [1.4.3] - Jan 26, 2020 +* Improve `database.user` and `database.password` logic in order to support more use cases and make the configuration less repetitive + +## [1.4.2] - Jan 22, 2020 +* Refined pod disruption budgets to separate nginx and Artifactory pods + +## [1.4.1] - Jan 19, 2020 +* Fix replicator port config in nginx replicator configmap + +## [1.4.0] - Jan 19, 2020 +* Updated Artifactory version to 6.17.0 + +## [1.3.8] - Jan 16, 2020 +* Added example for external nginx-ingress + +## [1.3.7] - Jan 07, 2020 +* Add support for customizable `mountOptions` of NFS PVs + +## [1.3.6] - Dec 30, 2019 +* Fix for nginx probes failing when launched with http disabled + +## [1.3.5] - Dec 24, 2019 +* Better support for custom `artifactory.internalPort` + +## [1.3.4] - Dec 23, 2019 +* Mark empty map values with `{}` + +## [1.3.3] - Dec 16, 2019 +* Another fix for toggling nginx service ports + +## [1.3.2] - Dec 12, 2019 +* Fix for toggling nginx service ports + +## [1.3.1] - Dec 10, 2019 +* Add support for toggling nginx service ports + +## [1.3.0] - Dec 1, 2019 +* Updated Artifactory version to 6.16.0 + +## [1.2.4] - Nov 28, 2019 +* Add support for using existing PriorityClass + +## [1.2.3] - Nov 27, 2019 +* Add support for PriorityClass + +## [1.2.2] - Nov 20, 2019 +* Update Artifactory logo + +## [1.2.1] - Nov 18, 2019 +* Add the option to provide service account annotations (in order to support stuff like https://docs.aws.amazon.com/eks/latest/userguide/specify-service-account-role.html) + +## [1.2.0] - Nov 18, 2019 +* Updated Artifactory version to 6.15.0 + +## [1.1.12] - Nov 17, 2019 +* Fix `README.md` format (broken table) + +## [1.1.11] - Nov 17, 2019 +* Update comment on Artifactory master key + +## [1.1.10] - Nov 17, 2019 +* Fix creation of double slash in nginx artifactory configuration + +## [1.1.9] - Nov 14, 2019 +* Set explicit `postgresql.postgresqlPassword=""` to avoid helm v3 error + +## [1.1.8] - Nov 12, 2019 +* Updated Artifactory version to 6.14.1 + +## [1.1.7] - Nov 11, 2019 +* Additional documentation for masterKey + +## [1.1.6] - Nov 10, 2019 +* Update PostgreSQL chart version to 7.0.1 +* Use formal PostgreSQL configuration format + +## [1.1.5] - Nov 8, 2019 +* Add support `artifactory.service.loadBalancerSourceRanges` for whitelisting when setting `artifactory.service.type=LoadBalancer` + +## [1.1.4] - Nov 6, 2019 +* Add support for any type of environment variable by using `extraEnvironmentVariables` as-is + +## [1.1.3] - Nov 6, 2019 +* Add nodeselector support for Postgresql + +## [1.1.2] - Nov 5, 2019 +* Add support for the aws-s3-v3 filestore, which adds support for pod IAM roles + +## [1.1.1] - Nov 4, 2019 +* When using `copyOnEveryStartup`, make sure that the target base directories are created before copying the files + +## [1.1.0] - Nov 3, 2019 +* Updated Artifactory version to 6.14.0 + +## [1.0.1] - Nov 3, 2019 +* Make sure the artifactory pod exits when one of the pre-start stages fail + +## [1.0.0] - Oct 27, 2019 +**IMPORTANT - BREAKING CHANGES!**
+**DOWNTIME MIGHT BE REQUIRED FOR AN UPGRADE!** +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), must use the upgrade instructions in [UPGRADE_NOTES.md](UPGRADE_NOTES.md)! +* PostgreSQL sub chart was upgraded to version `6.5.x`. This version is **not backward compatible** with the old version (`0.9.5`)! +* Note the following **PostgreSQL** Helm chart changes + * The chart configuration has changed! See [values.yaml](values.yaml) for the new keys used + * **PostgreSQL** is deployed as a StatefulSet + * See [PostgreSQL helm chart](https://hub.helm.sh/charts/stable/postgresql) for all available configurations + +## [0.17.3] - Oct 24, 2019 +* Change the preStartCommand to support templating + +## [0.17.2] - Oct 21, 2019 +* Add support for setting `artifactory.primary.labels` +* Add support for setting `artifactory.node.labels` +* Add support for setting `nginx.labels` + +## [0.17.1] - Oct 10, 2019 +* Updated Artifactory version to 6.13.1 + +## [0.17.0] - Oct 7, 2019 +* Updated Artifactory version to 6.13.0 + +## [0.16.7] - Sep 24, 2019 +* Option to skip wait-for-db init container with '--set waitForDatabase=false' + +## [0.16.6] - Sep 24, 2019 +* Add support for setting `nginx.service.labels` + +## [0.16.5] - Sep 23, 2019 +* Add support for setting `artifactory.customInitContainersBegin` + +## [0.16.4] - Sep 20, 2019 +* Add support for setting `initContainers.resources` + +## [0.16.3] - Sep 11, 2019 +* Updated Artifactory version to 6.12.2 + +## [0.16.2] - Sep 9, 2019 +* Updated Artifactory version to 6.12.1 + +## [0.16.1] - Aug 22, 2019 +* Fix the nginx server_name directive used with ingress.hosts + +## [0.16.0] - Aug 21, 2019 +* Updated Artifactory version to 6.12.0 + +## [0.15.15] - Aug 18, 2019 +* Fix existingSharedClaim permissions issue and example + +## [0.15.14] - Aug 14, 2019 +* Updated Artifactory version to 6.11.6 + +## [0.15.13] - Aug 11, 2019 +* Fix Ingress routing and add an example + +## [0.15.12] - Aug 6, 2019 +* Do not mount `access/etc/bootstrap.creds` unless user specifies a custom password or secret (Access already generates a random password if not provided one) +* If custom `bootstrap.creds` is provided (using keys or custom secret), prepare it with an init container so the temp file does not persist + +## [0.15.11] - Aug 5, 2019 +* Improve binarystore config + 1. Convert to a secret + 2. Move config to values.yaml + 3. Support an external secret + +## [0.15.10] - Aug 5, 2019 +* Don't create the nginx configmaps when nginx.enabled is false + +## [0.15.9] - Aug 1, 2019 +* Fix masterkey/masterKeySecretName not specified warning render logic in NOTES.txt + +## [0.15.8] - Jul 28, 2019 +* Simplify nginx setup and shorten initial wait for probes + +## [0.15.7] - Jul 25, 2019 +* Updated README about how to apply Artifactory licenses + +## [0.15.6] - Jul 22, 2019 +* Change Ingress API to be compatible with recent kubernetes versions + +## [0.15.5] - Jul 22, 2019 +* Updated Artifactory version to 6.11.3 + +## [0.15.4] - Jul 11, 2019 +* Add `artifactory.customVolumeMounts` support to member node statefulset template + +## [0.15.3] - Jul 11, 2019 +* Add ingress.hosts to the Nginx server_name directive when ingress is enabled to help with Docker repository sub domain configuration + +## [0.15.2] - Jul 3, 2019 +* Add the option for changing nginx config using values.yaml and remove outdated reverse proxy documentation + +## [0.15.1] - Jul 1, 2019 +* Updated Artifactory version to 6.11.1 + +## [0.15.0] - Jun 27, 2019 +* Updated Artifactory version to 6.11.0 and Restart Primary node when bootstrap.creds file has been modified in artifactory-ha + +## [0.14.4] - Jun 24, 2019 +* Add the option to provide an IP for the access-admin endpoints + +## [0.14.3] - Jun 24, 2019 +* Update chart maintainers + +## [0.14.2] - Jun 24, 2019 +* Change Nginx to point to the artifactory externalPort + +## [0.14.1] - Jun 23, 2019 +* Add values files for small, medium and large installations + +## [0.14.0] - Jun 20, 2019 +* Use ConfigMaps for nginx configuration and remove nginx postStart command + +## [0.13.10] - Jun 19, 2019 +* Updated Artifactory version to 6.10.4 + +## [0.13.9] - Jun 18, 2019 +* Add the option to provide additional ingress rules + +## [0.13.8] - Jun 14, 2019 +* Updated readme with improved external database setup example + +## [0.13.7] - Jun 6, 2019 +* Updated Artifactory version to 6.10.3 +* Updated installer-info template + +## [0.13.6] - Jun 6, 2019 +* Updated Google Cloud Storage API URL and https settings + +## [0.13.5] - Jun 5, 2019 +* Delete the db.properties file on Artifactory startup + +## [0.13.4] - Jun 3, 2019 +* Updated Artifactory version to 6.10.2 + +## [0.13.3] - May 21, 2019 +* Updated Artifactory version to 6.10.1 + +## [0.13.2] - May 19, 2019 +* Fix missing logger image tag + +## [0.13.1] - May 15, 2019 +* Support `artifactory.persistence.cacheProviderDir` for on-premise cluster + +## [0.13.0] - May 7, 2019 +* Updated Artifactory version to 6.10.0 + +## [0.12.23] - May 5, 2019 +* Add support for setting `artifactory.async.corePoolSize` + +## [0.12.22] - May 2, 2019 +* Remove unused property `artifactory.releasebundle.feature.enabled` + +## [0.12.21] - Apr 30, 2019 +* Add support for JMX monitoring + +## [0.12.20] - Apr29, 2019 +* Added support for headless services + +## [0.12.19] - Apr 28, 2019 +* Added support for `cacheProviderDir` + +## [0.12.18] - Apr 18, 2019 +* Changing API StatefulSet version to `v1` and permission fix for custom `artifactory.conf` for Nginx + +## [0.12.17] - Apr 16, 2019 +* Updated documentation for Reverse Proxy Configuration + +## [0.12.16] - Apr 12, 2019 +* Added support for `customVolumeMounts` + +## [0.12.15] - Aprl 12, 2019 +* Added support for `bucketExists` flag for googleStorage + +## [0.12.14] - Apr 11, 2019 +* Replace `curl` examples with `wget` due to the new base image + +## [0.12.13] - Aprl 07, 2019 +* Add support for providing the Artifactory license as a parameter + +## [0.12.12] - Apr 10, 2019 +* Updated Artifactory version to 6.9.1 + +## [0.12.11] - Aprl 04, 2019 +* Add support for templated extraEnvironmentVariables + +## [0.12.10] - Aprl 07, 2019 +* Change network policy API group + +## [0.12.9] - Aprl 04, 2019 +* Apply the existing PVC for members (in addition to primary) + +## [0.12.8] - Aprl 03, 2019 +* Bugfix for userPluginSecrets + +## [0.12.7] - Apr 4, 2019 +* Add information about upgrading Artifactory with auto-generated postgres password + +## [0.12.6] - Aprl 03, 2019 +* Added installer info + +## [0.12.5] - Aprl 03, 2019 +* Allow secret names for user plugins to contain template language + +## [0.12.4] - Apr 02, 2019 +* Fix issue #253 (use existing PVC for data and backup storage) + +## [0.12.3] - Apr 02, 2019 +* Allow NetworkPolicy configurations (defaults to allow all) + +## [0.12.2] - Aprl 01, 2019 +* Add support for user plugin secret + +## [0.12.1] - Mar 26, 2019 +* Add the option to copy a list of files to ARTIFACTORY_HOME on startup + +## [0.12.0] - Mar 26, 2019 +* Updated Artifactory version to 6.9.0 + +## [0.11.18] - Mar 25, 2019 +* Add CI tests for persistence, ingress support and nginx + +## [0.11.17] - Mar 22, 2019 +* Add the option to change the default access-admin password + +## [0.11.16] - Mar 22, 2019 +* Added support for `.Probe.path` to customise the paths used for health probes + +## [0.11.15] - Mar 21, 2019 +* Added support for `artifactory.customSidecarContainers` to create custom sidecar containers +* Added support for `artifactory.customVolumes` to create custom volumes + +## [0.11.14] - Mar 21, 2019 +* Make ingress path configurable + +## [0.11.13] - Mar 19, 2019 +* Move the copy of bootstrap config from postStart to preStart for Primary + +## [0.11.12] - Mar 19, 2019 +* Fix existingClaim example + +## [0.11.11] - Mar 18, 2019 +* Disable the option to use nginx PVC with more than one replica + +## [0.11.10] - Mar 15, 2019 +* Wait for nginx configuration file before using it + +## [0.11.9] - Mar 15, 2019 +* Revert securityContext changes since they were causing issues + +## [0.11.8] - Mar 15, 2019 +* Fix issue #247 (init container failing to run) + +## [0.11.7] - Mar 14, 2019 +* Updated Artifactory version to 6.8.7 + +## [0.11.6] - Mar 13, 2019 +* Move securityContext to container level + +## [0.11.5] - Mar 11, 2019 +* Add the option to use existing volume claims for Artifactory storage + +## [0.11.4] - Mar 11, 2019 +* Updated Artifactory version to 6.8.6 + +## [0.11.3] - Mar 5, 2019 +* Updated Artifactory version to 6.8.4 + +## [0.11.2] - Mar 4, 2019 +* Add support for catalina logs sidecars + +## [0.11.1] - Feb 27, 2019 +* Updated Artifactory version to 6.8.3 + +## [0.11.0] - Feb 25, 2019 +* Add nginx support for tail sidecars + +## [0.10.3] - Feb 21, 2019 +* Add s3AwsVersion option to awsS3 configuration for use with IAM roles + +## [0.10.2] - Feb 19, 2019 +* Updated Artifactory version to 6.8.2 + +## [0.10.1] - Feb 17, 2019 +* Updated Artifactory version to 6.8.1 +* Add example of `SERVER_XML_EXTRA_CONNECTOR` usage + +## [0.10.0] - Feb 15, 2019 +* Updated Artifactory version to 6.8.0 + +## [0.9.7] - Feb 13, 2019 +* Updated Artifactory version to 6.7.3 + +## [0.9.6] - Feb 7, 2019 +* Add support for tail sidecars to view logs from k8s api + +## [0.9.5] - Feb 6, 2019 +* Fix support for customizing statefulset `terminationGracePeriodSeconds` + +## [0.9.4] - Feb 5, 2019 +* Add support for customizing statefulset `terminationGracePeriodSeconds` + +## [0.9.3] - Feb 5, 2019 +* Remove the inactive server remove plugin + +## [0.9.2] - Feb 3, 2019 +* Updated Artifactory version to 6.7.2 + +## [0.9.1] - Jan 27, 2019 +* Fix support for Azure Blob Storage Binary provider + +## [0.9.0] - Jan 23, 2019 +* Updated Artifactory version to 6.7.0 + +## [0.8.10] - Jan 22, 2019 +* Added support for `artifactory.customInitContainers` to create custom init containers + +## [0.8.9] - Jan 18, 2019 +* Added support of values ingress.labels + +## [0.8.8] - Jan 16, 2019 +* Mount replicator.yaml (config) directly to /replicator_extra_conf + +## [0.8.7] - Jan 15, 2018 +* Add support for Azure Blob Storage Binary provider + +## [0.8.6] - Jan 13, 2019 +* Fix documentation about nginx group id + +## [0.8.5] - Jan 13, 2019 +* Updated Artifactory version to 6.6.5 + +## [0.8.4] - Jan 8, 2019 +* Make artifactory.replicator.publicUrl required when the replicator is enabled + +## [0.8.3] - Jan 1, 2019 +* Updated Artifactory version to 6.6.3 +* Add support for `artifactory.extraEnvironmentVariables` to pass more environment variables to Artifactory + +## [0.8.2] - Dec 28, 2018 +* Fix location `replicator.yaml` is copied to + +## [0.8.1] - Dec 27, 2018 +* Updated Artifactory version to 6.6.1 + +## [0.8.0] - Dec 20, 2018 +* Updated Artifactory version to 6.6.0 + +## [0.7.17] - Dec 17, 2018 +* Updated Artifactory version to 6.5.13 + +## [0.7.16] - Dec 12, 2018 +* Fix documentation about Artifactory license setup using secret + +## [0.7.15] - Dec 9, 2018 +* AWS S3 add `roleName` for using IAM role + +## [0.7.14] - Dec 6, 2018 +* AWS S3 `identity` and `credential` are now added only if have a value to allow using IAM role + +## [0.7.13] - Dec 5, 2018 +* Remove Distribution certificates creation. + +## [0.7.12] - Dec 2, 2018 +* Remove Java option "-Dartifactory.locking.provider.type=db". This is already the default setting. + +## [0.7.11] - Nov 30, 2018 +* Updated Artifactory version to 6.5.9 + +## [0.7.10] - Nov 29, 2018 +* Fixed the volumeMount for the replicator.yaml + +## [0.7.9] - Nov 29, 2018 +* Optionally include primary node into poddisruptionbudget + +## [0.7.8] - Nov 29, 2018 +* Updated postgresql version to 9.6.11 + +## [0.7.7] - Nov 27, 2018 +* Updated Artifactory version to 6.5.8 + +## [0.7.6] - Nov 18, 2018 +* Added support for configMap to use custom Reverse Proxy Configuration with Nginx + +## [0.7.5] - Nov 14, 2018 +* Updated Artifactory version to 6.5.3 + +## [0.7.4] - Nov 13, 2018 +* Allow pod anti-affinity settings to include primary node + +## [0.7.3] - Nov 12, 2018 +* Support artifactory.preStartCommand for running command before entrypoint starts + +## [0.7.2] - Nov 7, 2018 +* Support database.url parameter (DB_URL) + +## [0.7.1] - Oct 29, 2018 +* Change probes port to 8040 (so they will not be blocked when all tomcat threads on 8081 are exhausted) + +## [0.7.0] - Oct 28, 2018 +* Update postgresql chart to version 0.9.5 to be able and use `postgresConfig` options + +## [0.6.9] - Oct 23, 2018 +* Fix providing external secret for database credentials + +## [0.6.8] - Oct 22, 2018 +* Allow user to configure externalTrafficPolicy for Loadbalancer + +## [0.6.7] - Oct 22, 2018 +* Updated ingress annotation support (with examples) to support docker registry v2 + +## [0.6.6] - Oct 21, 2018 +* Updated Artifactory version to 6.5.2 + +## [0.6.5] - Oct 19, 2018 +* Allow providing pre-existing secret containing master key +* Allow arbitrary annotations on primary and member node pods +* Enforce size limits when using local storage with `emptyDir` +* Allow `soft` or `hard` specification of member node anti-affinity +* Allow providing pre-existing secrets containing external database credentials +* Fix `s3` binary store provider to properly use the `cache-fs` provider +* Allow arbitrary properties when using the `s3` binary store provider + +## [0.6.4] - Oct 18, 2018 +* Updated Artifactory version to 6.5.1 + +## [0.6.3] - Oct 17, 2018 +* Add Apache 2.0 license + +## [0.6.2] - Oct 14, 2018 +* Make S3 endpoint configurable (was hardcoded with `s3.amazonaws.com`) + +## [0.6.1] - Oct 11, 2018 +* Allows ingress default `backend` to be enabled or disabled (defaults to enabled) + +## [0.6.0] - Oct 11, 2018 +* Updated Artifactory version to 6.5.0 + +## [0.5.3] - Oct 9, 2018 +* Quote ingress hosts to support wildcard names + +## [0.5.2] - Oct 2, 2018 +* Add `helm repo add jfrog https://charts.jfrog.io` to README + +## [0.5.1] - Oct 2, 2018 +* Set Artifactory to 6.4.1 + +## [0.5.0] - Sep 27, 2018 +* Set Artifactory to 6.4.0 + +## [0.4.7] - Sep 26, 2018 +* Add ci/test-values.yaml + +## [0.4.6] - Sep 25, 2018 +* Add PodDisruptionBudget for member nodes, defaulting to minAvailable of 1 + +## [0.4.4] - Sep 2, 2018 +* Updated Artifactory version to 6.3.2 + +## [0.4.0] - Aug 22, 2018 +* Added support to run as non root +* Updated Artifactory version to 6.2.0 + +## [0.3.0] - Aug 22, 2018 +* Enabled RBAC Support +* Added support for PostStartCommand (To download Database JDBC connector) +* Increased postgresql max_connections +* Added support for `nginx.conf` ConfigMap +* Updated Artifactory version to 6.1.0 diff --git a/charts/jfrog/artifactory-ha/107.90.14/Chart.lock b/charts/jfrog/artifactory-ha/107.90.14/Chart.lock new file mode 100644 index 0000000000..eb94099719 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: postgresql + repository: https://charts.jfrog.io/ + version: 10.3.18 +digest: sha256:404ce007353baaf92a6c5f24b249d5b336c232e5fd2c29f8a0e4d0095a09fd53 +generated: "2022-03-08T08:54:51.805126+05:30" diff --git a/charts/jfrog/artifactory-ha/107.90.14/Chart.yaml b/charts/jfrog/artifactory-ha/107.90.14/Chart.yaml new file mode 100644 index 0000000000..fae27e2b97 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/Chart.yaml @@ -0,0 +1,30 @@ +annotations: + artifactoryServiceVersion: 7.90.20 + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: JFrog Artifactory HA + catalog.cattle.io/kube-version: '>= 1.19.0-0' + catalog.cattle.io/release-name: artifactory-ha +apiVersion: v2 +appVersion: 7.90.14 +dependencies: +- condition: postgresql.enabled + name: postgresql + repository: https://charts.jfrog.io/ + version: 10.3.18 +description: Universal Repository Manager supporting all major packaging formats, + build tools and CI servers. +home: https://www.jfrog.com/artifactory/ +icon: file://assets/icons/artifactory-ha.png +keywords: +- artifactory +- jfrog +- devops +kubeVersion: '>= 1.19.0-0' +maintainers: +- email: installers@jfrog.com + name: Chart Maintainers at JFrog +name: artifactory-ha +sources: +- https://github.com/jfrog/charts +type: application +version: 107.90.14 diff --git a/charts/jfrog/artifactory-ha/107.90.14/LICENSE b/charts/jfrog/artifactory-ha/107.90.14/LICENSE new file mode 100644 index 0000000000..8dada3edaf --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/charts/jfrog/artifactory-ha/107.90.14/README.md b/charts/jfrog/artifactory-ha/107.90.14/README.md new file mode 100644 index 0000000000..49155926e0 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/README.md @@ -0,0 +1,69 @@ +# JFrog Artifactory High Availability Helm Chart + +**IMPORTANT!** Our Helm Chart docs have moved to our main documentation site. Below you will find the basic instructions for installing, uninstalling, and deleting Artifactory. For all other information, refer to [Installing Artifactory - Helm HA Installation](https://www.jfrog.com/confluence/display/JFROG/Installing+Artifactory#InstallingArtifactory-HelmHAInstallation). + +**Note:** From Artifactory 7.17.4 and above, the Helm HA installation can be installed so that each node you install can run all tasks in the cluster. + +Below you will find the basic instructions for installing, uninstalling, and deleting Artifactory. For all other information, refer to the documentation site. + +## Prerequisites Details + +* Kubernetes 1.19+ +* Artifactory HA license + +## Chart Details +This chart will do the following: + +* Deploy Artifactory highly available cluster. 1 primary node and 2 member nodes. +* Deploy a PostgreSQL database **NOTE:** For production grade installations it is recommended to use an external PostgreSQL +* Deploy an Nginx server + +## Installing the Chart + +### Add JFrog Helm repository + +Before installing JFrog helm charts, you need to add the [JFrog helm repository](https://charts.jfrog.io) to your helm client + +```bash +helm repo add jfrog https://charts.jfrog.io +``` +2. Next, create a unique Master Key (Artifactory requires a unique master key) and pass it to the template during installation. +3. Now, update the repository. + +```bash +helm repo update +``` + +### Install Chart +To install the chart with the release name `artifactory`: +```bash +helm upgrade --install artifactory-ha jfrog/artifactory-ha --namespace artifactory-ha --create-namespace +``` + +### Apply Sizing configurations to the Chart +To apply the chart with recommended sizing configurations : +For small configurations : +```bash +helm upgrade --install artifactory-ha jfrog/artifactory-ha -f sizing/artifactory-small-extra-config.yaml -f sizing/artifactory-small.yaml --namespace artifactory-ha --create-namespace +``` + +## Uninstalling Artifactory + +Uninstall is supported only on Helm v3 and on. + +Uninstall Artifactory using the following command. + +```bash +helm uninstall artifactory-ha && sleep 90 && kubectl delete pvc -l app=artifactory-ha +``` + +## Deleting Artifactory + +**IMPORTANT:** Deleting Artifactory will also delete your data volumes and you will lose all of your data. You must back up all this information before deletion. You do not need to uninstall Artifactory before deleting it. + +To delete Artifactory use the following command. + +```bash +helm delete artifactory-ha --namespace artifactory-ha +``` + diff --git a/charts/jfrog/artifactory-ha/107.90.14/app-readme.md b/charts/jfrog/artifactory-ha/107.90.14/app-readme.md new file mode 100644 index 0000000000..a5aa5fd478 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/app-readme.md @@ -0,0 +1,16 @@ +# JFrog Artifactory High Availability Helm Chart + +Universal Repository Manager supporting all major packaging formats, build tools and CI servers. + +## Chart Details +This chart will do the following: + +* Deploy Artifactory highly available cluster. 1 primary node and 2 member nodes. +* Deploy a PostgreSQL database +* Deploy an Nginx server(optional) + +## Useful links +Blog: [Herd Trust Into Your Rancher Labs Multi-Cloud Strategy with Artifactory](https://jfrog.com/blog/herd-trust-into-your-rancher-labs-multi-cloud-strategy-with-artifactory/) + +## Activate Your Artifactory Instance +Don't have a license? Please send an email to [rancher-jfrog-licenses@jfrog.com](mailto:rancher-jfrog-licenses@jfrog.com) to get it. diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/.helmignore b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/.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/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.lock b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.lock new file mode 100644 index 0000000000..3687f52df5 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common + repository: https://charts.bitnami.com/bitnami + version: 1.4.2 +digest: sha256:dce0349883107e3ff103f4f17d3af4ad1ea3c7993551b1c28865867d3e53d37c +generated: "2021-03-30T09:13:28.360322819Z" diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.yaml new file mode 100644 index 0000000000..4b197b2071 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/Chart.yaml @@ -0,0 +1,29 @@ +annotations: + category: Database +apiVersion: v2 +appVersion: 11.11.0 +dependencies: +- name: common + repository: https://charts.bitnami.com/bitnami + version: 1.x.x +description: Chart for PostgreSQL, an object-relational database management system + (ORDBMS) with an emphasis on extensibility and on standards-compliance. +home: https://github.com/bitnami/charts/tree/master/bitnami/postgresql +icon: https://bitnami.com/assets/stacks/postgresql/img/postgresql-stack-220x234.png +keywords: +- postgresql +- postgres +- database +- sql +- replication +- cluster +maintainers: +- email: containers@bitnami.com + name: Bitnami +- email: cedric@desaintmartin.fr + name: desaintmartin +name: postgresql +sources: +- https://github.com/bitnami/bitnami-docker-postgresql +- https://www.postgresql.org/ +version: 10.3.18 diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/README.md b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/README.md new file mode 100644 index 0000000000..63d3605bb8 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/README.md @@ -0,0 +1,770 @@ +# PostgreSQL + +[PostgreSQL](https://www.postgresql.org/) is an object-relational database management system (ORDBMS) with an emphasis on extensibility and on standards-compliance. + +For HA, please see [this repo](https://github.com/bitnami/charts/tree/master/bitnami/postgresql-ha) + +## TL;DR + +```console +$ helm repo add bitnami https://charts.bitnami.com/bitnami +$ helm install my-release bitnami/postgresql +``` + +## Introduction + +This chart bootstraps a [PostgreSQL](https://github.com/bitnami/bitnami-docker-postgresql) deployment on a [Kubernetes](http://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. + +Bitnami charts can be used with [Kubeapps](https://kubeapps.com/) for deployment and management of Helm Charts in clusters. This chart has been tested to work with NGINX Ingress, cert-manager, fluentd and Prometheus on top of the [BKPR](https://kubeprod.io/). + +## Prerequisites + +- Kubernetes 1.12+ +- Helm 3.1.0 +- PV provisioner support in the underlying infrastructure + +## Installing the Chart +To install the chart with the release name `my-release`: + +```console +$ helm install my-release bitnami/postgresql +``` + +The command deploys PostgreSQL on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. + +> **Tip**: List all releases using `helm list` + +## Uninstalling the Chart + +To uninstall/delete the `my-release` deployment: + +```console +$ helm delete my-release +``` + +The command removes all the Kubernetes components but PVC's associated with the chart and deletes the release. + +To delete the PVC's associated with `my-release`: + +```console +$ kubectl delete pvc -l release=my-release +``` + +> **Note**: Deleting the PVC's will delete postgresql data as well. Please be cautious before doing it. + +## Parameters + +The following tables lists the configurable parameters of the PostgreSQL chart and their default values. + +| Parameter | Description | Default | +|-----------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------| +| `global.imageRegistry` | Global Docker Image registry | `nil` | +| `global.postgresql.postgresqlDatabase` | PostgreSQL database (overrides `postgresqlDatabase`) | `nil` | +| `global.postgresql.postgresqlUsername` | PostgreSQL username (overrides `postgresqlUsername`) | `nil` | +| `global.postgresql.existingSecret` | Name of existing secret to use for PostgreSQL passwords (overrides `existingSecret`) | `nil` | +| `global.postgresql.postgresqlPassword` | PostgreSQL admin password (overrides `postgresqlPassword`) | `nil` | +| `global.postgresql.servicePort` | PostgreSQL port (overrides `service.port`) | `nil` | +| `global.postgresql.replicationPassword` | Replication user password (overrides `replication.password`) | `nil` | +| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` (does not add image pull secrets to deployed pods) | +| `global.storageClass` | Global storage class for dynamic provisioning | `nil` | +| `image.registry` | PostgreSQL Image registry | `docker.io` | +| `image.repository` | PostgreSQL Image name | `bitnami/postgresql` | +| `image.tag` | PostgreSQL Image tag | `{TAG_NAME}` | +| `image.pullPolicy` | PostgreSQL Image pull policy | `IfNotPresent` | +| `image.pullSecrets` | Specify Image pull secrets | `nil` (does not add image pull secrets to deployed pods) | +| `image.debug` | Specify if debug values should be set | `false` | +| `nameOverride` | String to partially override common.names.fullname template with a string (will prepend the release name) | `nil` | +| `fullnameOverride` | String to fully override common.names.fullname template with a string | `nil` | +| `volumePermissions.enabled` | Enable init container that changes volume permissions in the data directory (for cases where the default k8s `runAsUser` and `fsUser` values do not work) | `false` | +| `volumePermissions.image.registry` | Init container volume-permissions image registry | `docker.io` | +| `volumePermissions.image.repository` | Init container volume-permissions image name | `bitnami/bitnami-shell` | +| `volumePermissions.image.tag` | Init container volume-permissions image tag | `"10"` | +| `volumePermissions.image.pullPolicy` | Init container volume-permissions image pull policy | `Always` | +| `volumePermissions.securityContext.*` | Other container security context to be included as-is in the container spec | `{}` | +| `volumePermissions.securityContext.runAsUser` | User ID for the init container (when facing issues in OpenShift or uid unknown, try value "auto") | `0` | +| `usePasswordFile` | Have the secrets mounted as a file instead of env vars | `false` | +| `ldap.enabled` | Enable LDAP support | `false` | +| `ldap.existingSecret` | Name of existing secret to use for LDAP passwords | `nil` | +| `ldap.url` | LDAP URL beginning in the form `ldap[s]://host[:port]/basedn[?[attribute][?[scope][?[filter]]]]` | `nil` | +| `ldap.server` | IP address or name of the LDAP server. | `nil` | +| `ldap.port` | Port number on the LDAP server to connect to | `nil` | +| `ldap.scheme` | Set to `ldaps` to use LDAPS. | `nil` | +| `ldap.tls` | Set to `1` to use TLS encryption | `nil` | +| `ldap.prefix` | String to prepend to the user name when forming the DN to bind | `nil` | +| `ldap.suffix` | String to append to the user name when forming the DN to bind | `nil` | +| `ldap.search_attr` | Attribute to match against the user name in the search | `nil` | +| `ldap.search_filter` | The search filter to use when doing search+bind authentication | `nil` | +| `ldap.baseDN` | Root DN to begin the search for the user in | `nil` | +| `ldap.bindDN` | DN of user to bind to LDAP | `nil` | +| `ldap.bind_password` | Password for the user to bind to LDAP | `nil` | +| `replication.enabled` | Enable replication | `false` | +| `replication.user` | Replication user | `repl_user` | +| `replication.password` | Replication user password | `repl_password` | +| `replication.readReplicas` | Number of read replicas replicas | `1` | +| `replication.synchronousCommit` | Set synchronous commit mode. Allowed values: `on`, `remote_apply`, `remote_write`, `local` and `off` | `off` | +| `replication.numSynchronousReplicas` | Number of replicas that will have synchronous replication. Note: Cannot be greater than `replication.readReplicas`. | `0` | +| `replication.applicationName` | Cluster application name. Useful for advanced replication settings | `my_application` | +| `existingSecret` | Name of existing secret to use for PostgreSQL passwords. The secret has to contain the keys `postgresql-password` which is the password for `postgresqlUsername` when it is different of `postgres`, `postgresql-postgres-password` which will override `postgresqlPassword`, `postgresql-replication-password` which will override `replication.password` and `postgresql-ldap-password` which will be used to authenticate on LDAP. The value is evaluated as a template. | `nil` | +| `postgresqlPostgresPassword` | PostgreSQL admin password (used when `postgresqlUsername` is not `postgres`, in which case`postgres` is the admin username). | _random 10 character alphanumeric string_ | +| `postgresqlUsername` | PostgreSQL user (creates a non-admin user when `postgresqlUsername` is not `postgres`) | `postgres` | +| `postgresqlPassword` | PostgreSQL user password | _random 10 character alphanumeric string_ | +| `postgresqlDatabase` | PostgreSQL database | `nil` | +| `postgresqlDataDir` | PostgreSQL data dir folder | `/bitnami/postgresql` (same value as persistence.mountPath) | +| `extraEnv` | Any extra environment variables you would like to pass on to the pod. The value is evaluated as a template. | `[]` | +| `extraEnvVarsCM` | Name of a Config Map containing extra environment variables you would like to pass on to the pod. The value is evaluated as a template. | `nil` | +| `postgresqlInitdbArgs` | PostgreSQL initdb extra arguments | `nil` | +| `postgresqlInitdbWalDir` | PostgreSQL location for transaction log | `nil` | +| `postgresqlConfiguration` | Runtime Config Parameters | `nil` | +| `postgresqlExtendedConf` | Extended Runtime Config Parameters (appended to main or default configuration) | `nil` | +| `pgHbaConfiguration` | Content of pg_hba.conf | `nil (do not create pg_hba.conf)` | +| `postgresqlSharedPreloadLibraries` | Shared preload libraries (comma-separated list) | `pgaudit` | +| `postgresqlMaxConnections` | Maximum total connections | `nil` | +| `postgresqlPostgresConnectionLimit` | Maximum total connections for the postgres user | `nil` | +| `postgresqlDbUserConnectionLimit` | Maximum total connections for the non-admin user | `nil` | +| `postgresqlTcpKeepalivesInterval` | TCP keepalives interval | `nil` | +| `postgresqlTcpKeepalivesIdle` | TCP keepalives idle | `nil` | +| `postgresqlTcpKeepalivesCount` | TCP keepalives count | `nil` | +| `postgresqlStatementTimeout` | Statement timeout | `nil` | +| `postgresqlPghbaRemoveFilters` | Comma-separated list of patterns to remove from the pg_hba.conf file | `nil` | +| `customStartupProbe` | Override default startup probe | `nil` | +| `customLivenessProbe` | Override default liveness probe | `nil` | +| `customReadinessProbe` | Override default readiness probe | `nil` | +| `audit.logHostname` | Add client hostnames to the log file | `false` | +| `audit.logConnections` | Add client log-in operations to the log file | `false` | +| `audit.logDisconnections` | Add client log-outs operations to the log file | `false` | +| `audit.pgAuditLog` | Add operations to log using the pgAudit extension | `nil` | +| `audit.clientMinMessages` | Message log level to share with the user | `nil` | +| `audit.logLinePrefix` | Template string for the log line prefix | `nil` | +| `audit.logTimezone` | Timezone for the log timestamps | `nil` | +| `configurationConfigMap` | ConfigMap with the PostgreSQL configuration files (Note: Overrides `postgresqlConfiguration` and `pgHbaConfiguration`). The value is evaluated as a template. | `nil` | +| `extendedConfConfigMap` | ConfigMap with the extended PostgreSQL configuration files. The value is evaluated as a template. | `nil` | +| `initdbScripts` | Dictionary of initdb scripts | `nil` | +| `initdbUser` | PostgreSQL user to execute the .sql and sql.gz scripts | `nil` | +| `initdbPassword` | Password for the user specified in `initdbUser` | `nil` | +| `initdbScriptsConfigMap` | ConfigMap with the initdb scripts (Note: Overrides `initdbScripts`). The value is evaluated as a template. | `nil` | +| `initdbScriptsSecret` | Secret with initdb scripts that contain sensitive information (Note: can be used with `initdbScriptsConfigMap` or `initdbScripts`). The value is evaluated as a template. | `nil` | +| `service.type` | Kubernetes Service type | `ClusterIP` | +| `service.port` | PostgreSQL port | `5432` | +| `service.nodePort` | Kubernetes Service nodePort | `nil` | +| `service.annotations` | Annotations for PostgreSQL service | `{}` (evaluated as a template) | +| `service.loadBalancerIP` | loadBalancerIP if service type is `LoadBalancer` | `nil` | +| `service.loadBalancerSourceRanges` | Address that are allowed when svc is LoadBalancer | `[]` (evaluated as a template) | +| `schedulerName` | Name of the k8s scheduler (other than default) | `nil` | +| `shmVolume.enabled` | Enable emptyDir volume for /dev/shm for primary and read replica(s) Pod(s) | `true` | +| `shmVolume.chmod.enabled` | Run at init chmod 777 of the /dev/shm (ignored if `volumePermissions.enabled` is `false`) | `true` | +| `persistence.enabled` | Enable persistence using PVC | `true` | +| `persistence.existingClaim` | Provide an existing `PersistentVolumeClaim`, the value is evaluated as a template. | `nil` | +| `persistence.mountPath` | Path to mount the volume at | `/bitnami/postgresql` | +| `persistence.subPath` | Subdirectory of the volume to mount at | `""` | +| `persistence.storageClass` | PVC Storage Class for PostgreSQL volume | `nil` | +| `persistence.accessModes` | PVC Access Mode for PostgreSQL volume | `[ReadWriteOnce]` | +| `persistence.size` | PVC Storage Request for PostgreSQL volume | `8Gi` | +| `persistence.annotations` | Annotations for the PVC | `{}` | +| `persistence.selector` | Selector to match an existing Persistent Volume (this value is evaluated as a template) | `{}` | +| `commonAnnotations` | Annotations to be added to all deployed resources (rendered as a template) | `{}` | +| `primary.podAffinityPreset` | PostgreSQL primary pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `primary.podAntiAffinityPreset` | PostgreSQL primary pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | +| `primary.nodeAffinityPreset.type` | PostgreSQL primary node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `primary.nodeAffinityPreset.key` | PostgreSQL primary node label key to match Ignored if `primary.affinity` is set. | `""` | +| `primary.nodeAffinityPreset.values` | PostgreSQL primary node label values to match. Ignored if `primary.affinity` is set. | `[]` | +| `primary.affinity` | Affinity for PostgreSQL primary pods assignment | `{}` (evaluated as a template) | +| `primary.nodeSelector` | Node labels for PostgreSQL primary pods assignment | `{}` (evaluated as a template) | +| `primary.tolerations` | Tolerations for PostgreSQL primary pods assignment | `[]` (evaluated as a template) | +| `primary.anotations` | Map of annotations to add to the statefulset (postgresql primary) | `{}` | +| `primary.labels` | Map of labels to add to the statefulset (postgresql primary) | `{}` | +| `primary.podAnnotations` | Map of annotations to add to the pods (postgresql primary) | `{}` | +| `primary.podLabels` | Map of labels to add to the pods (postgresql primary) | `{}` | +| `primary.priorityClassName` | Priority Class to use for each pod (postgresql primary) | `nil` | +| `primary.extraInitContainers` | Additional init containers to add to the pods (postgresql primary) | `[]` | +| `primary.extraVolumeMounts` | Additional volume mounts to add to the pods (postgresql primary) | `[]` | +| `primary.extraVolumes` | Additional volumes to add to the pods (postgresql primary) | `[]` | +| `primary.sidecars` | Add additional containers to the pod | `[]` | +| `primary.service.type` | Allows using a different service type for primary | `nil` | +| `primary.service.nodePort` | Allows using a different nodePort for primary | `nil` | +| `primary.service.clusterIP` | Allows using a different clusterIP for primary | `nil` | +| `primaryAsStandBy.enabled` | Whether to enable current cluster's primary as standby server of another cluster or not. | `false` | +| `primaryAsStandBy.primaryHost` | The Host of replication primary in the other cluster. | `nil` | +| `primaryAsStandBy.primaryPort ` | The Port of replication primary in the other cluster. | `nil` | +| `readReplicas.podAffinityPreset` | PostgreSQL read only pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `readReplicas.podAntiAffinityPreset` | PostgreSQL read only pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | +| `readReplicas.nodeAffinityPreset.type` | PostgreSQL read only node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `readReplicas.nodeAffinityPreset.key` | PostgreSQL read only node label key to match Ignored if `primary.affinity` is set. | `""` | +| `readReplicas.nodeAffinityPreset.values` | PostgreSQL read only node label values to match. Ignored if `primary.affinity` is set. | `[]` | +| `readReplicas.affinity` | Affinity for PostgreSQL read only pods assignment | `{}` (evaluated as a template) | +| `readReplicas.nodeSelector` | Node labels for PostgreSQL read only pods assignment | `{}` (evaluated as a template) | +| `readReplicas.anotations` | Map of annotations to add to the statefulsets (postgresql readReplicas) | `{}` | +| `readReplicas.resources` | CPU/Memory resource requests/limits override for readReplicass. Will fallback to `values.resources` if not defined. | `{}` | +| `readReplicas.labels` | Map of labels to add to the statefulsets (postgresql readReplicas) | `{}` | +| `readReplicas.podAnnotations` | Map of annotations to add to the pods (postgresql readReplicas) | `{}` | +| `readReplicas.podLabels` | Map of labels to add to the pods (postgresql readReplicas) | `{}` | +| `readReplicas.priorityClassName` | Priority Class to use for each pod (postgresql readReplicas) | `nil` | +| `readReplicas.extraInitContainers` | Additional init containers to add to the pods (postgresql readReplicas) | `[]` | +| `readReplicas.extraVolumeMounts` | Additional volume mounts to add to the pods (postgresql readReplicas) | `[]` | +| `readReplicas.extraVolumes` | Additional volumes to add to the pods (postgresql readReplicas) | `[]` | +| `readReplicas.sidecars` | Add additional containers to the pod | `[]` | +| `readReplicas.service.type` | Allows using a different service type for readReplicas | `nil` | +| `readReplicas.service.nodePort` | Allows using a different nodePort for readReplicas | `nil` | +| `readReplicas.service.clusterIP` | Allows using a different clusterIP for readReplicas | `nil` | +| `readReplicas.persistence.enabled` | Whether to enable readReplicas replicas persistence | `true` | +| `terminationGracePeriodSeconds` | Seconds the pod needs to terminate gracefully | `nil` | +| `resources` | CPU/Memory resource requests/limits | Memory: `256Mi`, CPU: `250m` | +| `securityContext.*` | Other pod security context to be included as-is in the pod spec | `{}` | +| `securityContext.enabled` | Enable security context | `true` | +| `securityContext.fsGroup` | Group ID for the pod | `1001` | +| `containerSecurityContext.*` | Other container security context to be included as-is in the container spec | `{}` | +| `containerSecurityContext.enabled` | Enable container security context | `true` | +| `containerSecurityContext.runAsUser` | User ID for the container | `1001` | +| `serviceAccount.enabled` | Enable service account (Note: Service Account will only be automatically created if `serviceAccount.name` is not set) | `false` | +| `serviceAccount.name` | Name of existing service account | `nil` | +| `networkPolicy.enabled` | Enable NetworkPolicy | `false` | +| `networkPolicy.allowExternal` | Don't require client label for connections | `true` | +| `networkPolicy.explicitNamespacesSelector` | A Kubernetes LabelSelector to explicitly select namespaces from which ingress traffic could be allowed | `{}` | +| `startupProbe.enabled` | Enable startupProbe | `false` | +| `startupProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 30 | +| `startupProbe.periodSeconds` | How often to perform the probe | 15 | +| `startupProbe.timeoutSeconds` | When the probe times | 5 | +| `startupProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 10 | +| `startupProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed. | 1 | +| `livenessProbe.enabled` | Enable livenessProbe | `true` | +| `livenessProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 30 | +| `livenessProbe.periodSeconds` | How often to perform the probe | 10 | +| `livenessProbe.timeoutSeconds` | When the probe times out | 5 | +| `livenessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `livenessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `readinessProbe.enabled` | Enable readinessProbe | `true` | +| `readinessProbe.initialDelaySeconds` | Delay before readiness probe is initiated | 5 | +| `readinessProbe.periodSeconds` | How often to perform the probe | 10 | +| `readinessProbe.timeoutSeconds` | When the probe times out | 5 | +| `readinessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `readinessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `tls.enabled` | Enable TLS traffic support | `false` | +| `tls.preferServerCiphers` | Whether to use the server's TLS cipher preferences rather than the client's | `true` | +| `tls.certificatesSecret` | Name of an existing secret that contains the certificates | `nil` | +| `tls.certFilename` | Certificate filename | `""` | +| `tls.certKeyFilename` | Certificate key filename | `""` | +| `tls.certCAFilename` | CA Certificate filename. If provided, PostgreSQL will authenticate TLS/SSL clients by requesting them a certificate. | `nil` | +| `tls.crlFilename` | File containing a Certificate Revocation List | `nil` | +| `metrics.enabled` | Start a prometheus exporter | `false` | +| `metrics.service.type` | Kubernetes Service type | `ClusterIP` | +| `service.clusterIP` | Static clusterIP or None for headless services | `nil` | +| `metrics.service.annotations` | Additional annotations for metrics exporter pod | `{ prometheus.io/scrape: "true", prometheus.io/port: "9187"}` | +| `metrics.service.loadBalancerIP` | loadBalancerIP if redis metrics service type is `LoadBalancer` | `nil` | +| `metrics.serviceMonitor.enabled` | Set this to `true` to create ServiceMonitor for Prometheus operator | `false` | +| `metrics.serviceMonitor.additionalLabels` | Additional labels that can be used so ServiceMonitor will be discovered by Prometheus | `{}` | +| `metrics.serviceMonitor.namespace` | Optional namespace in which to create ServiceMonitor | `nil` | +| `metrics.serviceMonitor.interval` | Scrape interval. If not set, the Prometheus default scrape interval is used | `nil` | +| `metrics.serviceMonitor.scrapeTimeout` | Scrape timeout. If not set, the Prometheus default scrape timeout is used | `nil` | +| `metrics.prometheusRule.enabled` | Set this to true to create prometheusRules for Prometheus operator | `false` | +| `metrics.prometheusRule.additionalLabels` | Additional labels that can be used so prometheusRules will be discovered by Prometheus | `{}` | +| `metrics.prometheusRule.namespace` | namespace where prometheusRules resource should be created | the same namespace as postgresql | +| `metrics.prometheusRule.rules` | [rules](https://prometheus.io/docs/prometheus/latest/configuration/alerting_rules/) to be created, check values for an example. | `[]` | +| `metrics.image.registry` | PostgreSQL Exporter Image registry | `docker.io` | +| `metrics.image.repository` | PostgreSQL Exporter Image name | `bitnami/postgres-exporter` | +| `metrics.image.tag` | PostgreSQL Exporter Image tag | `{TAG_NAME}` | +| `metrics.image.pullPolicy` | PostgreSQL Exporter Image pull policy | `IfNotPresent` | +| `metrics.image.pullSecrets` | Specify Image pull secrets | `nil` (does not add image pull secrets to deployed pods) | +| `metrics.customMetrics` | Additional custom metrics | `nil` | +| `metrics.extraEnvVars` | Extra environment variables to add to exporter | `{}` (evaluated as a template) | +| `metrics.securityContext.*` | Other container security context to be included as-is in the container spec | `{}` | +| `metrics.securityContext.enabled` | Enable security context for metrics | `false` | +| `metrics.securityContext.runAsUser` | User ID for the container for metrics | `1001` | +| `metrics.livenessProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 30 | +| `metrics.livenessProbe.periodSeconds` | How often to perform the probe | 10 | +| `metrics.livenessProbe.timeoutSeconds` | When the probe times out | 5 | +| `metrics.livenessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `metrics.livenessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `metrics.readinessProbe.enabled` | would you like a readinessProbe to be enabled | `true` | +| `metrics.readinessProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 5 | +| `metrics.readinessProbe.periodSeconds` | How often to perform the probe | 10 | +| `metrics.readinessProbe.timeoutSeconds` | When the probe times out | 5 | +| `metrics.readinessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `metrics.readinessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `updateStrategy` | Update strategy policy | `{type: "RollingUpdate"}` | +| `psp.create` | Create Pod Security Policy | `false` | +| `rbac.create` | Create Role and RoleBinding (required for PSP to work) | `false` | +| `extraDeploy` | Array of extra objects to deploy with the release (evaluated as a template). | `nil` | + +Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, + +```console +$ helm install my-release \ + --set postgresqlPassword=secretpassword,postgresqlDatabase=my-database \ + bitnami/postgresql +``` + +The above command sets the PostgreSQL `postgres` account password to `secretpassword`. Additionally it creates a database named `my-database`. + +> NOTE: Once this chart is deployed, it is not possible to change the application's access credentials, such as usernames or passwords, using Helm. To change these application credentials after deployment, delete any persistent volumes (PVs) used by the chart and re-deploy it, or use the application's built-in administrative tools if available. + +Alternatively, a YAML file that specifies the values for the parameters can be provided while installing the chart. For example, + +```console +$ helm install my-release -f values.yaml bitnami/postgresql +``` + +> **Tip**: You can use the default [values.yaml](values.yaml) + +## Configuration and installation details + +### [Rolling VS Immutable tags](https://docs.bitnami.com/containers/how-to/understand-rolling-tags-containers/) + +It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. + +Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. + +### Customizing primary and read replica services in a replicated configuration + +At the top level, there is a service object which defines the services for both primary and readReplicas. For deeper customization, there are service objects for both the primary and read types individually. This allows you to override the values in the top level service object so that the primary and read can be of different service types and with different clusterIPs / nodePorts. Also in the case you want the primary and read to be of type nodePort, you will need to set the nodePorts to different values to prevent a collision. The values that are deeper in the primary.service or readReplicas.service objects will take precedence over the top level service object. + +### Change PostgreSQL version + +To modify the PostgreSQL version used in this chart you can specify a [valid image tag](https://hub.docker.com/r/bitnami/postgresql/tags/) using the `image.tag` parameter. For example, `image.tag=X.Y.Z`. This approach is also applicable to other images like exporters. + +### postgresql.conf / pg_hba.conf files as configMap + +This helm chart also supports to customize the whole configuration file. + +Add your custom file to "files/postgresql.conf" in your working directory. This file will be mounted as configMap to the containers and it will be used for configuring the PostgreSQL server. + +Alternatively, you can add additional PostgreSQL configuration parameters using the `postgresqlExtendedConf` parameter as a dict, using camelCase, e.g. {"sharedBuffers": "500MB"}. Alternatively, to replace the entire default configuration use `postgresqlConfiguration`. + +In addition to these options, you can also set an external ConfigMap with all the configuration files. This is done by setting the `configurationConfigMap` parameter. Note that this will override the two previous options. + +### Allow settings to be loaded from files other than the default `postgresql.conf` + +If you don't want to provide the whole PostgreSQL configuration file and only specify certain parameters, you can add your extended `.conf` files to "files/conf.d/" in your working directory. +Those files will be mounted as configMap to the containers adding/overwriting the default configuration using the `include_dir` directive that allows settings to be loaded from files other than the default `postgresql.conf`. + +Alternatively, you can also set an external ConfigMap with all the extra configuration files. This is done by setting the `extendedConfConfigMap` parameter. Note that this will override the previous option. + +### Initialize a fresh instance + +The [Bitnami PostgreSQL](https://github.com/bitnami/bitnami-docker-postgresql) image allows you to use your custom scripts to initialize a fresh instance. In order to execute the scripts, they must be located inside the chart folder `files/docker-entrypoint-initdb.d` so they can be consumed as a ConfigMap. + +Alternatively, you can specify custom scripts using the `initdbScripts` parameter as dict. + +In addition to these options, you can also set an external ConfigMap with all the initialization scripts. This is done by setting the `initdbScriptsConfigMap` parameter. Note that this will override the two previous options. If your initialization scripts contain sensitive information such as credentials or passwords, you can use the `initdbScriptsSecret` parameter. + +The allowed extensions are `.sh`, `.sql` and `.sql.gz`. + +### Securing traffic using TLS + +TLS support can be enabled in the chart by specifying the `tls.` parameters while creating a release. The following parameters should be configured to properly enable the TLS support in the chart: + +- `tls.enabled`: Enable TLS support. Defaults to `false` +- `tls.certificatesSecret`: Name of an existing secret that contains the certificates. No defaults. +- `tls.certFilename`: Certificate filename. No defaults. +- `tls.certKeyFilename`: Certificate key filename. No defaults. + +For example: + +* First, create the secret with the cetificates files: + + ```console + kubectl create secret generic certificates-tls-secret --from-file=./cert.crt --from-file=./cert.key --from-file=./ca.crt + ``` + +* Then, use the following parameters: + + ```console + volumePermissions.enabled=true + tls.enabled=true + tls.certificatesSecret="certificates-tls-secret" + tls.certFilename="cert.crt" + tls.certKeyFilename="cert.key" + ``` + + > Note TLS and VolumePermissions: PostgreSQL requires certain permissions on sensitive files (such as certificate keys) to start up. Due to an on-going [issue](https://github.com/kubernetes/kubernetes/issues/57923) regarding kubernetes permissions and the use of `containerSecurityContext.runAsUser`, you must enable `volumePermissions` to ensure everything works as expected. + +### Sidecars + +If you need additional containers to run within the same pod as PostgreSQL (e.g. an additional metrics or logging exporter), you can do so via the `sidecars` config parameter. Simply define your container according to the Kubernetes container spec. + +```yaml +# For the PostgreSQL primary +primary: + sidecars: + - name: your-image-name + image: your-image + imagePullPolicy: Always + ports: + - name: portname + containerPort: 1234 +# For the PostgreSQL replicas +readReplicas: + sidecars: + - name: your-image-name + image: your-image + imagePullPolicy: Always + ports: + - name: portname + containerPort: 1234 +``` + +### Metrics + +The chart optionally can start a metrics exporter for [prometheus](https://prometheus.io). The metrics endpoint (port 9187) is not exposed and it is expected that the metrics are collected from inside the k8s cluster using something similar as the described in the [example Prometheus scrape configuration](https://github.com/prometheus/prometheus/blob/master/documentation/examples/prometheus-kubernetes.yml). + +The exporter allows to create custom metrics from additional SQL queries. See the Chart's `values.yaml` for an example and consult the [exporters documentation](https://github.com/wrouesnel/postgres_exporter#adding-new-metrics-via-a-config-file) for more details. + +### Use of global variables + +In more complex scenarios, we may have the following tree of dependencies + +``` + +--------------+ + | | + +------------+ Chart 1 +-----------+ + | | | | + | --------+------+ | + | | | + | | | + | | | + | | | + v v v ++-------+------+ +--------+------+ +--------+------+ +| | | | | | +| PostgreSQL | | Sub-chart 1 | | Sub-chart 2 | +| | | | | | ++--------------+ +---------------+ +---------------+ +``` + +The three charts below depend on the parent chart Chart 1. However, subcharts 1 and 2 may need to connect to PostgreSQL as well. In order to do so, subcharts 1 and 2 need to know the PostgreSQL credentials, so one option for deploying could be deploy Chart 1 with the following parameters: + +``` +postgresql.postgresqlPassword=testtest +subchart1.postgresql.postgresqlPassword=testtest +subchart2.postgresql.postgresqlPassword=testtest +postgresql.postgresqlDatabase=db1 +subchart1.postgresql.postgresqlDatabase=db1 +subchart2.postgresql.postgresqlDatabase=db1 +``` + +If the number of dependent sub-charts increases, installing the chart with parameters can become increasingly difficult. An alternative would be to set the credentials using global variables as follows: + +``` +global.postgresql.postgresqlPassword=testtest +global.postgresql.postgresqlDatabase=db1 +``` + +This way, the credentials will be available in all of the subcharts. + +## Persistence + +The [Bitnami PostgreSQL](https://github.com/bitnami/bitnami-docker-postgresql) image stores the PostgreSQL data and configurations at the `/bitnami/postgresql` path of the container. + +Persistent Volume Claims are used to keep the data across deployments. This is known to work in GCE, AWS, and minikube. +See the [Parameters](#parameters) section to configure the PVC or to disable persistence. + +If you already have data in it, you will fail to sync to standby nodes for all commits, details can refer to [code](https://github.com/bitnami/bitnami-docker-postgresql/blob/8725fe1d7d30ebe8d9a16e9175d05f7ad9260c93/9.6/debian-9/rootfs/libpostgresql.sh#L518-L556). If you need to use those data, please covert them to sql and import after `helm install` finished. + +## NetworkPolicy + +To enable network policy for PostgreSQL, install [a networking plugin that implements the Kubernetes NetworkPolicy spec](https://kubernetes.io/docs/tasks/administer-cluster/declare-network-policy#before-you-begin), and set `networkPolicy.enabled` to `true`. + +For Kubernetes v1.5 & v1.6, you must also turn on NetworkPolicy by setting the DefaultDeny namespace annotation. Note: this will enforce policy for _all_ pods in the namespace: + +```console +$ kubectl annotate namespace default "net.beta.kubernetes.io/network-policy={\"ingress\":{\"isolation\":\"DefaultDeny\"}}" +``` + +With NetworkPolicy enabled, traffic will be limited to just port 5432. + +For more precise policy, set `networkPolicy.allowExternal=false`. This will only allow pods with the generated client label to connect to PostgreSQL. +This label will be displayed in the output of a successful install. + +## Differences between Bitnami PostgreSQL image and [Docker Official](https://hub.docker.com/_/postgres) image + +- The Docker Official PostgreSQL image does not support replication. If you pass any replication environment variable, this would be ignored. The only environment variables supported by the Docker Official image are POSTGRES_USER, POSTGRES_DB, POSTGRES_PASSWORD, POSTGRES_INITDB_ARGS, POSTGRES_INITDB_WALDIR and PGDATA. All the remaining environment variables are specific to the Bitnami PostgreSQL image. +- The Bitnami PostgreSQL image is non-root by default. This requires that you run the pod with `securityContext` and updates the permissions of the volume with an `initContainer`. A key benefit of this configuration is that the pod follows security best practices and is prepared to run on Kubernetes distributions with hard security constraints like OpenShift. +- For OpenShift, one may either define the runAsUser and fsGroup accordingly, or try this more dynamic option: volumePermissions.securityContext.runAsUser="auto",securityContext.enabled=false,containerSecurityContext.enabled=false,shmVolume.chmod.enabled=false + +### Deploy chart using Docker Official PostgreSQL Image + +From chart version 4.0.0, it is possible to use this chart with the Docker Official PostgreSQL image. +Besides specifying the new Docker repository and tag, it is important to modify the PostgreSQL data directory and volume mount point. Basically, the PostgreSQL data dir cannot be the mount point directly, it has to be a subdirectory. + +``` +image.repository=postgres +image.tag=10.6 +postgresqlDataDir=/data/pgdata +persistence.mountPath=/data/ +``` + +### Setting Pod's affinity + +This chart allows you to set your custom affinity using the `XXX.affinity` paremeter(s). Find more infomation about Pod's affinity in the [kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity). + +As an alternative, you can use of the preset configurations for pod affinity, pod anti-affinity, and node affinity available at the [bitnami/common](https://github.com/bitnami/charts/tree/master/bitnami/common#affinities) chart. To do so, set the `XXX.podAffinityPreset`, `XXX.podAntiAffinityPreset`, or `XXX.nodeAffinityPreset` parameters. + +## Troubleshooting + +Find more information about how to deal with common errors related to Bitnami’s Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). + +## Upgrading + +It's necessary to specify the existing passwords while performing an upgrade to ensure the secrets are not updated with invalid randomly generated passwords. Remember to specify the existing values of the `postgresqlPassword` and `replication.password` parameters when upgrading the chart: + +```bash +$ helm upgrade my-release bitnami/postgresql \ + --set postgresqlPassword=[POSTGRESQL_PASSWORD] \ + --set replication.password=[REPLICATION_PASSWORD] +``` + +> Note: you need to substitute the placeholders _[POSTGRESQL_PASSWORD]_, and _[REPLICATION_PASSWORD]_ with the values obtained from instructions in the installation notes. + +### To 10.0.0 + +[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. + +**What changes were introduced in this major version?** + +- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. +- Move dependency information from the *requirements.yaml* to the *Chart.yaml* +- After running `helm dependency update`, a *Chart.lock* file is generated containing the same structure used in the previous *requirements.lock* +- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Chart. + +**Considerations when upgrading to this version** + +- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore +- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 + +**Useful links** + +- https://docs.bitnami.com/tutorials/resolve-helm2-helm3-post-migration-issues/ +- https://helm.sh/docs/topics/v2_v3_migration/ +- https://helm.sh/blog/migrate-from-helm-v2-to-helm-v3/ + +#### Breaking changes + +- The term `master` has been replaced with `primary` and `slave` with `readReplicas` throughout the chart. Role names have changed from `master` and `slave` to `primary` and `read`. + +To upgrade to `10.0.0`, it should be done reusing the PVCs used to hold the PostgreSQL data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is `postgresql`): + +> NOTE: Please, create a backup of your database before running any of those actions. + +Obtain the credentials and the names of the PVCs used to hold the PostgreSQL data on your current release: + +```console +$ export POSTGRESQL_PASSWORD=$(kubectl get secret --namespace default postgresql -o jsonpath="{.data.postgresql-password}" | base64 --decode) +$ export POSTGRESQL_PVC=$(kubectl get pvc -l app.kubernetes.io/instance=postgresql,role=master -o jsonpath="{.items[0].metadata.name}") +``` + +Delete the PostgreSQL statefulset. Notice the option `--cascade=false`: + +```console +$ kubectl delete statefulsets.apps postgresql-postgresql --cascade=false +``` + +Now the upgrade works: + +```console +$ helm upgrade postgresql bitnami/postgresql --set postgresqlPassword=$POSTGRESQL_PASSWORD --set persistence.existingClaim=$POSTGRESQL_PVC +``` + +You will have to delete the existing PostgreSQL pod and the new statefulset is going to create a new one + +```console +$ kubectl delete pod postgresql-postgresql-0 +``` + +Finally, you should see the lines below in PostgreSQL container logs: + +```console +$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=postgresql,app.kubernetes.io/name=postgresql,role=primary -o jsonpath="{.items[0].metadata.name}") +... +postgresql 08:05:12.59 INFO ==> Deploying PostgreSQL with persisted data... +... +``` + +### To 9.0.0 + +In this version the chart was adapted to follow the Helm label best practices, see [PR 3021](https://github.com/bitnami/charts/pull/3021). That means the backward compatibility is not guarantee when upgrading the chart to this major version. + +As a workaround, you can delete the existing statefulset (using the `--cascade=false` flag pods are not deleted) before upgrade the chart. For example, this can be a valid workflow: + +- Deploy an old version (8.X.X) + +```console +$ helm install postgresql bitnami/postgresql --version 8.10.14 +``` + +- Old version is up and running + +```console +$ helm ls +NAME NAMESPACE REVISION UPDATED STATUS CHART APP VERSION +postgresql default 1 2020-08-04 13:39:54.783480286 +0000 UTC deployed postgresql-8.10.14 11.8.0 + +$ kubectl get pods +NAME READY STATUS RESTARTS AGE +postgresql-postgresql-0 1/1 Running 0 76s +``` + +- The upgrade to the latest one (9.X.X) is going to fail + +```console +$ helm upgrade postgresql bitnami/postgresql +Error: UPGRADE FAILED: cannot patch "postgresql-postgresql" with kind StatefulSet: StatefulSet.apps "postgresql-postgresql" is invalid: spec: Forbidden: updates to statefulset spec for fields other than 'replicas', 'template', and 'updateStrategy' are forbidden +``` + +- Delete the statefulset + +```console +$ kubectl delete statefulsets.apps --cascade=false postgresql-postgresql +statefulset.apps "postgresql-postgresql" deleted +``` + +- Now the upgrade works + +```console +$ helm upgrade postgresql bitnami/postgresql +$ helm ls +NAME NAMESPACE REVISION UPDATED STATUS CHART APP VERSION +postgresql default 3 2020-08-04 13:42:08.020385884 +0000 UTC deployed postgresql-9.1.2 11.8.0 +``` + +- We can kill the existing pod and the new statefulset is going to create a new one: + +```console +$ kubectl delete pod postgresql-postgresql-0 +pod "postgresql-postgresql-0" deleted + +$ kubectl get pods +NAME READY STATUS RESTARTS AGE +postgresql-postgresql-0 1/1 Running 0 19s +``` + +Please, note that without the `--cascade=false` both objects (statefulset and pod) are going to be removed and both objects will be deployed again with the `helm upgrade` command + +### To 8.0.0 + +Prefixes the port names with their protocols to comply with Istio conventions. + +If you depend on the port names in your setup, make sure to update them to reflect this change. + +### To 7.1.0 + +Adds support for LDAP configuration. + +### To 7.0.0 + +Helm performs a lookup for the object based on its group (apps), version (v1), and kind (Deployment). Also known as its GroupVersionKind, or GVK. Changing the GVK is considered a compatibility breaker from Kubernetes' point of view, so you cannot "upgrade" those objects to the new GVK in-place. Earlier versions of Helm 3 did not perform the lookup correctly which has since been fixed to match the spec. + +In https://github.com/helm/charts/pull/17281 the `apiVersion` of the statefulset resources was updated to `apps/v1` in tune with the api's deprecated, resulting in compatibility breakage. + +This major version bump signifies this change. + +### To 6.5.7 + +In this version, the chart will use PostgreSQL with the Postgis extension included. The version used with Postgresql version 10, 11 and 12 is Postgis 2.5. It has been compiled with the following dependencies: + +- protobuf +- protobuf-c +- json-c +- geos +- proj + +### To 5.0.0 + +In this version, the **chart is using PostgreSQL 11 instead of PostgreSQL 10**. You can find the main difference and notable changes in the following links: [https://www.postgresql.org/about/news/1894/](https://www.postgresql.org/about/news/1894/) and [https://www.postgresql.org/about/featurematrix/](https://www.postgresql.org/about/featurematrix/). + +For major releases of PostgreSQL, the internal data storage format is subject to change, thus complicating upgrades, you can see some errors like the following one in the logs: + +```console +Welcome to the Bitnami postgresql container +Subscribe to project updates by watching https://github.com/bitnami/bitnami-docker-postgresql +Submit issues and feature requests at https://github.com/bitnami/bitnami-docker-postgresql/issues +Send us your feedback at containers@bitnami.com + +INFO ==> ** Starting PostgreSQL setup ** +NFO ==> Validating settings in POSTGRESQL_* env vars.. +INFO ==> Initializing PostgreSQL database... +INFO ==> postgresql.conf file not detected. Generating it... +INFO ==> pg_hba.conf file not detected. Generating it... +INFO ==> Deploying PostgreSQL with persisted data... +INFO ==> Configuring replication parameters +INFO ==> Loading custom scripts... +INFO ==> Enabling remote connections +INFO ==> Stopping PostgreSQL... +INFO ==> ** PostgreSQL setup finished! ** + +INFO ==> ** Starting PostgreSQL ** + [1] FATAL: database files are incompatible with server + [1] DETAIL: The data directory was initialized by PostgreSQL version 10, which is not compatible with this version 11.3. +``` + +In this case, you should migrate the data from the old chart to the new one following an approach similar to that described in [this section](https://www.postgresql.org/docs/current/upgrading.html#UPGRADING-VIA-PGDUMPALL) from the official documentation. Basically, create a database dump in the old chart, move and restore it in the new one. + +### To 4.0.0 + +This chart will use by default the Bitnami PostgreSQL container starting from version `10.7.0-r68`. This version moves the initialization logic from node.js to bash. This new version of the chart requires setting the `POSTGRES_PASSWORD` in the slaves as well, in order to properly configure the `pg_hba.conf` file. Users from previous versions of the chart are advised to upgrade immediately. + +IMPORTANT: If you do not want to upgrade the chart version then make sure you use the `10.7.0-r68` version of the container. Otherwise, you will get this error + +``` +The POSTGRESQL_PASSWORD environment variable is empty or not set. Set the environment variable ALLOW_EMPTY_PASSWORD=yes to allow the container to be started with blank passwords. This is recommended only for development +``` + +### To 3.0.0 + +This releases make it possible to specify different nodeSelector, affinity and tolerations for master and slave pods. +It also fixes an issue with `postgresql.master.fullname` helper template not obeying fullnameOverride. + +#### Breaking changes + +- `affinty` has been renamed to `master.affinity` and `slave.affinity`. +- `tolerations` has been renamed to `master.tolerations` and `slave.tolerations`. +- `nodeSelector` has been renamed to `master.nodeSelector` and `slave.nodeSelector`. + +### To 2.0.0 + +In order to upgrade from the `0.X.X` branch to `1.X.X`, you should follow the below steps: + +- Obtain the service name (`SERVICE_NAME`) and password (`OLD_PASSWORD`) of the existing postgresql chart. You can find the instructions to obtain the password in the NOTES.txt, the service name can be obtained by running + +```console +$ kubectl get svc +``` + +- Install (not upgrade) the new version + +```console +$ helm repo update +$ helm install my-release bitnami/postgresql +``` + +- Connect to the new pod (you can obtain the name by running `kubectl get pods`): + +```console +$ kubectl exec -it NAME bash +``` + +- Once logged in, create a dump file from the previous database using `pg_dump`, for that we should connect to the previous postgresql chart: + +```console +$ pg_dump -h SERVICE_NAME -U postgres DATABASE_NAME > /tmp/backup.sql +``` + +After run above command you should be prompted for a password, this password is the previous chart password (`OLD_PASSWORD`). +This operation could take some time depending on the database size. + +- Once you have the backup file, you can restore it with a command like the one below: + +```console +$ psql -U postgres DATABASE_NAME < /tmp/backup.sql +``` + +In this case, you are accessing to the local postgresql, so the password should be the new one (you can find it in NOTES.txt). + +If you want to restore the database and the database schema does not exist, it is necessary to first follow the steps described below. + +```console +$ psql -U postgres +postgres=# drop database DATABASE_NAME; +postgres=# create database DATABASE_NAME; +postgres=# create user USER_NAME; +postgres=# alter role USER_NAME with password 'BITNAMI_USER_PASSWORD'; +postgres=# grant all privileges on database DATABASE_NAME to USER_NAME; +postgres=# alter database DATABASE_NAME owner to USER_NAME; +``` diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/.helmignore b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/.helmignore new file mode 100644 index 0000000000..50af031725 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/Chart.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/Chart.yaml new file mode 100644 index 0000000000..bcc3808d08 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/Chart.yaml @@ -0,0 +1,23 @@ +annotations: + category: Infrastructure +apiVersion: v2 +appVersion: 1.4.2 +description: A Library Helm Chart for grouping common logic between bitnami charts. + This chart is not deployable by itself. +home: https://github.com/bitnami/charts/tree/master/bitnami/common +icon: https://bitnami.com/downloads/logos/bitnami-mark.png +keywords: +- common +- helper +- template +- function +- bitnami +maintainers: +- email: containers@bitnami.com + name: Bitnami +name: common +sources: +- https://github.com/bitnami/charts +- http://www.bitnami.com/ +type: library +version: 1.4.2 diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/README.md b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/README.md new file mode 100644 index 0000000000..7287cbb5fc --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/README.md @@ -0,0 +1,322 @@ +# Bitnami Common Library Chart + +A [Helm Library Chart](https://helm.sh/docs/topics/library_charts/#helm) for grouping common logic between bitnami charts. + +## TL;DR + +```yaml +dependencies: + - name: common + version: 0.x.x + repository: https://charts.bitnami.com/bitnami +``` + +```bash +$ helm dependency update +``` + +```yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "common.names.fullname" . }} +data: + myvalue: "Hello World" +``` + +## Introduction + +This chart provides a common template helpers which can be used to develop new charts using [Helm](https://helm.sh) package manager. + +Bitnami charts can be used with [Kubeapps](https://kubeapps.com/) for deployment and management of Helm Charts in clusters. This Helm chart has been tested on top of [Bitnami Kubernetes Production Runtime](https://kubeprod.io/) (BKPR). Deploy BKPR to get automated TLS certificates, logging and monitoring for your applications. + +## Prerequisites + +- Kubernetes 1.12+ +- Helm 3.1.0 + +## Parameters + +The following table lists the helpers available in the library which are scoped in different sections. + +### Affinities + +| Helper identifier | Description | Expected Input | +|-------------------------------|------------------------------------------------------|------------------------------------------------| +| `common.affinities.node.soft` | Return a soft nodeAffinity definition | `dict "key" "FOO" "values" (list "BAR" "BAZ")` | +| `common.affinities.node.hard` | Return a hard nodeAffinity definition | `dict "key" "FOO" "values" (list "BAR" "BAZ")` | +| `common.affinities.pod.soft` | Return a soft podAffinity/podAntiAffinity definition | `dict "component" "FOO" "context" $` | +| `common.affinities.pod.hard` | Return a hard podAffinity/podAntiAffinity definition | `dict "component" "FOO" "context" $` | + +### Capabilities + +| Helper identifier | Description | Expected Input | +|----------------------------------------------|------------------------------------------------------------------------------------------------|-------------------| +| `common.capabilities.kubeVersion` | Return the target Kubernetes version (using client default if .Values.kubeVersion is not set). | `.` Chart context | +| `common.capabilities.deployment.apiVersion` | Return the appropriate apiVersion for deployment. | `.` Chart context | +| `common.capabilities.statefulset.apiVersion` | Return the appropriate apiVersion for statefulset. | `.` Chart context | +| `common.capabilities.ingress.apiVersion` | Return the appropriate apiVersion for ingress. | `.` Chart context | +| `common.capabilities.rbac.apiVersion` | Return the appropriate apiVersion for RBAC resources. | `.` Chart context | +| `common.capabilities.crd.apiVersion` | Return the appropriate apiVersion for CRDs. | `.` Chart context | +| `common.capabilities.supportsHelmVersion` | Returns true if the used Helm version is 3.3+ | `.` Chart context | + +### Errors + +| Helper identifier | Description | Expected Input | +|-----------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------| +| `common.errors.upgrade.passwords.empty` | It will ensure required passwords are given when we are upgrading a chart. If `validationErrors` is not empty it will throw an error and will stop the upgrade action. | `dict "validationErrors" (list $validationError00 $validationError01) "context" $` | + +### Images + +| Helper identifier | Description | Expected Input | +|-----------------------------|------------------------------------------------------|---------------------------------------------------------------------------------------------------------| +| `common.images.image` | Return the proper and full image name | `dict "imageRoot" .Values.path.to.the.image "global" $`, see [ImageRoot](#imageroot) for the structure. | +| `common.images.pullSecrets` | Return the proper Docker Image Registry Secret Names | `dict "images" (list .Values.path.to.the.image1, .Values.path.to.the.image2) "global" .Values.global` | + +### Ingress + +| Helper identifier | Description | Expected Input | +|--------------------------|----------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.ingress.backend` | Generate a proper Ingress backend entry depending on the API version | `dict "serviceName" "foo" "servicePort" "bar"`, see the [Ingress deprecation notice](https://kubernetes.io/blog/2019/07/18/api-deprecations-in-1-16/) for the syntax differences | + +### Labels + +| Helper identifier | Description | Expected Input | +|-----------------------------|------------------------------------------------------|-------------------| +| `common.labels.standard` | Return Kubernetes standard labels | `.` Chart context | +| `common.labels.matchLabels` | Return the proper Docker Image Registry Secret Names | `.` Chart context | + +### Names + +| Helper identifier | Description | Expected Inpput | +|-------------------------|------------------------------------------------------------|-------------------| +| `common.names.name` | Expand the name of the chart or use `.Values.nameOverride` | `.` Chart context | +| `common.names.fullname` | Create a default fully qualified app name. | `.` Chart context | +| `common.names.chart` | Chart name plus version | `.` Chart context | + +### Secrets + +| Helper identifier | Description | Expected Input | +|---------------------------|--------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.secrets.name` | Generate the name of the secret. | `dict "existingSecret" .Values.path.to.the.existingSecret "defaultNameSuffix" "mySuffix" "context" $` see [ExistingSecret](#existingsecret) for the structure. | +| `common.secrets.key` | Generate secret key. | `dict "existingSecret" .Values.path.to.the.existingSecret "key" "keyName"` see [ExistingSecret](#existingsecret) for the structure. | +| `common.passwords.manage` | Generate secret password or retrieve one if already created. | `dict "secret" "secret-name" "key" "keyName" "providedValues" (list "path.to.password1" "path.to.password2") "length" 10 "strong" false "chartName" "chartName" "context" $`, length, strong and chartNAme fields are optional. | +| `common.secrets.exists` | Returns whether a previous generated secret already exists. | `dict "secret" "secret-name" "context" $` | + +### Storage + +| Helper identifier | Description | Expected Input | +|-------------------------------|---------------------------------------|---------------------------------------------------------------------------------------------------------------------| +| `common.affinities.node.soft` | Return a soft nodeAffinity definition | `dict "persistence" .Values.path.to.the.persistence "global" $`, see [Persistence](#persistence) for the structure. | + +### TplValues + +| Helper identifier | Description | Expected Input | +|---------------------------|----------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.tplvalues.render` | Renders a value that contains template | `dict "value" .Values.path.to.the.Value "context" $`, value is the value should rendered as template, context frequently is the chart context `$` or `.` | + +### Utils + +| Helper identifier | Description | Expected Input | +|--------------------------------|------------------------------------------------------------------------------------------|------------------------------------------------------------------------| +| `common.utils.fieldToEnvVar` | Build environment variable name given a field. | `dict "field" "my-password"` | +| `common.utils.secret.getvalue` | Print instructions to get a secret value. | `dict "secret" "secret-name" "field" "secret-value-field" "context" $` | +| `common.utils.getValueFromKey` | Gets a value from `.Values` object given its key path | `dict "key" "path.to.key" "context" $` | +| `common.utils.getKeyFromList` | Returns first `.Values` key with a defined value or first of the list if all non-defined | `dict "keys" (list "path.to.key1" "path.to.key2") "context" $` | + +### Validations + +| Helper identifier | Description | Expected Input | +|--------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.validations.values.single.empty` | Validate a value must not be empty. | `dict "valueKey" "path.to.value" "secret" "secret.name" "field" "my-password" "subchart" "subchart" "context" $` secret, field and subchart are optional. In case they are given, the helper will generate a how to get instruction. See [ValidateValue](#validatevalue) | +| `common.validations.values.multiple.empty` | Validate a multiple values must not be empty. It returns a shared error for all the values. | `dict "required" (list $validateValueConf00 $validateValueConf01) "context" $`. See [ValidateValue](#validatevalue) | +| `common.validations.values.mariadb.passwords` | This helper will ensure required password for MariaDB are not empty. It returns a shared error for all the values. | `dict "secret" "mariadb-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use mariadb chart and the helper. | +| `common.validations.values.postgresql.passwords` | This helper will ensure required password for PostgreSQL are not empty. It returns a shared error for all the values. | `dict "secret" "postgresql-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use postgresql chart and the helper. | +| `common.validations.values.redis.passwords` | This helper will ensure required password for RedisTM are not empty. It returns a shared error for all the values. | `dict "secret" "redis-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use redis chart and the helper. | +| `common.validations.values.cassandra.passwords` | This helper will ensure required password for Cassandra are not empty. It returns a shared error for all the values. | `dict "secret" "cassandra-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use cassandra chart and the helper. | +| `common.validations.values.mongodb.passwords` | This helper will ensure required password for MongoDB® are not empty. It returns a shared error for all the values. | `dict "secret" "mongodb-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use mongodb chart and the helper. | + +### Warnings + +| Helper identifier | Description | Expected Input | +|------------------------------|----------------------------------|------------------------------------------------------------| +| `common.warnings.rollingTag` | Warning about using rolling tag. | `ImageRoot` see [ImageRoot](#imageroot) for the structure. | + +## Special input schemas + +### ImageRoot + +```yaml +registry: + type: string + description: Docker registry where the image is located + example: docker.io + +repository: + type: string + description: Repository and image name + example: bitnami/nginx + +tag: + type: string + description: image tag + example: 1.16.1-debian-10-r63 + +pullPolicy: + type: string + description: Specify a imagePullPolicy. Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' + +pullSecrets: + type: array + items: + type: string + description: Optionally specify an array of imagePullSecrets. + +debug: + type: boolean + description: Set to true if you would like to see extra information on logs + example: false + +## An instance would be: +# registry: docker.io +# repository: bitnami/nginx +# tag: 1.16.1-debian-10-r63 +# pullPolicy: IfNotPresent +# debug: false +``` + +### Persistence + +```yaml +enabled: + type: boolean + description: Whether enable persistence. + example: true + +storageClass: + type: string + description: Ghost data Persistent Volume Storage Class, If set to "-", storageClassName: "" which disables dynamic provisioning. + example: "-" + +accessMode: + type: string + description: Access mode for the Persistent Volume Storage. + example: ReadWriteOnce + +size: + type: string + description: Size the Persistent Volume Storage. + example: 8Gi + +path: + type: string + description: Path to be persisted. + example: /bitnami + +## An instance would be: +# enabled: true +# storageClass: "-" +# accessMode: ReadWriteOnce +# size: 8Gi +# path: /bitnami +``` + +### ExistingSecret + +```yaml +name: + type: string + description: Name of the existing secret. + example: mySecret +keyMapping: + description: Mapping between the expected key name and the name of the key in the existing secret. + type: object + +## An instance would be: +# name: mySecret +# keyMapping: +# password: myPasswordKey +``` + +#### Example of use + +When we store sensitive data for a deployment in a secret, some times we want to give to users the possibility of using theirs existing secrets. + +```yaml +# templates/secret.yaml +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "common.names.fullname" . }} + labels: + app: {{ include "common.names.fullname" . }} +type: Opaque +data: + password: {{ .Values.password | b64enc | quote }} + +# templates/dpl.yaml +--- +... + env: + - name: PASSWORD + valueFrom: + secretKeyRef: + name: {{ include "common.secrets.name" (dict "existingSecret" .Values.existingSecret "context" $) }} + key: {{ include "common.secrets.key" (dict "existingSecret" .Values.existingSecret "key" "password") }} +... + +# values.yaml +--- +name: mySecret +keyMapping: + password: myPasswordKey +``` + +### ValidateValue + +#### NOTES.txt + +```console +{{- $validateValueConf00 := (dict "valueKey" "path.to.value00" "secret" "secretName" "field" "password-00") -}} +{{- $validateValueConf01 := (dict "valueKey" "path.to.value01" "secret" "secretName" "field" "password-01") -}} + +{{ include "common.validations.values.multiple.empty" (dict "required" (list $validateValueConf00 $validateValueConf01) "context" $) }} +``` + +If we force those values to be empty we will see some alerts + +```console +$ helm install test mychart --set path.to.value00="",path.to.value01="" + 'path.to.value00' must not be empty, please add '--set path.to.value00=$PASSWORD_00' to the command. To get the current value: + + export PASSWORD_00=$(kubectl get secret --namespace default secretName -o jsonpath="{.data.password-00}" | base64 --decode) + + 'path.to.value01' must not be empty, please add '--set path.to.value01=$PASSWORD_01' to the command. To get the current value: + + export PASSWORD_01=$(kubectl get secret --namespace default secretName -o jsonpath="{.data.password-01}" | base64 --decode) +``` + +## Upgrading + +### To 1.0.0 + +[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. + +**What changes were introduced in this major version?** + +- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. +- Use `type: library`. [Here](https://v3.helm.sh/docs/faq/#library-chart-support) you can find more information. +- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Charts + +**Considerations when upgrading to this version** + +- If you want to upgrade to this version from a previous one installed with Helm v3, you shouldn't face any issues +- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore +- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 + +**Useful links** + +- https://docs.bitnami.com/tutorials/resolve-helm2-helm3-post-migration-issues/ +- https://helm.sh/docs/topics/v2_v3_migration/ +- https://helm.sh/blog/migrate-from-helm-v2-to-helm-v3/ diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_affinities.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_affinities.tpl new file mode 100644 index 0000000000..493a6dc7e4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_affinities.tpl @@ -0,0 +1,94 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Return a soft nodeAffinity definition +{{ include "common.affinities.nodes.soft" (dict "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.nodes.soft" -}} +preferredDuringSchedulingIgnoredDuringExecution: + - preference: + matchExpressions: + - key: {{ .key }} + operator: In + values: + {{- range .values }} + - {{ . }} + {{- end }} + weight: 1 +{{- end -}} + +{{/* +Return a hard nodeAffinity definition +{{ include "common.affinities.nodes.hard" (dict "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.nodes.hard" -}} +requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: {{ .key }} + operator: In + values: + {{- range .values }} + - {{ . }} + {{- end }} +{{- end -}} + +{{/* +Return a nodeAffinity definition +{{ include "common.affinities.nodes" (dict "type" "soft" "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.nodes" -}} + {{- if eq .type "soft" }} + {{- include "common.affinities.nodes.soft" . -}} + {{- else if eq .type "hard" }} + {{- include "common.affinities.nodes.hard" . -}} + {{- end -}} +{{- end -}} + +{{/* +Return a soft podAffinity/podAntiAffinity definition +{{ include "common.affinities.pods.soft" (dict "component" "FOO" "context" $) -}} +*/}} +{{- define "common.affinities.pods.soft" -}} +{{- $component := default "" .component -}} +preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchLabels: {{- (include "common.labels.matchLabels" .context) | nindent 10 }} + {{- if not (empty $component) }} + {{ printf "app.kubernetes.io/component: %s" $component }} + {{- end }} + namespaces: + - {{ .context.Release.Namespace | quote }} + topologyKey: kubernetes.io/hostname + weight: 1 +{{- end -}} + +{{/* +Return a hard podAffinity/podAntiAffinity definition +{{ include "common.affinities.pods.hard" (dict "component" "FOO" "context" $) -}} +*/}} +{{- define "common.affinities.pods.hard" -}} +{{- $component := default "" .component -}} +requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: {{- (include "common.labels.matchLabels" .context) | nindent 8 }} + {{- if not (empty $component) }} + {{ printf "app.kubernetes.io/component: %s" $component }} + {{- end }} + namespaces: + - {{ .context.Release.Namespace | quote }} + topologyKey: kubernetes.io/hostname +{{- end -}} + +{{/* +Return a podAffinity/podAntiAffinity definition +{{ include "common.affinities.pods" (dict "type" "soft" "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.pods" -}} + {{- if eq .type "soft" }} + {{- include "common.affinities.pods.soft" . -}} + {{- else if eq .type "hard" }} + {{- include "common.affinities.pods.hard" . -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_capabilities.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_capabilities.tpl new file mode 100644 index 0000000000..4dde56a38d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_capabilities.tpl @@ -0,0 +1,95 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Return the target Kubernetes version +*/}} +{{- define "common.capabilities.kubeVersion" -}} +{{- if .Values.global }} + {{- if .Values.global.kubeVersion }} + {{- .Values.global.kubeVersion -}} + {{- else }} + {{- default .Capabilities.KubeVersion.Version .Values.kubeVersion -}} + {{- end -}} +{{- else }} +{{- default .Capabilities.KubeVersion.Version .Values.kubeVersion -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for deployment. +*/}} +{{- define "common.capabilities.deployment.apiVersion" -}} +{{- if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "extensions/v1beta1" -}} +{{- else -}} +{{- print "apps/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for statefulset. +*/}} +{{- define "common.capabilities.statefulset.apiVersion" -}} +{{- if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "apps/v1beta1" -}} +{{- else -}} +{{- print "apps/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "common.capabilities.ingress.apiVersion" -}} +{{- if .Values.ingress -}} +{{- if .Values.ingress.apiVersion -}} +{{- .Values.ingress.apiVersion -}} +{{- else if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "extensions/v1beta1" -}} +{{- else if semverCompare "<1.19-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "networking.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "networking.k8s.io/v1" -}} +{{- end }} +{{- else if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "extensions/v1beta1" -}} +{{- else if semverCompare "<1.19-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "networking.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "networking.k8s.io/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for RBAC resources. +*/}} +{{- define "common.capabilities.rbac.apiVersion" -}} +{{- if semverCompare "<1.17-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "rbac.authorization.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "rbac.authorization.k8s.io/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for CRDs. +*/}} +{{- define "common.capabilities.crd.apiVersion" -}} +{{- if semverCompare "<1.19-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "apiextensions.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "apiextensions.k8s.io/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Returns true if the used Helm version is 3.3+. +A way to check the used Helm version was not introduced until version 3.3.0 with .Capabilities.HelmVersion, which contains an additional "{}}" structure. +This check is introduced as a regexMatch instead of {{ if .Capabilities.HelmVersion }} because checking for the key HelmVersion in <3.3 results in a "interface not found" error. +**To be removed when the catalog's minimun Helm version is 3.3** +*/}} +{{- define "common.capabilities.supportsHelmVersion" -}} +{{- if regexMatch "{(v[0-9])*[^}]*}}$" (.Capabilities | toString ) }} + {{- true -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_errors.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_errors.tpl new file mode 100644 index 0000000000..a79cc2e322 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_errors.tpl @@ -0,0 +1,23 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Through error when upgrading using empty passwords values that must not be empty. + +Usage: +{{- $validationError00 := include "common.validations.values.single.empty" (dict "valueKey" "path.to.password00" "secret" "secretName" "field" "password-00") -}} +{{- $validationError01 := include "common.validations.values.single.empty" (dict "valueKey" "path.to.password01" "secret" "secretName" "field" "password-01") -}} +{{ include "common.errors.upgrade.passwords.empty" (dict "validationErrors" (list $validationError00 $validationError01) "context" $) }} + +Required password params: + - validationErrors - String - Required. List of validation strings to be return, if it is empty it won't throw error. + - context - Context - Required. Parent context. +*/}} +{{- define "common.errors.upgrade.passwords.empty" -}} + {{- $validationErrors := join "" .validationErrors -}} + {{- if and $validationErrors .context.Release.IsUpgrade -}} + {{- $errorString := "\nPASSWORDS ERROR: You must provide your current passwords when upgrading the release." -}} + {{- $errorString = print $errorString "\n Note that even after reinstallation, old credentials may be needed as they may be kept in persistent volume claims." -}} + {{- $errorString = print $errorString "\n Further information can be obtained at https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues/#credential-errors-while-upgrading-chart-releases" -}} + {{- $errorString = print $errorString "\n%s" -}} + {{- printf $errorString $validationErrors | fail -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_images.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_images.tpl new file mode 100644 index 0000000000..60f04fd6e2 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_images.tpl @@ -0,0 +1,47 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Return the proper image name +{{ include "common.images.image" ( dict "imageRoot" .Values.path.to.the.image "global" $) }} +*/}} +{{- define "common.images.image" -}} +{{- $registryName := .imageRoot.registry -}} +{{- $repositoryName := .imageRoot.repository -}} +{{- $tag := .imageRoot.tag | toString -}} +{{- if .global }} + {{- if .global.imageRegistry }} + {{- $registryName = .global.imageRegistry -}} + {{- end -}} +{{- end -}} +{{- if $registryName }} +{{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} +{{- else -}} +{{- printf "%s:%s" $repositoryName $tag -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper Docker Image Registry Secret Names +{{ include "common.images.pullSecrets" ( dict "images" (list .Values.path.to.the.image1, .Values.path.to.the.image2) "global" .Values.global) }} +*/}} +{{- define "common.images.pullSecrets" -}} + {{- $pullSecrets := list }} + + {{- if .global }} + {{- range .global.imagePullSecrets -}} + {{- $pullSecrets = append $pullSecrets . -}} + {{- end -}} + {{- end -}} + + {{- range .images -}} + {{- range .pullSecrets -}} + {{- $pullSecrets = append $pullSecrets . -}} + {{- end -}} + {{- end -}} + + {{- if (not (empty $pullSecrets)) }} +imagePullSecrets: + {{- range $pullSecrets }} + - name: {{ . }} + {{- end }} + {{- end }} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_ingress.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_ingress.tpl new file mode 100644 index 0000000000..622ef50e3c --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_ingress.tpl @@ -0,0 +1,42 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Generate backend entry that is compatible with all Kubernetes API versions. + +Usage: +{{ include "common.ingress.backend" (dict "serviceName" "backendName" "servicePort" "backendPort" "context" $) }} + +Params: + - serviceName - String. Name of an existing service backend + - servicePort - String/Int. Port name (or number) of the service. It will be translated to different yaml depending if it is a string or an integer. + - context - Dict - Required. The context for the template evaluation. +*/}} +{{- define "common.ingress.backend" -}} +{{- $apiVersion := (include "common.capabilities.ingress.apiVersion" .context) -}} +{{- if or (eq $apiVersion "extensions/v1beta1") (eq $apiVersion "networking.k8s.io/v1beta1") -}} +serviceName: {{ .serviceName }} +servicePort: {{ .servicePort }} +{{- else -}} +service: + name: {{ .serviceName }} + port: + {{- if typeIs "string" .servicePort }} + name: {{ .servicePort }} + {{- else if typeIs "int" .servicePort }} + number: {{ .servicePort }} + {{- end }} +{{- end -}} +{{- end -}} + +{{/* +Print "true" if the API pathType field is supported +Usage: +{{ include "common.ingress.supportsPathType" . }} +*/}} +{{- define "common.ingress.supportsPathType" -}} +{{- if (semverCompare "<1.18-0" (include "common.capabilities.kubeVersion" .)) -}} +{{- print "false" -}} +{{- else -}} +{{- print "true" -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_labels.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_labels.tpl new file mode 100644 index 0000000000..252066c7e2 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_labels.tpl @@ -0,0 +1,18 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Kubernetes standard labels +*/}} +{{- define "common.labels.standard" -}} +app.kubernetes.io/name: {{ include "common.names.name" . }} +helm.sh/chart: {{ include "common.names.chart" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} + +{{/* +Labels to use on deploy.spec.selector.matchLabels and svc.spec.selector +*/}} +{{- define "common.labels.matchLabels" -}} +app.kubernetes.io/name: {{ include "common.names.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_names.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_names.tpl new file mode 100644 index 0000000000..adf2a74f48 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_names.tpl @@ -0,0 +1,32 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "common.names.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "common.names.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | 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 "common.names.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 -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_secrets.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_secrets.tpl new file mode 100644 index 0000000000..60b84a7019 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_secrets.tpl @@ -0,0 +1,129 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Generate secret name. + +Usage: +{{ include "common.secrets.name" (dict "existingSecret" .Values.path.to.the.existingSecret "defaultNameSuffix" "mySuffix" "context" $) }} + +Params: + - existingSecret - ExistingSecret/String - Optional. The path to the existing secrets in the values.yaml given by the user + to be used instead of the default one. Allows for it to be of type String (just the secret name) for backwards compatibility. + +info: https://github.com/bitnami/charts/tree/master/bitnami/common#existingsecret + - defaultNameSuffix - String - Optional. It is used only if we have several secrets in the same deployment. + - context - Dict - Required. The context for the template evaluation. +*/}} +{{- define "common.secrets.name" -}} +{{- $name := (include "common.names.fullname" .context) -}} + +{{- if .defaultNameSuffix -}} +{{- $name = printf "%s-%s" $name .defaultNameSuffix | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{- with .existingSecret -}} +{{- if not (typeIs "string" .) -}} +{{- with .name -}} +{{- $name = . -}} +{{- end -}} +{{- else -}} +{{- $name = . -}} +{{- end -}} +{{- end -}} + +{{- printf "%s" $name -}} +{{- end -}} + +{{/* +Generate secret key. + +Usage: +{{ include "common.secrets.key" (dict "existingSecret" .Values.path.to.the.existingSecret "key" "keyName") }} + +Params: + - existingSecret - ExistingSecret/String - Optional. The path to the existing secrets in the values.yaml given by the user + to be used instead of the default one. Allows for it to be of type String (just the secret name) for backwards compatibility. + +info: https://github.com/bitnami/charts/tree/master/bitnami/common#existingsecret + - key - String - Required. Name of the key in the secret. +*/}} +{{- define "common.secrets.key" -}} +{{- $key := .key -}} + +{{- if .existingSecret -}} + {{- if not (typeIs "string" .existingSecret) -}} + {{- if .existingSecret.keyMapping -}} + {{- $key = index .existingSecret.keyMapping $.key -}} + {{- end -}} + {{- end }} +{{- end -}} + +{{- printf "%s" $key -}} +{{- end -}} + +{{/* +Generate secret password or retrieve one if already created. + +Usage: +{{ include "common.secrets.passwords.manage" (dict "secret" "secret-name" "key" "keyName" "providedValues" (list "path.to.password1" "path.to.password2") "length" 10 "strong" false "chartName" "chartName" "context" $) }} + +Params: + - secret - String - Required - Name of the 'Secret' resource where the password is stored. + - key - String - Required - Name of the key in the secret. + - providedValues - List - Required - The path to the validating value in the values.yaml, e.g: "mysql.password". Will pick first parameter with a defined value. + - length - int - Optional - Length of the generated random password. + - strong - Boolean - Optional - Whether to add symbols to the generated random password. + - chartName - String - Optional - Name of the chart used when said chart is deployed as a subchart. + - context - Context - Required - Parent context. +*/}} +{{- define "common.secrets.passwords.manage" -}} + +{{- $password := "" }} +{{- $subchart := "" }} +{{- $chartName := default "" .chartName }} +{{- $passwordLength := default 10 .length }} +{{- $providedPasswordKey := include "common.utils.getKeyFromList" (dict "keys" .providedValues "context" $.context) }} +{{- $providedPasswordValue := include "common.utils.getValueFromKey" (dict "key" $providedPasswordKey "context" $.context) }} +{{- $secret := (lookup "v1" "Secret" $.context.Release.Namespace .secret) }} +{{- if $secret }} + {{- if index $secret.data .key }} + {{- $password = index $secret.data .key }} + {{- end -}} +{{- else if $providedPasswordValue }} + {{- $password = $providedPasswordValue | toString | b64enc | quote }} +{{- else }} + + {{- if .context.Values.enabled }} + {{- $subchart = $chartName }} + {{- end -}} + + {{- $requiredPassword := dict "valueKey" $providedPasswordKey "secret" .secret "field" .key "subchart" $subchart "context" $.context -}} + {{- $requiredPasswordError := include "common.validations.values.single.empty" $requiredPassword -}} + {{- $passwordValidationErrors := list $requiredPasswordError -}} + {{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" $passwordValidationErrors "context" $.context) -}} + + {{- if .strong }} + {{- $subStr := list (lower (randAlpha 1)) (randNumeric 1) (upper (randAlpha 1)) | join "_" }} + {{- $password = randAscii $passwordLength }} + {{- $password = regexReplaceAllLiteral "\\W" $password "@" | substr 5 $passwordLength }} + {{- $password = printf "%s%s" $subStr $password | toString | shuffle | b64enc | quote }} + {{- else }} + {{- $password = randAlphaNum $passwordLength | b64enc | quote }} + {{- end }} +{{- end -}} +{{- printf "%s" $password -}} +{{- end -}} + +{{/* +Returns whether a previous generated secret already exists + +Usage: +{{ include "common.secrets.exists" (dict "secret" "secret-name" "context" $) }} + +Params: + - secret - String - Required - Name of the 'Secret' resource where the password is stored. + - context - Context - Required - Parent context. +*/}} +{{- define "common.secrets.exists" -}} +{{- $secret := (lookup "v1" "Secret" $.context.Release.Namespace .secret) }} +{{- if $secret }} + {{- true -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_storage.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_storage.tpl new file mode 100644 index 0000000000..60e2a844f6 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_storage.tpl @@ -0,0 +1,23 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Return the proper Storage Class +{{ include "common.storage.class" ( dict "persistence" .Values.path.to.the.persistence "global" $) }} +*/}} +{{- define "common.storage.class" -}} + +{{- $storageClass := .persistence.storageClass -}} +{{- if .global -}} + {{- if .global.storageClass -}} + {{- $storageClass = .global.storageClass -}} + {{- end -}} +{{- end -}} + +{{- if $storageClass -}} + {{- if (eq "-" $storageClass) -}} + {{- printf "storageClassName: \"\"" -}} + {{- else }} + {{- printf "storageClassName: %s" $storageClass -}} + {{- end -}} +{{- end -}} + +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_tplvalues.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_tplvalues.tpl new file mode 100644 index 0000000000..2db166851b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_tplvalues.tpl @@ -0,0 +1,13 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Renders a value that contains template. +Usage: +{{ include "common.tplvalues.render" ( dict "value" .Values.path.to.the.Value "context" $) }} +*/}} +{{- define "common.tplvalues.render" -}} + {{- if typeIs "string" .value }} + {{- tpl .value .context }} + {{- else }} + {{- tpl (.value | toYaml) .context }} + {{- end }} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_utils.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_utils.tpl new file mode 100644 index 0000000000..ea083a249f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_utils.tpl @@ -0,0 +1,62 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Print instructions to get a secret value. +Usage: +{{ include "common.utils.secret.getvalue" (dict "secret" "secret-name" "field" "secret-value-field" "context" $) }} +*/}} +{{- define "common.utils.secret.getvalue" -}} +{{- $varname := include "common.utils.fieldToEnvVar" . -}} +export {{ $varname }}=$(kubectl get secret --namespace {{ .context.Release.Namespace | quote }} {{ .secret }} -o jsonpath="{.data.{{ .field }}}" | base64 --decode) +{{- end -}} + +{{/* +Build env var name given a field +Usage: +{{ include "common.utils.fieldToEnvVar" dict "field" "my-password" }} +*/}} +{{- define "common.utils.fieldToEnvVar" -}} + {{- $fieldNameSplit := splitList "-" .field -}} + {{- $upperCaseFieldNameSplit := list -}} + + {{- range $fieldNameSplit -}} + {{- $upperCaseFieldNameSplit = append $upperCaseFieldNameSplit ( upper . ) -}} + {{- end -}} + + {{ join "_" $upperCaseFieldNameSplit }} +{{- end -}} + +{{/* +Gets a value from .Values given +Usage: +{{ include "common.utils.getValueFromKey" (dict "key" "path.to.key" "context" $) }} +*/}} +{{- define "common.utils.getValueFromKey" -}} +{{- $splitKey := splitList "." .key -}} +{{- $value := "" -}} +{{- $latestObj := $.context.Values -}} +{{- range $splitKey -}} + {{- if not $latestObj -}} + {{- printf "please review the entire path of '%s' exists in values" $.key | fail -}} + {{- end -}} + {{- $value = ( index $latestObj . ) -}} + {{- $latestObj = $value -}} +{{- end -}} +{{- printf "%v" (default "" $value) -}} +{{- end -}} + +{{/* +Returns first .Values key with a defined value or first of the list if all non-defined +Usage: +{{ include "common.utils.getKeyFromList" (dict "keys" (list "path.to.key1" "path.to.key2") "context" $) }} +*/}} +{{- define "common.utils.getKeyFromList" -}} +{{- $key := first .keys -}} +{{- $reverseKeys := reverse .keys }} +{{- range $reverseKeys }} + {{- $value := include "common.utils.getValueFromKey" (dict "key" . "context" $.context ) }} + {{- if $value -}} + {{- $key = . }} + {{- end -}} +{{- end -}} +{{- printf "%s" $key -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_warnings.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_warnings.tpl new file mode 100644 index 0000000000..ae10fa41ee --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/_warnings.tpl @@ -0,0 +1,14 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Warning about using rolling tag. +Usage: +{{ include "common.warnings.rollingTag" .Values.path.to.the.imageRoot }} +*/}} +{{- define "common.warnings.rollingTag" -}} + +{{- if and (contains "bitnami/" .repository) (not (.tag | toString | regexFind "-r\\d+$|sha256:")) }} +WARNING: Rolling tag detected ({{ .repository }}:{{ .tag }}), please note that it is strongly recommended to avoid using rolling tags in a production environment. ++info https://docs.bitnami.com/containers/how-to/understand-rolling-tags-containers/ +{{- end }} + +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_cassandra.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_cassandra.tpl new file mode 100644 index 0000000000..8679ddffb1 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_cassandra.tpl @@ -0,0 +1,72 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate Cassandra required passwords are not empty. + +Usage: +{{ include "common.validations.values.cassandra.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where Cassandra values are stored, e.g: "cassandra-passwords-secret" + - subchart - Boolean - Optional. Whether Cassandra is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.cassandra.passwords" -}} + {{- $existingSecret := include "common.cassandra.values.existingSecret" . -}} + {{- $enabled := include "common.cassandra.values.enabled" . -}} + {{- $dbUserPrefix := include "common.cassandra.values.key.dbUser" . -}} + {{- $valueKeyPassword := printf "%s.password" $dbUserPrefix -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredPassword := dict "valueKey" $valueKeyPassword "secret" .secret "field" "cassandra-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPassword -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.cassandra.values.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether Cassandra is used as subchart or not. Default: false +*/}} +{{- define "common.cassandra.values.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.cassandra.dbUser.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.dbUser.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled cassandra. + +Usage: +{{ include "common.cassandra.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.cassandra.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.cassandra.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key dbUser + +Usage: +{{ include "common.cassandra.values.key.dbUser" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether Cassandra is used as subchart or not. Default: false +*/}} +{{- define "common.cassandra.values.key.dbUser" -}} + {{- if .subchart -}} + cassandra.dbUser + {{- else -}} + dbUser + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mariadb.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mariadb.tpl new file mode 100644 index 0000000000..bb5ed7253d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mariadb.tpl @@ -0,0 +1,103 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate MariaDB required passwords are not empty. + +Usage: +{{ include "common.validations.values.mariadb.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where MariaDB values are stored, e.g: "mysql-passwords-secret" + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.mariadb.passwords" -}} + {{- $existingSecret := include "common.mariadb.values.auth.existingSecret" . -}} + {{- $enabled := include "common.mariadb.values.enabled" . -}} + {{- $architecture := include "common.mariadb.values.architecture" . -}} + {{- $authPrefix := include "common.mariadb.values.key.auth" . -}} + {{- $valueKeyRootPassword := printf "%s.rootPassword" $authPrefix -}} + {{- $valueKeyUsername := printf "%s.username" $authPrefix -}} + {{- $valueKeyPassword := printf "%s.password" $authPrefix -}} + {{- $valueKeyReplicationPassword := printf "%s.replicationPassword" $authPrefix -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredRootPassword := dict "valueKey" $valueKeyRootPassword "secret" .secret "field" "mariadb-root-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredRootPassword -}} + + {{- $valueUsername := include "common.utils.getValueFromKey" (dict "key" $valueKeyUsername "context" .context) }} + {{- if not (empty $valueUsername) -}} + {{- $requiredPassword := dict "valueKey" $valueKeyPassword "secret" .secret "field" "mariadb-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPassword -}} + {{- end -}} + + {{- if (eq $architecture "replication") -}} + {{- $requiredReplicationPassword := dict "valueKey" $valueKeyReplicationPassword "secret" .secret "field" "mariadb-replication-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredReplicationPassword -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.mariadb.values.auth.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mariadb.values.auth.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.mariadb.auth.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.auth.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled mariadb. + +Usage: +{{ include "common.mariadb.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.mariadb.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.mariadb.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for architecture + +Usage: +{{ include "common.mariadb.values.architecture" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mariadb.values.architecture" -}} + {{- if .subchart -}} + {{- .context.Values.mariadb.architecture -}} + {{- else -}} + {{- .context.Values.architecture -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key auth + +Usage: +{{ include "common.mariadb.values.key.auth" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mariadb.values.key.auth" -}} + {{- if .subchart -}} + mariadb.auth + {{- else -}} + auth + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mongodb.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mongodb.tpl new file mode 100644 index 0000000000..7d5ecbccb4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_mongodb.tpl @@ -0,0 +1,108 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate MongoDB(R) required passwords are not empty. + +Usage: +{{ include "common.validations.values.mongodb.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where MongoDB(R) values are stored, e.g: "mongodb-passwords-secret" + - subchart - Boolean - Optional. Whether MongoDB(R) is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.mongodb.passwords" -}} + {{- $existingSecret := include "common.mongodb.values.auth.existingSecret" . -}} + {{- $enabled := include "common.mongodb.values.enabled" . -}} + {{- $authPrefix := include "common.mongodb.values.key.auth" . -}} + {{- $architecture := include "common.mongodb.values.architecture" . -}} + {{- $valueKeyRootPassword := printf "%s.rootPassword" $authPrefix -}} + {{- $valueKeyUsername := printf "%s.username" $authPrefix -}} + {{- $valueKeyDatabase := printf "%s.database" $authPrefix -}} + {{- $valueKeyPassword := printf "%s.password" $authPrefix -}} + {{- $valueKeyReplicaSetKey := printf "%s.replicaSetKey" $authPrefix -}} + {{- $valueKeyAuthEnabled := printf "%s.enabled" $authPrefix -}} + + {{- $authEnabled := include "common.utils.getValueFromKey" (dict "key" $valueKeyAuthEnabled "context" .context) -}} + + {{- if and (not $existingSecret) (eq $enabled "true") (eq $authEnabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredRootPassword := dict "valueKey" $valueKeyRootPassword "secret" .secret "field" "mongodb-root-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredRootPassword -}} + + {{- $valueUsername := include "common.utils.getValueFromKey" (dict "key" $valueKeyUsername "context" .context) }} + {{- $valueDatabase := include "common.utils.getValueFromKey" (dict "key" $valueKeyDatabase "context" .context) }} + {{- if and $valueUsername $valueDatabase -}} + {{- $requiredPassword := dict "valueKey" $valueKeyPassword "secret" .secret "field" "mongodb-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPassword -}} + {{- end -}} + + {{- if (eq $architecture "replicaset") -}} + {{- $requiredReplicaSetKey := dict "valueKey" $valueKeyReplicaSetKey "secret" .secret "field" "mongodb-replica-set-key" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredReplicaSetKey -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.mongodb.values.auth.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MongoDb is used as subchart or not. Default: false +*/}} +{{- define "common.mongodb.values.auth.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.mongodb.auth.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.auth.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled mongodb. + +Usage: +{{ include "common.mongodb.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.mongodb.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.mongodb.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key auth + +Usage: +{{ include "common.mongodb.values.key.auth" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MongoDB(R) is used as subchart or not. Default: false +*/}} +{{- define "common.mongodb.values.key.auth" -}} + {{- if .subchart -}} + mongodb.auth + {{- else -}} + auth + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for architecture + +Usage: +{{ include "common.mongodb.values.architecture" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mongodb.values.architecture" -}} + {{- if .subchart -}} + {{- .context.Values.mongodb.architecture -}} + {{- else -}} + {{- .context.Values.architecture -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_postgresql.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_postgresql.tpl new file mode 100644 index 0000000000..992bcd3908 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_postgresql.tpl @@ -0,0 +1,131 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate PostgreSQL required passwords are not empty. + +Usage: +{{ include "common.validations.values.postgresql.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where postgresql values are stored, e.g: "postgresql-passwords-secret" + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.postgresql.passwords" -}} + {{- $existingSecret := include "common.postgresql.values.existingSecret" . -}} + {{- $enabled := include "common.postgresql.values.enabled" . -}} + {{- $valueKeyPostgresqlPassword := include "common.postgresql.values.key.postgressPassword" . -}} + {{- $valueKeyPostgresqlReplicationEnabled := include "common.postgresql.values.key.replicationPassword" . -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredPostgresqlPassword := dict "valueKey" $valueKeyPostgresqlPassword "secret" .secret "field" "postgresql-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPostgresqlPassword -}} + + {{- $enabledReplication := include "common.postgresql.values.enabled.replication" . -}} + {{- if (eq $enabledReplication "true") -}} + {{- $requiredPostgresqlReplicationPassword := dict "valueKey" $valueKeyPostgresqlReplicationEnabled "secret" .secret "field" "postgresql-replication-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPostgresqlReplicationPassword -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to decide whether evaluate global values. + +Usage: +{{ include "common.postgresql.values.use.global" (dict "key" "key-of-global" "context" $) }} +Params: + - key - String - Required. Field to be evaluated within global, e.g: "existingSecret" +*/}} +{{- define "common.postgresql.values.use.global" -}} + {{- if .context.Values.global -}} + {{- if .context.Values.global.postgresql -}} + {{- index .context.Values.global.postgresql .key | quote -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.postgresql.values.existingSecret" (dict "context" $) }} +*/}} +{{- define "common.postgresql.values.existingSecret" -}} + {{- $globalValue := include "common.postgresql.values.use.global" (dict "key" "existingSecret" "context" .context) -}} + + {{- if .subchart -}} + {{- default (.context.Values.postgresql.existingSecret | quote) $globalValue -}} + {{- else -}} + {{- default (.context.Values.existingSecret | quote) $globalValue -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled postgresql. + +Usage: +{{ include "common.postgresql.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.postgresql.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.postgresql.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key postgressPassword. + +Usage: +{{ include "common.postgresql.values.key.postgressPassword" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.postgresql.values.key.postgressPassword" -}} + {{- $globalValue := include "common.postgresql.values.use.global" (dict "key" "postgresqlUsername" "context" .context) -}} + + {{- if not $globalValue -}} + {{- if .subchart -}} + postgresql.postgresqlPassword + {{- else -}} + postgresqlPassword + {{- end -}} + {{- else -}} + global.postgresql.postgresqlPassword + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled.replication. + +Usage: +{{ include "common.postgresql.values.enabled.replication" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.postgresql.values.enabled.replication" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.postgresql.replication.enabled -}} + {{- else -}} + {{- printf "%v" .context.Values.replication.enabled -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key replication.password. + +Usage: +{{ include "common.postgresql.values.key.replicationPassword" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.postgresql.values.key.replicationPassword" -}} + {{- if .subchart -}} + postgresql.replication.password + {{- else -}} + replication.password + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_redis.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_redis.tpl new file mode 100644 index 0000000000..3e2a47c039 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_redis.tpl @@ -0,0 +1,72 @@ + +{{/* vim: set filetype=mustache: */}} +{{/* +Validate Redis(TM) required passwords are not empty. + +Usage: +{{ include "common.validations.values.redis.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where redis values are stored, e.g: "redis-passwords-secret" + - subchart - Boolean - Optional. Whether redis is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.redis.passwords" -}} + {{- $existingSecret := include "common.redis.values.existingSecret" . -}} + {{- $enabled := include "common.redis.values.enabled" . -}} + {{- $valueKeyPrefix := include "common.redis.values.keys.prefix" . -}} + {{- $valueKeyRedisPassword := printf "%s%s" $valueKeyPrefix "password" -}} + {{- $valueKeyRedisUsePassword := printf "%s%s" $valueKeyPrefix "usePassword" -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $usePassword := include "common.utils.getValueFromKey" (dict "key" $valueKeyRedisUsePassword "context" .context) -}} + {{- if eq $usePassword "true" -}} + {{- $requiredRedisPassword := dict "valueKey" $valueKeyRedisPassword "secret" .secret "field" "redis-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredRedisPassword -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + {{- end -}} +{{- end -}} + +{{/* +Redis Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.redis.values.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether Redis(TM) is used as subchart or not. Default: false +*/}} +{{- define "common.redis.values.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.redis.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled redis. + +Usage: +{{ include "common.redis.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.redis.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.redis.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right prefix path for the values + +Usage: +{{ include "common.redis.values.key.prefix" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether redis is used as subchart or not. Default: false +*/}} +{{- define "common.redis.values.keys.prefix" -}} + {{- if .subchart -}}redis.{{- else -}}{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_validations.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_validations.tpl new file mode 100644 index 0000000000..9a814cf40d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/templates/validations/_validations.tpl @@ -0,0 +1,46 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate values must not be empty. + +Usage: +{{- $validateValueConf00 := (dict "valueKey" "path.to.value" "secret" "secretName" "field" "password-00") -}} +{{- $validateValueConf01 := (dict "valueKey" "path.to.value" "secret" "secretName" "field" "password-01") -}} +{{ include "common.validations.values.empty" (dict "required" (list $validateValueConf00 $validateValueConf01) "context" $) }} + +Validate value params: + - valueKey - String - Required. The path to the validating value in the values.yaml, e.g: "mysql.password" + - secret - String - Optional. Name of the secret where the validating value is generated/stored, e.g: "mysql-passwords-secret" + - field - String - Optional. Name of the field in the secret data, e.g: "mysql-password" +*/}} +{{- define "common.validations.values.multiple.empty" -}} + {{- range .required -}} + {{- include "common.validations.values.single.empty" (dict "valueKey" .valueKey "secret" .secret "field" .field "context" $.context) -}} + {{- end -}} +{{- end -}} + +{{/* +Validate a value must not be empty. + +Usage: +{{ include "common.validations.value.empty" (dict "valueKey" "mariadb.password" "secret" "secretName" "field" "my-password" "subchart" "subchart" "context" $) }} + +Validate value params: + - valueKey - String - Required. The path to the validating value in the values.yaml, e.g: "mysql.password" + - secret - String - Optional. Name of the secret where the validating value is generated/stored, e.g: "mysql-passwords-secret" + - field - String - Optional. Name of the field in the secret data, e.g: "mysql-password" + - subchart - String - Optional - Name of the subchart that the validated password is part of. +*/}} +{{- define "common.validations.values.single.empty" -}} + {{- $value := include "common.utils.getValueFromKey" (dict "key" .valueKey "context" .context) }} + {{- $subchart := ternary "" (printf "%s." .subchart) (empty .subchart) }} + + {{- if not $value -}} + {{- $varname := "my-value" -}} + {{- $getCurrentValue := "" -}} + {{- if and .secret .field -}} + {{- $varname = include "common.utils.fieldToEnvVar" . -}} + {{- $getCurrentValue = printf " To get the current value:\n\n %s\n" (include "common.utils.secret.getvalue" .) -}} + {{- end -}} + {{- printf "\n '%s' must not be empty, please add '--set %s%s=$%s' to the command.%s" .valueKey $subchart .valueKey $varname $getCurrentValue -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/values.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/values.yaml new file mode 100644 index 0000000000..9ecdc93f58 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/charts/common/values.yaml @@ -0,0 +1,3 @@ +## bitnami/common +## It is required by CI/CD tools and processes. +exampleValue: common-chart diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/commonAnnotations.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/commonAnnotations.yaml new file mode 100644 index 0000000000..97e18a4cc0 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/commonAnnotations.yaml @@ -0,0 +1,3 @@ +commonAnnotations: + helm.sh/hook: "\"pre-install, pre-upgrade\"" + helm.sh/hook-weight: "-1" diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/default-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/default-values.yaml new file mode 100644 index 0000000000..fc2ba605ad --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/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/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/shmvolume-disabled-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/shmvolume-disabled-values.yaml new file mode 100644 index 0000000000..347d3b40a8 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/ci/shmvolume-disabled-values.yaml @@ -0,0 +1,2 @@ +shmVolume: + enabled: false diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/README.md b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/README.md new file mode 100644 index 0000000000..1813a2feaa --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/README.md @@ -0,0 +1 @@ +Copy here your postgresql.conf and/or pg_hba.conf files to use it as a config map. diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/conf.d/README.md b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/conf.d/README.md new file mode 100644 index 0000000000..184c1875d5 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/conf.d/README.md @@ -0,0 +1,4 @@ +If you don't want to provide the whole configuration file and only specify certain parameters, you can copy here your extended `.conf` files. +These files will be injected as a config maps and add/overwrite the default configuration using the `include_dir` directive that allows settings to be loaded from files other than the default `postgresql.conf`. + +More info in the [bitnami-docker-postgresql README](https://github.com/bitnami/bitnami-docker-postgresql#configuration-file). diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/docker-entrypoint-initdb.d/README.md b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/docker-entrypoint-initdb.d/README.md new file mode 100644 index 0000000000..cba38091e0 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/files/docker-entrypoint-initdb.d/README.md @@ -0,0 +1,3 @@ +You can copy here your custom `.sh`, `.sql` or `.sql.gz` file so they are executed during the first boot of the image. + +More info in the [bitnami-docker-postgresql](https://github.com/bitnami/bitnami-docker-postgresql#initializing-a-new-instance) repository. \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/NOTES.txt b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/NOTES.txt new file mode 100644 index 0000000000..4e98958c13 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/NOTES.txt @@ -0,0 +1,59 @@ +** Please be patient while the chart is being deployed ** + +PostgreSQL can be accessed via port {{ template "postgresql.port" . }} on the following DNS name from within your cluster: + + {{ template "common.names.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local - Read/Write connection +{{- if .Values.replication.enabled }} + {{ template "common.names.fullname" . }}-read.{{ .Release.Namespace }}.svc.cluster.local - Read only connection +{{- end }} + +{{- if not (eq (include "postgresql.username" .) "postgres") }} + +To get the password for "postgres" run: + + export POSTGRES_ADMIN_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "postgresql.secretName" . }} -o jsonpath="{.data.postgresql-postgres-password}" | base64 --decode) +{{- end }} + +To get the password for "{{ template "postgresql.username" . }}" run: + + export POSTGRES_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "postgresql.secretName" . }} -o jsonpath="{.data.postgresql-password}" | base64 --decode) + +To connect to your database run the following command: + + kubectl run {{ template "common.names.fullname" . }}-client --rm --tty -i --restart='Never' --namespace {{ .Release.Namespace }} --image {{ template "postgresql.image" . }} --env="PGPASSWORD=$POSTGRES_PASSWORD" {{- if and (.Values.networkPolicy.enabled) (not .Values.networkPolicy.allowExternal) }} + --labels="{{ template "common.names.fullname" . }}-client=true" {{- end }} --command -- psql --host {{ template "common.names.fullname" . }} -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} -p {{ template "postgresql.port" . }} + +{{ if and (.Values.networkPolicy.enabled) (not .Values.networkPolicy.allowExternal) }} +Note: Since NetworkPolicy is enabled, only pods with label {{ template "common.names.fullname" . }}-client=true" will be able to connect to this PostgreSQL cluster. +{{- end }} + +To connect to your database from outside the cluster execute the following commands: + +{{- if contains "NodePort" .Values.service.type }} + + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "common.names.fullname" . }}) + {{ if (include "postgresql.password" . ) }}PGPASSWORD="$POSTGRES_PASSWORD" {{ end }}psql --host $NODE_IP --port $NODE_PORT -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} + +{{- else if contains "LoadBalancer" .Values.service.type }} + + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + Watch the status with: 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "common.names.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "common.names.fullname" . }} --template "{{"{{ range (index .status.loadBalancer.ingress 0) }}{{.}}{{ end }}"}}") + {{ if (include "postgresql.password" . ) }}PGPASSWORD="$POSTGRES_PASSWORD" {{ end }}psql --host $SERVICE_IP --port {{ template "postgresql.port" . }} -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} + +{{- else if contains "ClusterIP" .Values.service.type }} + + kubectl port-forward --namespace {{ .Release.Namespace }} svc/{{ template "common.names.fullname" . }} {{ template "postgresql.port" . }}:{{ template "postgresql.port" . }} & + {{ if (include "postgresql.password" . ) }}PGPASSWORD="$POSTGRES_PASSWORD" {{ end }}psql --host 127.0.0.1 -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} -p {{ template "postgresql.port" . }} + +{{- end }} + +{{- include "postgresql.validateValues" . -}} + +{{- include "common.warnings.rollingTag" .Values.image -}} + +{{- $passwordValidationErrors := include "common.validations.values.postgresql.passwords" (dict "secret" (include "common.names.fullname" .) "context" $) -}} + +{{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" (list $passwordValidationErrors) "context" $) -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/_helpers.tpl b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/_helpers.tpl new file mode 100644 index 0000000000..1f98efe789 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/_helpers.tpl @@ -0,0 +1,337 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Expand the name of the chart. +*/}} +{{- define "postgresql.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 "postgresql.primary.fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- $fullname := default (printf "%s-%s" .Release.Name $name) .Values.fullnameOverride -}} +{{- if .Values.replication.enabled -}} +{{- printf "%s-%s" $fullname "primary" | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s" $fullname | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper PostgreSQL image name +*/}} +{{- define "postgresql.image" -}} +{{ include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) }} +{{- end -}} + +{{/* +Return the proper PostgreSQL metrics image name +*/}} +{{- define "postgresql.metrics.image" -}} +{{ include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) }} +{{- end -}} + +{{/* +Return the proper image name (for the init container volume-permissions image) +*/}} +{{- define "postgresql.volumePermissions.image" -}} +{{ include "common.images.image" (dict "imageRoot" .Values.volumePermissions.image "global" .Values.global) }} +{{- end -}} + +{{/* +Return the proper Docker Image Registry Secret Names +*/}} +{{- define "postgresql.imagePullSecrets" -}} +{{ include "common.images.pullSecrets" (dict "images" (list .Values.image .Values.metrics.image .Values.volumePermissions.image) "global" .Values.global) }} +{{- end -}} + +{{/* +Return PostgreSQL postgres user password +*/}} +{{- define "postgresql.postgres.password" -}} +{{- if .Values.global.postgresql.postgresqlPostgresPassword }} + {{- .Values.global.postgresql.postgresqlPostgresPassword -}} +{{- else if .Values.postgresqlPostgresPassword -}} + {{- .Values.postgresqlPostgresPassword -}} +{{- else -}} + {{- randAlphaNum 10 -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL password +*/}} +{{- define "postgresql.password" -}} +{{- if .Values.global.postgresql.postgresqlPassword }} + {{- .Values.global.postgresql.postgresqlPassword -}} +{{- else if .Values.postgresqlPassword -}} + {{- .Values.postgresqlPassword -}} +{{- else -}} + {{- randAlphaNum 10 -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL replication password +*/}} +{{- define "postgresql.replication.password" -}} +{{- if .Values.global.postgresql.replicationPassword }} + {{- .Values.global.postgresql.replicationPassword -}} +{{- else if .Values.replication.password -}} + {{- .Values.replication.password -}} +{{- else -}} + {{- randAlphaNum 10 -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL username +*/}} +{{- define "postgresql.username" -}} +{{- if .Values.global.postgresql.postgresqlUsername }} + {{- .Values.global.postgresql.postgresqlUsername -}} +{{- else -}} + {{- .Values.postgresqlUsername -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL replication username +*/}} +{{- define "postgresql.replication.username" -}} +{{- if .Values.global.postgresql.replicationUser }} + {{- .Values.global.postgresql.replicationUser -}} +{{- else -}} + {{- .Values.replication.user -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL port +*/}} +{{- define "postgresql.port" -}} +{{- if .Values.global.postgresql.servicePort }} + {{- .Values.global.postgresql.servicePort -}} +{{- else -}} + {{- .Values.service.port -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL created database +*/}} +{{- define "postgresql.database" -}} +{{- if .Values.global.postgresql.postgresqlDatabase }} + {{- .Values.global.postgresql.postgresqlDatabase -}} +{{- else if .Values.postgresqlDatabase -}} + {{- .Values.postgresqlDatabase -}} +{{- end -}} +{{- end -}} + +{{/* +Get the password secret. +*/}} +{{- define "postgresql.secretName" -}} +{{- if .Values.global.postgresql.existingSecret }} + {{- printf "%s" (tpl .Values.global.postgresql.existingSecret $) -}} +{{- else if .Values.existingSecret -}} + {{- printf "%s" (tpl .Values.existingSecret $) -}} +{{- else -}} + {{- printf "%s" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Return true if we should use an existingSecret. +*/}} +{{- define "postgresql.useExistingSecret" -}} +{{- if or .Values.global.postgresql.existingSecret .Values.existingSecret -}} + {{- true -}} +{{- end -}} +{{- end -}} + +{{/* +Return true if a secret object should be created +*/}} +{{- define "postgresql.createSecret" -}} +{{- if not (include "postgresql.useExistingSecret" .) -}} + {{- true -}} +{{- end -}} +{{- end -}} + +{{/* +Get the configuration ConfigMap name. +*/}} +{{- define "postgresql.configurationCM" -}} +{{- if .Values.configurationConfigMap -}} +{{- printf "%s" (tpl .Values.configurationConfigMap $) -}} +{{- else -}} +{{- printf "%s-configuration" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Get the extended configuration ConfigMap name. +*/}} +{{- define "postgresql.extendedConfigurationCM" -}} +{{- if .Values.extendedConfConfigMap -}} +{{- printf "%s" (tpl .Values.extendedConfConfigMap $) -}} +{{- else -}} +{{- printf "%s-extended-configuration" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Return true if a configmap should be mounted with PostgreSQL configuration +*/}} +{{- define "postgresql.mountConfigurationCM" -}} +{{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap }} + {{- true -}} +{{- end -}} +{{- end -}} + +{{/* +Get the initialization scripts ConfigMap name. +*/}} +{{- define "postgresql.initdbScriptsCM" -}} +{{- if .Values.initdbScriptsConfigMap -}} +{{- printf "%s" (tpl .Values.initdbScriptsConfigMap $) -}} +{{- else -}} +{{- printf "%s-init-scripts" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Get the initialization scripts Secret name. +*/}} +{{- define "postgresql.initdbScriptsSecret" -}} +{{- printf "%s" (tpl .Values.initdbScriptsSecret $) -}} +{{- end -}} + +{{/* +Get the metrics ConfigMap name. +*/}} +{{- define "postgresql.metricsCM" -}} +{{- printf "%s-metrics" (include "common.names.fullname" .) -}} +{{- end -}} + +{{/* +Get the readiness probe command +*/}} +{{- define "postgresql.readinessProbeCommand" -}} +- | +{{- if (include "postgresql.database" .) }} + exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} +{{- else }} + exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} +{{- end }} +{{- if contains "bitnami/" .Values.image.repository }} + [ -f /opt/bitnami/postgresql/tmp/.initialized ] || [ -f /bitnami/postgresql/.initialized ] +{{- end -}} +{{- end -}} + +{{/* +Compile all warnings into a single message, and call fail. +*/}} +{{- define "postgresql.validateValues" -}} +{{- $messages := list -}} +{{- $messages := append $messages (include "postgresql.validateValues.ldapConfigurationMethod" .) -}} +{{- $messages := append $messages (include "postgresql.validateValues.psp" .) -}} +{{- $messages := append $messages (include "postgresql.validateValues.tls" .) -}} +{{- $messages := without $messages "" -}} +{{- $message := join "\n" $messages -}} + +{{- if $message -}} +{{- printf "\nVALUES VALIDATION:\n%s" $message | fail -}} +{{- end -}} +{{- end -}} + +{{/* +Validate values of Postgresql - If ldap.url is used then you don't need the other settings for ldap +*/}} +{{- define "postgresql.validateValues.ldapConfigurationMethod" -}} +{{- if and .Values.ldap.enabled (and (not (empty .Values.ldap.url)) (not (empty .Values.ldap.server))) }} +postgresql: ldap.url, ldap.server + You cannot set both `ldap.url` and `ldap.server` at the same time. + Please provide a unique way to configure LDAP. + More info at https://www.postgresql.org/docs/current/auth-ldap.html +{{- end -}} +{{- end -}} + +{{/* +Validate values of Postgresql - If PSP is enabled RBAC should be enabled too +*/}} +{{- define "postgresql.validateValues.psp" -}} +{{- if and .Values.psp.create (not .Values.rbac.create) }} +postgresql: psp.create, rbac.create + RBAC should be enabled if PSP is enabled in order for PSP to work. + More info at https://kubernetes.io/docs/concepts/policy/pod-security-policy/#authorizing-policies +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for podsecuritypolicy. +*/}} +{{- define "podsecuritypolicy.apiVersion" -}} +{{- if semverCompare "<1.10-0" .Capabilities.KubeVersion.GitVersion -}} +{{- print "extensions/v1beta1" -}} +{{- else -}} +{{- print "policy/v1beta1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for networkpolicy. +*/}} +{{- define "postgresql.networkPolicy.apiVersion" -}} +{{- if semverCompare ">=1.4-0, <1.7-0" .Capabilities.KubeVersion.GitVersion -}} +"extensions/v1beta1" +{{- else if semverCompare "^1.7-0" .Capabilities.KubeVersion.GitVersion -}} +"networking.k8s.io/v1" +{{- end -}} +{{- end -}} + +{{/* +Validate values of Postgresql TLS - When TLS is enabled, so must be VolumePermissions +*/}} +{{- define "postgresql.validateValues.tls" -}} +{{- if and .Values.tls.enabled (not .Values.volumePermissions.enabled) }} +postgresql: tls.enabled, volumePermissions.enabled + When TLS is enabled you must enable volumePermissions as well to ensure certificates files have + the right permissions. +{{- end -}} +{{- end -}} + +{{/* +Return the path to the cert file. +*/}} +{{- define "postgresql.tlsCert" -}} +{{- required "Certificate filename is required when TLS in enabled" .Values.tls.certFilename | printf "/opt/bitnami/postgresql/certs/%s" -}} +{{- end -}} + +{{/* +Return the path to the cert key file. +*/}} +{{- define "postgresql.tlsCertKey" -}} +{{- required "Certificate Key filename is required when TLS in enabled" .Values.tls.certKeyFilename | printf "/opt/bitnami/postgresql/certs/%s" -}} +{{- end -}} + +{{/* +Return the path to the CA cert file. +*/}} +{{- define "postgresql.tlsCACert" -}} +{{- printf "/opt/bitnami/postgresql/certs/%s" .Values.tls.certCAFilename -}} +{{- end -}} + +{{/* +Return the path to the CRL file. +*/}} +{{- define "postgresql.tlsCRL" -}} +{{- if .Values.tls.crlFilename -}} +{{- printf "/opt/bitnami/postgresql/certs/%s" .Values.tls.crlFilename -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/configmap.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/configmap.yaml new file mode 100644 index 0000000000..3a5ea18ae9 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/configmap.yaml @@ -0,0 +1,31 @@ +{{ if and (or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration) (not .Values.configurationConfigMap) }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "common.names.fullname" . }}-configuration + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +data: +{{- if (.Files.Glob "files/postgresql.conf") }} +{{ (.Files.Glob "files/postgresql.conf").AsConfig | indent 2 }} +{{- else if .Values.postgresqlConfiguration }} + postgresql.conf: | +{{- range $key, $value := default dict .Values.postgresqlConfiguration }} + {{- if kindIs "string" $value }} + {{ $key | snakecase }} = '{{ $value }}' + {{- else }} + {{ $key | snakecase }} = {{ $value }} + {{- end }} +{{- end }} +{{- end }} +{{- if (.Files.Glob "files/pg_hba.conf") }} +{{ (.Files.Glob "files/pg_hba.conf").AsConfig | indent 2 }} +{{- else if .Values.pgHbaConfiguration }} + pg_hba.conf: | +{{ .Values.pgHbaConfiguration | indent 4 }} +{{- end }} +{{ end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extended-config-configmap.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extended-config-configmap.yaml new file mode 100644 index 0000000000..b0dad253b5 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extended-config-configmap.yaml @@ -0,0 +1,26 @@ +{{- if and (or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf) (not .Values.extendedConfConfigMap)}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "common.names.fullname" . }}-extended-configuration + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +data: +{{- with .Files.Glob "files/conf.d/*.conf" }} +{{ .AsConfig | indent 2 }} +{{- end }} +{{ with .Values.postgresqlExtendedConf }} + override.conf: | +{{- range $key, $value := . }} + {{- if kindIs "string" $value }} + {{ $key | snakecase }} = '{{ $value }}' + {{- else }} + {{ $key | snakecase }} = {{ $value }} + {{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extra-list.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extra-list.yaml new file mode 100644 index 0000000000..9ac65f9e16 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/extra-list.yaml @@ -0,0 +1,4 @@ +{{- range .Values.extraDeploy }} +--- +{{ include "common.tplvalues.render" (dict "value" . "context" $) }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/initialization-configmap.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/initialization-configmap.yaml new file mode 100644 index 0000000000..7796c67a93 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/initialization-configmap.yaml @@ -0,0 +1,25 @@ +{{- if and (or (.Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql,sql.gz}") .Values.initdbScripts) (not .Values.initdbScriptsConfigMap) }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "common.names.fullname" . }}-init-scripts + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +{{- with .Files.Glob "files/docker-entrypoint-initdb.d/*.sql.gz" }} +binaryData: +{{- range $path, $bytes := . }} + {{ base $path }}: {{ $.Files.Get $path | b64enc | quote }} +{{- end }} +{{- end }} +data: +{{- with .Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql}" }} +{{ .AsConfig | indent 2 }} +{{- end }} +{{- with .Values.initdbScripts }} +{{ toYaml . | indent 2 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-configmap.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-configmap.yaml new file mode 100644 index 0000000000..fa539582bb --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-configmap.yaml @@ -0,0 +1,14 @@ +{{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "postgresql.metricsCM" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +data: + custom-metrics.yaml: {{ toYaml .Values.metrics.customMetrics | quote }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-svc.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-svc.yaml new file mode 100644 index 0000000000..af8b67e2ff --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/metrics-svc.yaml @@ -0,0 +1,26 @@ +{{- if .Values.metrics.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }}-metrics + labels: + {{- include "common.labels.standard" . | nindent 4 }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- toYaml .Values.metrics.service.annotations | nindent 4 }} + namespace: {{ .Release.Namespace }} +spec: + type: {{ .Values.metrics.service.type }} + {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.metrics.service.loadBalancerIP }} + {{- end }} + ports: + - name: http-metrics + port: 9187 + targetPort: http-metrics + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} + role: primary +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/networkpolicy.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/networkpolicy.yaml new file mode 100644 index 0000000000..4f2740ea0c --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/networkpolicy.yaml @@ -0,0 +1,39 @@ +{{- if .Values.networkPolicy.enabled }} +kind: NetworkPolicy +apiVersion: {{ template "postgresql.networkPolicy.apiVersion" . }} +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + podSelector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} + ingress: + # Allow inbound connections + - ports: + - port: {{ template "postgresql.port" . }} + {{- if not .Values.networkPolicy.allowExternal }} + from: + - podSelector: + matchLabels: + {{ template "common.names.fullname" . }}-client: "true" + {{- if .Values.networkPolicy.explicitNamespacesSelector }} + namespaceSelector: +{{ toYaml .Values.networkPolicy.explicitNamespacesSelector | indent 12 }} + {{- end }} + - podSelector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 14 }} + role: read + {{- end }} + {{- if .Values.metrics.enabled }} + # Allow prometheus scrapes + - ports: + - port: 9187 + {{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/podsecuritypolicy.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..0c49694fad --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/podsecuritypolicy.yaml @@ -0,0 +1,38 @@ +{{- if .Values.psp.create }} +apiVersion: {{ include "podsecuritypolicy.apiVersion" . }} +kind: PodSecurityPolicy +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + privileged: false + volumes: + - 'configMap' + - 'secret' + - 'persistentVolumeClaim' + - 'emptyDir' + - 'projected' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/prometheusrule.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/prometheusrule.yaml new file mode 100644 index 0000000000..d0f408c78f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/prometheusrule.yaml @@ -0,0 +1,23 @@ +{{- if and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: PrometheusRule +metadata: + name: {{ template "common.names.fullname" . }} +{{- with .Values.metrics.prometheusRule.namespace }} + namespace: {{ . }} +{{- end }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- with .Values.metrics.prometheusRule.additionalLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} +spec: +{{- with .Values.metrics.prometheusRule.rules }} + groups: + - name: {{ template "postgresql.name" $ }} + rules: {{ tpl (toYaml .) $ | nindent 8 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/role.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/role.yaml new file mode 100644 index 0000000000..017a5716b1 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/role.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create }} +kind: Role +apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +rules: + {{- if .Values.psp.create }} + - apiGroups: ["extensions"] + resources: ["podsecuritypolicies"] + verbs: ["use"] + resourceNames: + - {{ template "common.names.fullname" . }} + {{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/rolebinding.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/rolebinding.yaml new file mode 100644 index 0000000000..189775a15a --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create }} +kind: RoleBinding +apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +roleRef: + kind: Role + name: {{ template "common.names.fullname" . }} + apiGroup: rbac.authorization.k8s.io +subjects: + - kind: ServiceAccount + name: {{ default (include "common.names.fullname" . ) .Values.serviceAccount.name }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/secrets.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/secrets.yaml new file mode 100644 index 0000000000..d492cd593b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/secrets.yaml @@ -0,0 +1,24 @@ +{{- if (include "postgresql.createSecret" .) }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +type: Opaque +data: + {{- if not (eq (include "postgresql.username" .) "postgres") }} + postgresql-postgres-password: {{ include "postgresql.postgres.password" . | b64enc | quote }} + {{- end }} + postgresql-password: {{ include "postgresql.password" . | b64enc | quote }} + {{- if .Values.replication.enabled }} + postgresql-replication-password: {{ include "postgresql.replication.password" . | b64enc | quote }} + {{- end }} + {{- if (and .Values.ldap.enabled .Values.ldap.bind_password)}} + postgresql-ldap-password: {{ .Values.ldap.bind_password | b64enc | quote }} + {{- end }} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/serviceaccount.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/serviceaccount.yaml new file mode 100644 index 0000000000..03f0f50e7d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/serviceaccount.yaml @@ -0,0 +1,12 @@ +{{- if and (.Values.serviceAccount.enabled) (not .Values.serviceAccount.name) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "common.labels.standard" . | nindent 4 }} + name: {{ template "common.names.fullname" . }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/servicemonitor.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/servicemonitor.yaml new file mode 100644 index 0000000000..587ce85b87 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/servicemonitor.yaml @@ -0,0 +1,33 @@ +{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "common.names.fullname" . }} + {{- if .Values.metrics.serviceMonitor.namespace }} + namespace: {{ .Values.metrics.serviceMonitor.namespace }} + {{- end }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.metrics.serviceMonitor.additionalLabels }} + {{- toYaml .Values.metrics.serviceMonitor.additionalLabels | nindent 4 }} + {{- end }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + +spec: + endpoints: + - port: http-metrics + {{- if .Values.metrics.serviceMonitor.interval }} + interval: {{ .Values.metrics.serviceMonitor.interval }} + {{- end }} + {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} + {{- end }} + namespaceSelector: + matchNames: + - {{ .Release.Namespace }} + selector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset-readreplicas.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset-readreplicas.yaml new file mode 100644 index 0000000000..b038299bf6 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset-readreplicas.yaml @@ -0,0 +1,411 @@ +{{- if .Values.replication.enabled }} +{{- $readReplicasResources := coalesce .Values.readReplicas.resources .Values.resources -}} +apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} +kind: StatefulSet +metadata: + name: "{{ template "common.names.fullname" . }}-read" + labels: {{- include "common.labels.standard" . | nindent 4 }} + app.kubernetes.io/component: read +{{- with .Values.readReplicas.labels }} +{{ toYaml . | indent 4 }} +{{- end }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- with .Values.readReplicas.annotations }} + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + serviceName: {{ template "common.names.fullname" . }}-headless + replicas: {{ .Values.replication.readReplicas }} + selector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} + role: read + template: + metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 8 }} + app.kubernetes.io/component: read + role: read +{{- with .Values.readReplicas.podLabels }} +{{ toYaml . | indent 8 }} +{{- end }} +{{- with .Values.readReplicas.podAnnotations }} + annotations: +{{ toYaml . | indent 8 }} +{{- end }} + spec: + {{- if .Values.schedulerName }} + schedulerName: "{{ .Values.schedulerName }}" + {{- end }} +{{- include "postgresql.imagePullSecrets" . | indent 6 }} + {{- if .Values.readReplicas.affinity }} + affinity: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.affinity "context" $) | nindent 8 }} + {{- else }} + affinity: + podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.readReplicas.podAffinityPreset "component" "read" "context" $) | nindent 10 }} + podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.readReplicas.podAntiAffinityPreset "component" "read" "context" $) | nindent 10 }} + nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.readReplicas.nodeAffinityPreset.type "key" .Values.readReplicas.nodeAffinityPreset.key "values" .Values.readReplicas.nodeAffinityPreset.values) | nindent 10 }} + {{- end }} + {{- if .Values.readReplicas.nodeSelector }} + nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.nodeSelector "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.readReplicas.tolerations }} + tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.tolerations "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.securityContext.enabled }} + securityContext: {{- omit .Values.securityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.serviceAccount.enabled }} + serviceAccountName: {{ default (include "common.names.fullname" . ) .Values.serviceAccount.name}} + {{- end }} + {{- if or .Values.readReplicas.extraInitContainers (and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled))) }} + initContainers: + {{- if and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled) .Values.tls.enabled) }} + - name: init-chmod-data + image: {{ template "postgresql.volumePermissions.image" . }} + imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} + {{- if .Values.resources }} + resources: {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + command: + - /bin/sh + - -cx + - | + {{- if .Values.persistence.enabled }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown `id -u`:`id -G | cut -d " " -f2` {{ .Values.persistence.mountPath }} + {{- else }} + chown {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} {{ .Values.persistence.mountPath }} + {{- end }} + mkdir -p {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + chmod 700 {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + find {{ .Values.persistence.mountPath }} -mindepth 1 -maxdepth 1 {{- if not (include "postgresql.mountConfigurationCM" .) }} -not -name "conf" {{- end }} -not -name ".snapshot" -not -name "lost+found" | \ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + xargs chown -R `id -u`:`id -G | cut -d " " -f2` + {{- else }} + xargs chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} + {{- end }} + {{- end }} + {{- if and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled }} + chmod -R 777 /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + cp /tmp/certs/* /opt/bitnami/postgresql/certs/ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown -R `id -u`:`id -G | cut -d " " -f2` /opt/bitnami/postgresql/certs/ + {{- else }} + chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} /opt/bitnami/postgresql/certs/ + {{- end }} + chmod 600 {{ template "postgresql.tlsCertKey" . }} + {{- end }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + securityContext: {{- omit .Values.volumePermissions.securityContext "runAsUser" | toYaml | nindent 12 }} + {{- else }} + securityContext: {{- .Values.volumePermissions.securityContext | toYaml | nindent 12 }} + {{- end }} + volumeMounts: + {{ if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + mountPath: /tmp/certs + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + {{- end }} + {{- end }} + {{- if .Values.readReplicas.extraInitContainers }} + {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.extraInitContainers "context" $ ) | nindent 8 }} + {{- end }} + {{- end }} + {{- if .Values.readReplicas.priorityClassName }} + priorityClassName: {{ .Values.readReplicas.priorityClassName }} + {{- end }} + containers: + - name: {{ template "common.names.fullname" . }} + image: {{ template "postgresql.image" . }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + {{- if $readReplicasResources }} + resources: {{- toYaml $readReplicasResources | nindent 12 }} + {{- end }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 12 }} + {{- end }} + env: + - name: BITNAMI_DEBUG + value: {{ ternary "true" "false" .Values.image.debug | quote }} + - name: POSTGRESQL_VOLUME_DIR + value: "{{ .Values.persistence.mountPath }}" + - name: POSTGRESQL_PORT_NUMBER + value: "{{ template "postgresql.port" . }}" + {{- if .Values.persistence.mountPath }} + - name: PGDATA + value: {{ .Values.postgresqlDataDir | quote }} + {{- end }} + - name: POSTGRES_REPLICATION_MODE + value: "slave" + - name: POSTGRES_REPLICATION_USER + value: {{ include "postgresql.replication.username" . | quote }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_REPLICATION_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-replication-password" + {{- else }} + - name: POSTGRES_REPLICATION_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-replication-password + {{- end }} + - name: POSTGRES_CLUSTER_APP_NAME + value: {{ .Values.replication.applicationName }} + - name: POSTGRES_MASTER_HOST + value: {{ template "common.names.fullname" . }} + - name: POSTGRES_MASTER_PORT_NUMBER + value: {{ include "postgresql.port" . | quote }} + {{- if not (eq (include "postgresql.username" .) "postgres") }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-postgres-password" + {{- else }} + - name: POSTGRES_POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-postgres-password + {{- end }} + {{- end }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-password" + {{- else }} + - name: POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-password + {{- end }} + - name: POSTGRESQL_ENABLE_TLS + value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} + {{- if .Values.tls.enabled }} + - name: POSTGRESQL_TLS_PREFER_SERVER_CIPHERS + value: {{ ternary "yes" "no" .Values.tls.preferServerCiphers | quote }} + - name: POSTGRESQL_TLS_CERT_FILE + value: {{ template "postgresql.tlsCert" . }} + - name: POSTGRESQL_TLS_KEY_FILE + value: {{ template "postgresql.tlsCertKey" . }} + {{- if .Values.tls.certCAFilename }} + - name: POSTGRESQL_TLS_CA_FILE + value: {{ template "postgresql.tlsCACert" . }} + {{- end }} + {{- if .Values.tls.crlFilename }} + - name: POSTGRESQL_TLS_CRL_FILE + value: {{ template "postgresql.tlsCRL" . }} + {{- end }} + {{- end }} + - name: POSTGRESQL_LOG_HOSTNAME + value: {{ .Values.audit.logHostname | quote }} + - name: POSTGRESQL_LOG_CONNECTIONS + value: {{ .Values.audit.logConnections | quote }} + - name: POSTGRESQL_LOG_DISCONNECTIONS + value: {{ .Values.audit.logDisconnections | quote }} + {{- if .Values.audit.logLinePrefix }} + - name: POSTGRESQL_LOG_LINE_PREFIX + value: {{ .Values.audit.logLinePrefix | quote }} + {{- end }} + {{- if .Values.audit.logTimezone }} + - name: POSTGRESQL_LOG_TIMEZONE + value: {{ .Values.audit.logTimezone | quote }} + {{- end }} + {{- if .Values.audit.pgAuditLog }} + - name: POSTGRESQL_PGAUDIT_LOG + value: {{ .Values.audit.pgAuditLog | quote }} + {{- end }} + - name: POSTGRESQL_PGAUDIT_LOG_CATALOG + value: {{ .Values.audit.pgAuditLogCatalog | quote }} + - name: POSTGRESQL_CLIENT_MIN_MESSAGES + value: {{ .Values.audit.clientMinMessages | quote }} + - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES + value: {{ .Values.postgresqlSharedPreloadLibraries | quote }} + {{- if .Values.postgresqlMaxConnections }} + - name: POSTGRESQL_MAX_CONNECTIONS + value: {{ .Values.postgresqlMaxConnections | quote }} + {{- end }} + {{- if .Values.postgresqlPostgresConnectionLimit }} + - name: POSTGRESQL_POSTGRES_CONNECTION_LIMIT + value: {{ .Values.postgresqlPostgresConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlDbUserConnectionLimit }} + - name: POSTGRESQL_USERNAME_CONNECTION_LIMIT + value: {{ .Values.postgresqlDbUserConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesInterval }} + - name: POSTGRESQL_TCP_KEEPALIVES_INTERVAL + value: {{ .Values.postgresqlTcpKeepalivesInterval | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesIdle }} + - name: POSTGRESQL_TCP_KEEPALIVES_IDLE + value: {{ .Values.postgresqlTcpKeepalivesIdle | quote }} + {{- end }} + {{- if .Values.postgresqlStatementTimeout }} + - name: POSTGRESQL_STATEMENT_TIMEOUT + value: {{ .Values.postgresqlStatementTimeout | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesCount }} + - name: POSTGRESQL_TCP_KEEPALIVES_COUNT + value: {{ .Values.postgresqlTcpKeepalivesCount | quote }} + {{- end }} + {{- if .Values.postgresqlPghbaRemoveFilters }} + - name: POSTGRESQL_PGHBA_REMOVE_FILTERS + value: {{ .Values.postgresqlPghbaRemoveFilters | quote }} + {{- end }} + ports: + - name: tcp-postgresql + containerPort: {{ template "postgresql.port" . }} + {{- if .Values.livenessProbe.enabled }} + livenessProbe: + exec: + command: + - /bin/sh + - -c + {{- if (include "postgresql.database" .) }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- else }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- end }} + initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.livenessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} + successThreshold: {{ .Values.livenessProbe.successThreshold }} + failureThreshold: {{ .Values.livenessProbe.failureThreshold }} + {{- else if .Values.customLivenessProbe }} + livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customLivenessProbe "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.readinessProbe.enabled }} + readinessProbe: + exec: + command: + - /bin/sh + - -c + - -e + {{- include "postgresql.readinessProbeCommand" . | nindent 16 }} + initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.readinessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} + successThreshold: {{ .Values.readinessProbe.successThreshold }} + failureThreshold: {{ .Values.readinessProbe.failureThreshold }} + {{- else if .Values.customReadinessProbe }} + readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customReadinessProbe "context" $) | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.usePasswordFile }} + - name: postgresql-password + mountPath: /opt/bitnami/postgresql/secrets/ + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{ end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + mountPath: /bitnami/postgresql/conf/conf.d/ + {{- end }} + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap }} + - name: postgresql-config + mountPath: /bitnami/postgresql/conf + {{- end }} + {{- if .Values.tls.enabled }} + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + readOnly: true + {{- end }} + {{- if .Values.readReplicas.extraVolumeMounts }} + {{- toYaml .Values.readReplicas.extraVolumeMounts | nindent 12 }} + {{- end }} +{{- if .Values.readReplicas.sidecars }} +{{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.sidecars "context" $ ) | nindent 8 }} +{{- end }} + volumes: + {{- if .Values.usePasswordFile }} + - name: postgresql-password + secret: + secretName: {{ template "postgresql.secretName" . }} + {{- end }} + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap}} + - name: postgresql-config + configMap: + name: {{ template "postgresql.configurationCM" . }} + {{- end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + configMap: + name: {{ template "postgresql.extendedConfigurationCM" . }} + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + secret: + secretName: {{ required "A secret containing TLS certificates is required when TLS is enabled" .Values.tls.certificatesSecret }} + - name: postgresql-certificates + emptyDir: {} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + emptyDir: + medium: Memory + sizeLimit: 1Gi + {{- end }} + {{- if or (not .Values.persistence.enabled) (not .Values.readReplicas.persistence.enabled) }} + - name: data + emptyDir: {} + {{- end }} + {{- if .Values.readReplicas.extraVolumes }} + {{- toYaml .Values.readReplicas.extraVolumes | nindent 8 }} + {{- end }} + updateStrategy: + type: {{ .Values.updateStrategy.type }} + {{- if (eq "Recreate" .Values.updateStrategy.type) }} + rollingUpdate: null + {{- end }} +{{- if and .Values.persistence.enabled .Values.readReplicas.persistence.enabled }} + volumeClaimTemplates: + - metadata: + name: data + {{- with .Values.persistence.annotations }} + annotations: + {{- range $key, $value := . }} + {{ $key }}: {{ $value }} + {{- end }} + {{- end }} + spec: + accessModes: + {{- range .Values.persistence.accessModes }} + - {{ . | quote }} + {{- end }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + {{ include "common.storage.class" (dict "persistence" .Values.persistence "global" .Values.global) }} + + {{- if .Values.persistence.selector }} + selector: {{- include "common.tplvalues.render" (dict "value" .Values.persistence.selector "context" $) | nindent 10 }} + {{- end -}} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset.yaml new file mode 100644 index 0000000000..f8163fd99f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/statefulset.yaml @@ -0,0 +1,609 @@ +apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} +kind: StatefulSet +metadata: + name: {{ template "postgresql.primary.fullname" . }} + labels: {{- include "common.labels.standard" . | nindent 4 }} + app.kubernetes.io/component: primary + {{- with .Values.primary.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- with .Values.primary.annotations }} + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + serviceName: {{ template "common.names.fullname" . }}-headless + replicas: 1 + updateStrategy: + type: {{ .Values.updateStrategy.type }} + {{- if (eq "Recreate" .Values.updateStrategy.type) }} + rollingUpdate: null + {{- end }} + selector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} + role: primary + template: + metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 8 }} + role: primary + app.kubernetes.io/component: primary + {{- with .Values.primary.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.primary.podAnnotations }} + annotations: {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- if .Values.schedulerName }} + schedulerName: "{{ .Values.schedulerName }}" + {{- end }} +{{- include "postgresql.imagePullSecrets" . | indent 6 }} + {{- if .Values.primary.affinity }} + affinity: {{- include "common.tplvalues.render" (dict "value" .Values.primary.affinity "context" $) | nindent 8 }} + {{- else }} + affinity: + podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAffinityPreset "component" "primary" "context" $) | nindent 10 }} + podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAntiAffinityPreset "component" "primary" "context" $) | nindent 10 }} + nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.primary.nodeAffinityPreset.type "key" .Values.primary.nodeAffinityPreset.key "values" .Values.primary.nodeAffinityPreset.values) | nindent 10 }} + {{- end }} + {{- if .Values.primary.nodeSelector }} + nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.primary.nodeSelector "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.primary.tolerations }} + tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.primary.tolerations "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.securityContext.enabled }} + securityContext: {{- omit .Values.securityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.serviceAccount.enabled }} + serviceAccountName: {{ default (include "common.names.fullname" . ) .Values.serviceAccount.name }} + {{- end }} + {{- if or .Values.primary.extraInitContainers (and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled))) }} + initContainers: + {{- if and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled) .Values.tls.enabled) }} + - name: init-chmod-data + image: {{ template "postgresql.volumePermissions.image" . }} + imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} + {{- if .Values.resources }} + resources: {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + command: + - /bin/sh + - -cx + - | + {{- if .Values.persistence.enabled }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown `id -u`:`id -G | cut -d " " -f2` {{ .Values.persistence.mountPath }} + {{- else }} + chown {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} {{ .Values.persistence.mountPath }} + {{- end }} + mkdir -p {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + chmod 700 {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + find {{ .Values.persistence.mountPath }} -mindepth 1 -maxdepth 1 {{- if not (include "postgresql.mountConfigurationCM" .) }} -not -name "conf" {{- end }} -not -name ".snapshot" -not -name "lost+found" | \ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + xargs chown -R `id -u`:`id -G | cut -d " " -f2` + {{- else }} + xargs chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} + {{- end }} + {{- end }} + {{- if and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled }} + chmod -R 777 /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + cp /tmp/certs/* /opt/bitnami/postgresql/certs/ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown -R `id -u`:`id -G | cut -d " " -f2` /opt/bitnami/postgresql/certs/ + {{- else }} + chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} /opt/bitnami/postgresql/certs/ + {{- end }} + chmod 600 {{ template "postgresql.tlsCertKey" . }} + {{- end }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + securityContext: {{- omit .Values.volumePermissions.securityContext "runAsUser" | toYaml | nindent 12 }} + {{- else }} + securityContext: {{- .Values.volumePermissions.securityContext | toYaml | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + mountPath: /tmp/certs + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + {{- end }} + {{- end }} + {{- if .Values.primary.extraInitContainers }} + {{- include "common.tplvalues.render" ( dict "value" .Values.primary.extraInitContainers "context" $ ) | nindent 8 }} + {{- end }} + {{- end }} + {{- if .Values.primary.priorityClassName }} + priorityClassName: {{ .Values.primary.priorityClassName }} + {{- end }} + containers: + - name: {{ template "common.names.fullname" . }} + image: {{ template "postgresql.image" . }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + {{- if .Values.resources }} + resources: {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 12 }} + {{- end }} + env: + - name: BITNAMI_DEBUG + value: {{ ternary "true" "false" .Values.image.debug | quote }} + - name: POSTGRESQL_PORT_NUMBER + value: "{{ template "postgresql.port" . }}" + - name: POSTGRESQL_VOLUME_DIR + value: "{{ .Values.persistence.mountPath }}" + {{- if .Values.postgresqlInitdbArgs }} + - name: POSTGRES_INITDB_ARGS + value: {{ .Values.postgresqlInitdbArgs | quote }} + {{- end }} + {{- if .Values.postgresqlInitdbWalDir }} + - name: POSTGRES_INITDB_WALDIR + value: {{ .Values.postgresqlInitdbWalDir | quote }} + {{- end }} + {{- if .Values.initdbUser }} + - name: POSTGRESQL_INITSCRIPTS_USERNAME + value: {{ .Values.initdbUser }} + {{- end }} + {{- if .Values.initdbPassword }} + - name: POSTGRESQL_INITSCRIPTS_PASSWORD + value: {{ .Values.initdbPassword }} + {{- end }} + {{- if .Values.persistence.mountPath }} + - name: PGDATA + value: {{ .Values.postgresqlDataDir | quote }} + {{- end }} + {{- if .Values.primaryAsStandBy.enabled }} + - name: POSTGRES_MASTER_HOST + value: {{ .Values.primaryAsStandBy.primaryHost }} + - name: POSTGRES_MASTER_PORT_NUMBER + value: {{ .Values.primaryAsStandBy.primaryPort | quote }} + {{- end }} + {{- if or .Values.replication.enabled .Values.primaryAsStandBy.enabled }} + - name: POSTGRES_REPLICATION_MODE + {{- if .Values.primaryAsStandBy.enabled }} + value: "slave" + {{- else }} + value: "master" + {{- end }} + - name: POSTGRES_REPLICATION_USER + value: {{ include "postgresql.replication.username" . | quote }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_REPLICATION_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-replication-password" + {{- else }} + - name: POSTGRES_REPLICATION_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-replication-password + {{- end }} + {{- if not (eq .Values.replication.synchronousCommit "off")}} + - name: POSTGRES_SYNCHRONOUS_COMMIT_MODE + value: {{ .Values.replication.synchronousCommit | quote }} + - name: POSTGRES_NUM_SYNCHRONOUS_REPLICAS + value: {{ .Values.replication.numSynchronousReplicas | quote }} + {{- end }} + - name: POSTGRES_CLUSTER_APP_NAME + value: {{ .Values.replication.applicationName }} + {{- end }} + {{- if not (eq (include "postgresql.username" .) "postgres") }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-postgres-password" + {{- else }} + - name: POSTGRES_POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-postgres-password + {{- end }} + {{- end }} + - name: POSTGRES_USER + value: {{ include "postgresql.username" . | quote }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-password" + {{- else }} + - name: POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-password + {{- end }} + {{- if (include "postgresql.database" .) }} + - name: POSTGRES_DB + value: {{ (include "postgresql.database" .) | quote }} + {{- end }} + {{- if .Values.extraEnv }} + {{- include "common.tplvalues.render" (dict "value" .Values.extraEnv "context" $) | nindent 12 }} + {{- end }} + - name: POSTGRESQL_ENABLE_LDAP + value: {{ ternary "yes" "no" .Values.ldap.enabled | quote }} + {{- if .Values.ldap.enabled }} + - name: POSTGRESQL_LDAP_SERVER + value: {{ .Values.ldap.server }} + - name: POSTGRESQL_LDAP_PORT + value: {{ .Values.ldap.port | quote }} + - name: POSTGRESQL_LDAP_SCHEME + value: {{ .Values.ldap.scheme }} + {{- if .Values.ldap.tls }} + - name: POSTGRESQL_LDAP_TLS + value: "1" + {{- end }} + - name: POSTGRESQL_LDAP_PREFIX + value: {{ .Values.ldap.prefix | quote }} + - name: POSTGRESQL_LDAP_SUFFIX + value: {{ .Values.ldap.suffix | quote }} + - name: POSTGRESQL_LDAP_BASE_DN + value: {{ .Values.ldap.baseDN }} + - name: POSTGRESQL_LDAP_BIND_DN + value: {{ .Values.ldap.bindDN }} + {{- if (not (empty .Values.ldap.bind_password)) }} + - name: POSTGRESQL_LDAP_BIND_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-ldap-password + {{- end}} + - name: POSTGRESQL_LDAP_SEARCH_ATTR + value: {{ .Values.ldap.search_attr }} + - name: POSTGRESQL_LDAP_SEARCH_FILTER + value: {{ .Values.ldap.search_filter }} + - name: POSTGRESQL_LDAP_URL + value: {{ .Values.ldap.url }} + {{- end}} + - name: POSTGRESQL_ENABLE_TLS + value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} + {{- if .Values.tls.enabled }} + - name: POSTGRESQL_TLS_PREFER_SERVER_CIPHERS + value: {{ ternary "yes" "no" .Values.tls.preferServerCiphers | quote }} + - name: POSTGRESQL_TLS_CERT_FILE + value: {{ template "postgresql.tlsCert" . }} + - name: POSTGRESQL_TLS_KEY_FILE + value: {{ template "postgresql.tlsCertKey" . }} + {{- if .Values.tls.certCAFilename }} + - name: POSTGRESQL_TLS_CA_FILE + value: {{ template "postgresql.tlsCACert" . }} + {{- end }} + {{- if .Values.tls.crlFilename }} + - name: POSTGRESQL_TLS_CRL_FILE + value: {{ template "postgresql.tlsCRL" . }} + {{- end }} + {{- end }} + - name: POSTGRESQL_LOG_HOSTNAME + value: {{ .Values.audit.logHostname | quote }} + - name: POSTGRESQL_LOG_CONNECTIONS + value: {{ .Values.audit.logConnections | quote }} + - name: POSTGRESQL_LOG_DISCONNECTIONS + value: {{ .Values.audit.logDisconnections | quote }} + {{- if .Values.audit.logLinePrefix }} + - name: POSTGRESQL_LOG_LINE_PREFIX + value: {{ .Values.audit.logLinePrefix | quote }} + {{- end }} + {{- if .Values.audit.logTimezone }} + - name: POSTGRESQL_LOG_TIMEZONE + value: {{ .Values.audit.logTimezone | quote }} + {{- end }} + {{- if .Values.audit.pgAuditLog }} + - name: POSTGRESQL_PGAUDIT_LOG + value: {{ .Values.audit.pgAuditLog | quote }} + {{- end }} + - name: POSTGRESQL_PGAUDIT_LOG_CATALOG + value: {{ .Values.audit.pgAuditLogCatalog | quote }} + - name: POSTGRESQL_CLIENT_MIN_MESSAGES + value: {{ .Values.audit.clientMinMessages | quote }} + - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES + value: {{ .Values.postgresqlSharedPreloadLibraries | quote }} + {{- if .Values.postgresqlMaxConnections }} + - name: POSTGRESQL_MAX_CONNECTIONS + value: {{ .Values.postgresqlMaxConnections | quote }} + {{- end }} + {{- if .Values.postgresqlPostgresConnectionLimit }} + - name: POSTGRESQL_POSTGRES_CONNECTION_LIMIT + value: {{ .Values.postgresqlPostgresConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlDbUserConnectionLimit }} + - name: POSTGRESQL_USERNAME_CONNECTION_LIMIT + value: {{ .Values.postgresqlDbUserConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesInterval }} + - name: POSTGRESQL_TCP_KEEPALIVES_INTERVAL + value: {{ .Values.postgresqlTcpKeepalivesInterval | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesIdle }} + - name: POSTGRESQL_TCP_KEEPALIVES_IDLE + value: {{ .Values.postgresqlTcpKeepalivesIdle | quote }} + {{- end }} + {{- if .Values.postgresqlStatementTimeout }} + - name: POSTGRESQL_STATEMENT_TIMEOUT + value: {{ .Values.postgresqlStatementTimeout | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesCount }} + - name: POSTGRESQL_TCP_KEEPALIVES_COUNT + value: {{ .Values.postgresqlTcpKeepalivesCount | quote }} + {{- end }} + {{- if .Values.postgresqlPghbaRemoveFilters }} + - name: POSTGRESQL_PGHBA_REMOVE_FILTERS + value: {{ .Values.postgresqlPghbaRemoveFilters | quote }} + {{- end }} + {{- if .Values.extraEnvVarsCM }} + envFrom: + - configMapRef: + name: {{ tpl .Values.extraEnvVarsCM . }} + {{- end }} + ports: + - name: tcp-postgresql + containerPort: {{ template "postgresql.port" . }} + {{- if .Values.startupProbe.enabled }} + startupProbe: + exec: + command: + - /bin/sh + - -c + {{- if (include "postgresql.database" .) }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- else }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- end }} + initialDelaySeconds: {{ .Values.startupProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.startupProbe.periodSeconds }} + timeoutSeconds: {{ .Values.startupProbe.timeoutSeconds }} + successThreshold: {{ .Values.startupProbe.successThreshold }} + failureThreshold: {{ .Values.startupProbe.failureThreshold }} + {{- else if .Values.customStartupProbe }} + startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customStartupProbe "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.livenessProbe.enabled }} + livenessProbe: + exec: + command: + - /bin/sh + - -c + {{- if (include "postgresql.database" .) }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- else }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- end }} + initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.livenessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} + successThreshold: {{ .Values.livenessProbe.successThreshold }} + failureThreshold: {{ .Values.livenessProbe.failureThreshold }} + {{- else if .Values.customLivenessProbe }} + livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customLivenessProbe "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.readinessProbe.enabled }} + readinessProbe: + exec: + command: + - /bin/sh + - -c + - -e + {{- include "postgresql.readinessProbeCommand" . | nindent 16 }} + initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.readinessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} + successThreshold: {{ .Values.readinessProbe.successThreshold }} + failureThreshold: {{ .Values.readinessProbe.failureThreshold }} + {{- else if .Values.customReadinessProbe }} + readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customReadinessProbe "context" $) | nindent 12 }} + {{- end }} + volumeMounts: + {{- if or (.Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql,sql.gz}") .Values.initdbScriptsConfigMap .Values.initdbScripts }} + - name: custom-init-scripts + mountPath: /docker-entrypoint-initdb.d/ + {{- end }} + {{- if .Values.initdbScriptsSecret }} + - name: custom-init-scripts-secret + mountPath: /docker-entrypoint-initdb.d/secret + {{- end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + mountPath: /bitnami/postgresql/conf/conf.d/ + {{- end }} + {{- if .Values.usePasswordFile }} + - name: postgresql-password + mountPath: /opt/bitnami/postgresql/secrets/ + {{- end }} + {{- if .Values.tls.enabled }} + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + readOnly: true + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap }} + - name: postgresql-config + mountPath: /bitnami/postgresql/conf + {{- end }} + {{- if .Values.primary.extraVolumeMounts }} + {{- toYaml .Values.primary.extraVolumeMounts | nindent 12 }} + {{- end }} +{{- if .Values.primary.sidecars }} +{{- include "common.tplvalues.render" ( dict "value" .Values.primary.sidecars "context" $ ) | nindent 8 }} +{{- end }} +{{- if .Values.metrics.enabled }} + - name: metrics + image: {{ template "postgresql.metrics.image" . }} + imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} + {{- if .Values.metrics.securityContext.enabled }} + securityContext: {{- omit .Values.metrics.securityContext "enabled" | toYaml | nindent 12 }} + {{- end }} + env: + {{- $database := required "In order to enable metrics you need to specify a database (.Values.postgresqlDatabase or .Values.global.postgresql.postgresqlDatabase)" (include "postgresql.database" .) }} + {{- $sslmode := ternary "require" "disable" .Values.tls.enabled }} + {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} + - name: DATA_SOURCE_NAME + value: {{ printf "host=127.0.0.1 port=%d user=%s sslmode=%s sslcert=%s sslkey=%s" (int (include "postgresql.port" .)) (include "postgresql.username" .) $sslmode (include "postgresql.tlsCert" .) (include "postgresql.tlsCertKey" .) }} + {{- else }} + - name: DATA_SOURCE_URI + value: {{ printf "127.0.0.1:%d/%s?sslmode=%s" (int (include "postgresql.port" .)) $database $sslmode }} + {{- end }} + {{- if .Values.usePasswordFile }} + - name: DATA_SOURCE_PASS_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-password" + {{- else }} + - name: DATA_SOURCE_PASS + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-password + {{- end }} + - name: DATA_SOURCE_USER + value: {{ template "postgresql.username" . }} + {{- if .Values.metrics.extraEnvVars }} + {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.livenessProbe.enabled }} + livenessProbe: + httpGet: + path: / + port: http-metrics + initialDelaySeconds: {{ .Values.metrics.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.metrics.livenessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.metrics.livenessProbe.timeoutSeconds }} + successThreshold: {{ .Values.metrics.livenessProbe.successThreshold }} + failureThreshold: {{ .Values.metrics.livenessProbe.failureThreshold }} + {{- end }} + {{- if .Values.readinessProbe.enabled }} + readinessProbe: + httpGet: + path: / + port: http-metrics + initialDelaySeconds: {{ .Values.metrics.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.metrics.readinessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.metrics.readinessProbe.timeoutSeconds }} + successThreshold: {{ .Values.metrics.readinessProbe.successThreshold }} + failureThreshold: {{ .Values.metrics.readinessProbe.failureThreshold }} + {{- end }} + volumeMounts: + {{- if .Values.usePasswordFile }} + - name: postgresql-password + mountPath: /opt/bitnami/postgresql/secrets/ + {{- end }} + {{- if .Values.tls.enabled }} + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + readOnly: true + {{- end }} + {{- if .Values.metrics.customMetrics }} + - name: custom-metrics + mountPath: /conf + readOnly: true + args: ["--extend.query-path", "/conf/custom-metrics.yaml"] + {{- end }} + ports: + - name: http-metrics + containerPort: 9187 + {{- if .Values.metrics.resources }} + resources: {{- toYaml .Values.metrics.resources | nindent 12 }} + {{- end }} +{{- end }} + volumes: + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap}} + - name: postgresql-config + configMap: + name: {{ template "postgresql.configurationCM" . }} + {{- end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + configMap: + name: {{ template "postgresql.extendedConfigurationCM" . }} + {{- end }} + {{- if .Values.usePasswordFile }} + - name: postgresql-password + secret: + secretName: {{ template "postgresql.secretName" . }} + {{- end }} + {{- if or (.Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql,sql.gz}") .Values.initdbScriptsConfigMap .Values.initdbScripts }} + - name: custom-init-scripts + configMap: + name: {{ template "postgresql.initdbScriptsCM" . }} + {{- end }} + {{- if .Values.initdbScriptsSecret }} + - name: custom-init-scripts-secret + secret: + secretName: {{ template "postgresql.initdbScriptsSecret" . }} + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + secret: + secretName: {{ required "A secret containing TLS certificates is required when TLS is enabled" .Values.tls.certificatesSecret }} + - name: postgresql-certificates + emptyDir: {} + {{- end }} + {{- if .Values.primary.extraVolumes }} + {{- toYaml .Values.primary.extraVolumes | nindent 8 }} + {{- end }} + {{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} + - name: custom-metrics + configMap: + name: {{ template "postgresql.metricsCM" . }} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + emptyDir: + medium: Memory + sizeLimit: 1Gi + {{- end }} +{{- if and .Values.persistence.enabled .Values.persistence.existingClaim }} + - name: data + persistentVolumeClaim: +{{- with .Values.persistence.existingClaim }} + claimName: {{ tpl . $ }} +{{- end }} +{{- else if not .Values.persistence.enabled }} + - name: data + emptyDir: {} +{{- else if and .Values.persistence.enabled (not .Values.persistence.existingClaim) }} + volumeClaimTemplates: + - metadata: + name: data + {{- with .Values.persistence.annotations }} + annotations: + {{- range $key, $value := . }} + {{ $key }}: {{ $value }} + {{- end }} + {{- end }} + spec: + accessModes: + {{- range .Values.persistence.accessModes }} + - {{ . | quote }} + {{- end }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + {{ include "common.storage.class" (dict "persistence" .Values.persistence "global" .Values.global) }} + {{- if .Values.persistence.selector }} + selector: {{- include "common.tplvalues.render" (dict "value" .Values.persistence.selector "context" $) | nindent 10 }} + {{- end -}} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-headless.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-headless.yaml new file mode 100644 index 0000000000..6f5f3b9ee4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-headless.yaml @@ -0,0 +1,28 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }}-headless + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + # Use this annotation in addition to the actual publishNotReadyAddresses + # field below because the annotation will stop being respected soon but the + # field is broken in some versions of Kubernetes: + # https://github.com/kubernetes/kubernetes/issues/58662 + service.alpha.kubernetes.io/tolerate-unready-endpoints: "true" + namespace: {{ .Release.Namespace }} +spec: + type: ClusterIP + clusterIP: None + # We want all pods in the StatefulSet to have their addresses published for + # the sake of the other Postgresql pods even before they're ready, since they + # have to be able to talk to each other in order to become ready. + publishNotReadyAddresses: true + ports: + - name: tcp-postgresql + port: {{ template "postgresql.port" . }} + targetPort: tcp-postgresql + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-read.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-read.yaml new file mode 100644 index 0000000000..56195ea1e6 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc-read.yaml @@ -0,0 +1,43 @@ +{{- if .Values.replication.enabled }} +{{- $serviceAnnotations := coalesce .Values.readReplicas.service.annotations .Values.service.annotations -}} +{{- $serviceType := coalesce .Values.readReplicas.service.type .Values.service.type -}} +{{- $serviceLoadBalancerIP := coalesce .Values.readReplicas.service.loadBalancerIP .Values.service.loadBalancerIP -}} +{{- $serviceLoadBalancerSourceRanges := coalesce .Values.readReplicas.service.loadBalancerSourceRanges .Values.service.loadBalancerSourceRanges -}} +{{- $serviceClusterIP := coalesce .Values.readReplicas.service.clusterIP .Values.service.clusterIP -}} +{{- $serviceNodePort := coalesce .Values.readReplicas.service.nodePort .Values.service.nodePort -}} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }}-read + labels: + {{- include "common.labels.standard" . | nindent 4 }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- if $serviceAnnotations }} + {{- include "common.tplvalues.render" (dict "value" $serviceAnnotations "context" $) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + type: {{ $serviceType }} + {{- if and $serviceLoadBalancerIP (eq $serviceType "LoadBalancer") }} + loadBalancerIP: {{ $serviceLoadBalancerIP }} + {{- end }} + {{- if and (eq $serviceType "LoadBalancer") $serviceLoadBalancerSourceRanges }} + loadBalancerSourceRanges: {{- include "common.tplvalues.render" (dict "value" $serviceLoadBalancerSourceRanges "context" $) | nindent 4 }} + {{- end }} + {{- if and (eq $serviceType "ClusterIP") $serviceClusterIP }} + clusterIP: {{ $serviceClusterIP }} + {{- end }} + ports: + - name: tcp-postgresql + port: {{ template "postgresql.port" . }} + targetPort: tcp-postgresql + {{- if $serviceNodePort }} + nodePort: {{ $serviceNodePort }} + {{- end }} + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} + role: read +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc.yaml new file mode 100644 index 0000000000..a29431b6a4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/templates/svc.yaml @@ -0,0 +1,41 @@ +{{- $serviceAnnotations := coalesce .Values.primary.service.annotations .Values.service.annotations -}} +{{- $serviceType := coalesce .Values.primary.service.type .Values.service.type -}} +{{- $serviceLoadBalancerIP := coalesce .Values.primary.service.loadBalancerIP .Values.service.loadBalancerIP -}} +{{- $serviceLoadBalancerSourceRanges := coalesce .Values.primary.service.loadBalancerSourceRanges .Values.service.loadBalancerSourceRanges -}} +{{- $serviceClusterIP := coalesce .Values.primary.service.clusterIP .Values.service.clusterIP -}} +{{- $serviceNodePort := coalesce .Values.primary.service.nodePort .Values.service.nodePort -}} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- if $serviceAnnotations }} + {{- include "common.tplvalues.render" (dict "value" $serviceAnnotations "context" $) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + type: {{ $serviceType }} + {{- if and $serviceLoadBalancerIP (eq $serviceType "LoadBalancer") }} + loadBalancerIP: {{ $serviceLoadBalancerIP }} + {{- end }} + {{- if and (eq $serviceType "LoadBalancer") $serviceLoadBalancerSourceRanges }} + loadBalancerSourceRanges: {{- include "common.tplvalues.render" (dict "value" $serviceLoadBalancerSourceRanges "context" $) | nindent 4 }} + {{- end }} + {{- if and (eq $serviceType "ClusterIP") $serviceClusterIP }} + clusterIP: {{ $serviceClusterIP }} + {{- end }} + ports: + - name: tcp-postgresql + port: {{ template "postgresql.port" . }} + targetPort: tcp-postgresql + {{- if $serviceNodePort }} + nodePort: {{ $serviceNodePort }} + {{- end }} + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} + role: primary diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.schema.json b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.schema.json new file mode 100644 index 0000000000..66a2a9dd06 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.schema.json @@ -0,0 +1,103 @@ +{ + "$schema": "http://json-schema.org/schema#", + "type": "object", + "properties": { + "postgresqlUsername": { + "type": "string", + "title": "Admin user", + "form": true + }, + "postgresqlPassword": { + "type": "string", + "title": "Password", + "form": true + }, + "persistence": { + "type": "object", + "properties": { + "size": { + "type": "string", + "title": "Persistent Volume Size", + "form": true, + "render": "slider", + "sliderMin": 1, + "sliderMax": 100, + "sliderUnit": "Gi" + } + } + }, + "resources": { + "type": "object", + "title": "Required Resources", + "description": "Configure resource requests", + "form": true, + "properties": { + "requests": { + "type": "object", + "properties": { + "memory": { + "type": "string", + "form": true, + "render": "slider", + "title": "Memory Request", + "sliderMin": 10, + "sliderMax": 2048, + "sliderUnit": "Mi" + }, + "cpu": { + "type": "string", + "form": true, + "render": "slider", + "title": "CPU Request", + "sliderMin": 10, + "sliderMax": 2000, + "sliderUnit": "m" + } + } + } + } + }, + "replication": { + "type": "object", + "form": true, + "title": "Replication Details", + "properties": { + "enabled": { + "type": "boolean", + "title": "Enable Replication", + "form": true + }, + "readReplicas": { + "type": "integer", + "title": "read Replicas", + "form": true, + "hidden": { + "value": false, + "path": "replication/enabled" + } + } + } + }, + "volumePermissions": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "form": true, + "title": "Enable Init Containers", + "description": "Change the owner of the persist volume mountpoint to RunAsUser:fsGroup" + } + } + }, + "metrics": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "title": "Configure metrics exporter", + "form": true + } + } + } + } +} diff --git a/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.yaml b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.yaml new file mode 100644 index 0000000000..82ce092344 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/charts/postgresql/values.yaml @@ -0,0 +1,824 @@ +## Global Docker image parameters +## Please, note that this will override the image parameters, including dependencies, configured to use the global value +## Current available global Docker image parameters: imageRegistry and imagePullSecrets +## +global: + postgresql: {} +# imageRegistry: myRegistryName +# imagePullSecrets: +# - myRegistryKeySecretName +# storageClass: myStorageClass + +## Bitnami PostgreSQL image version +## ref: https://hub.docker.com/r/bitnami/postgresql/tags/ +## +image: + registry: docker.io + repository: bitnami/postgresql + tag: 11.11.0-debian-10-r71 + ## Specify a imagePullPolicy + ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' + ## ref: http://kubernetes.io/docs/user-guide/images/#pre-pulling-images + ## + 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 + + ## Set to true if you would like to see extra information on logs + ## It turns BASH and/or NAMI debugging in the image + ## + debug: false + +## String to partially override common.names.fullname template (will maintain the release name) +## +# nameOverride: + +## String to fully override common.names.fullname template +## +# fullnameOverride: + +## +## Init containers parameters: +## volumePermissions: Change the owner of the persist volume mountpoint to RunAsUser:fsGroup +## +volumePermissions: + enabled: false + image: + registry: docker.io + repository: bitnami/bitnami-shell + tag: "10" + ## Specify a imagePullPolicy + ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' + ## ref: http://kubernetes.io/docs/user-guide/images/#pre-pulling-images + ## + pullPolicy: Always + ## 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 + ## Init container Security Context + ## Note: the chown of the data folder is done to securityContext.runAsUser + ## and not the below volumePermissions.securityContext.runAsUser + ## When runAsUser is set to special value "auto", init container will try to chwon the + ## data folder to autodetermined user&group, using commands: `id -u`:`id -G | cut -d" " -f2` + ## "auto" is especially useful for OpenShift which has scc with dynamic userids (and 0 is not allowed). + ## You may want to use this volumePermissions.securityContext.runAsUser="auto" in combination with + ## pod securityContext.enabled=false and shmVolume.chmod.enabled=false + ## + securityContext: + runAsUser: 0 + +## Use an alternate scheduler, e.g. "stork". +## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ +## +# schedulerName: + +## Pod Security Context +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ +## +securityContext: + enabled: true + fsGroup: 1001 + +## Container Security Context +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ +## +containerSecurityContext: + enabled: true + runAsUser: 1001 + +## Pod Service Account +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/ +## +serviceAccount: + enabled: false + ## Name of an already existing service account. Setting this value disables the automatic service account creation. + # name: + +## Pod Security Policy +## ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ +## +psp: + create: false + +## Creates role for ServiceAccount +## Required for PSP +## +rbac: + create: false + +replication: + enabled: false + user: repl_user + password: repl_password + readReplicas: 1 + ## Set synchronous commit mode: on, off, remote_apply, remote_write and local + ## ref: https://www.postgresql.org/docs/9.6/runtime-config-wal.html#GUC-WAL-LEVEL + synchronousCommit: 'off' + ## From the number of `readReplicas` defined above, set the number of those that will have synchronous replication + ## NOTE: It cannot be > readReplicas + numSynchronousReplicas: 0 + ## Replication Cluster application name. Useful for defining multiple replication policies + ## + applicationName: my_application + +## PostgreSQL admin password (used when `postgresqlUsername` is not `postgres`) +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#creating-a-database-user-on-first-run (see note!) +# postgresqlPostgresPassword: + +## PostgreSQL user (has superuser privileges if username is `postgres`) +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#setting-the-root-password-on-first-run +## +postgresqlUsername: postgres + +## PostgreSQL password +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#setting-the-root-password-on-first-run +## +# postgresqlPassword: + +## PostgreSQL password using existing secret +## existingSecret: secret +## + +## Mount PostgreSQL secret as a file instead of passing environment variable +# usePasswordFile: false + +## Create a database +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#creating-a-database-on-first-run +## +# postgresqlDatabase: + +## PostgreSQL data dir +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md +## +postgresqlDataDir: /bitnami/postgresql/data + +## An array to add extra environment variables +## For example: +## extraEnv: +## - name: FOO +## value: "bar" +## +# extraEnv: +extraEnv: [] + +## Name of a ConfigMap containing extra env vars +## +# extraEnvVarsCM: + +## Specify extra initdb args +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md +## +# postgresqlInitdbArgs: + +## Specify a custom location for the PostgreSQL transaction log +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md +## +# postgresqlInitdbWalDir: + +## PostgreSQL configuration +## Specify runtime configuration parameters as a dict, using camelCase, e.g. +## {"sharedBuffers": "500MB"} +## Alternatively, you can put your postgresql.conf under the files/ directory +## ref: https://www.postgresql.org/docs/current/static/runtime-config.html +## +# postgresqlConfiguration: + +## PostgreSQL extended configuration +## As above, but _appended_ to the main configuration +## Alternatively, you can put your *.conf under the files/conf.d/ directory +## https://github.com/bitnami/bitnami-docker-postgresql#allow-settings-to-be-loaded-from-files-other-than-the-default-postgresqlconf +## +# postgresqlExtendedConf: + +## Configure current cluster's primary server to be the standby server in other cluster. +## This will allow cross cluster replication and provide cross cluster high availability. +## You will need to configure pgHbaConfiguration if you want to enable this feature with local cluster replication enabled. +## +primaryAsStandBy: + enabled: false + # primaryHost: + # primaryPort: + +## PostgreSQL client authentication configuration +## Specify content for pg_hba.conf +## Default: do not create pg_hba.conf +## Alternatively, you can put your pg_hba.conf under the files/ directory +# pgHbaConfiguration: |- +# local all all trust +# host all all localhost trust +# host mydatabase mysuser 192.168.0.0/24 md5 + +## ConfigMap with PostgreSQL configuration +## NOTE: This will override postgresqlConfiguration and pgHbaConfiguration +# configurationConfigMap: + +## ConfigMap with PostgreSQL extended configuration +# extendedConfConfigMap: + +## initdb scripts +## Specify dictionary of scripts to be run at first boot +## Alternatively, you can put your scripts under the files/docker-entrypoint-initdb.d directory +## +# initdbScripts: +# my_init_script.sh: | +# #!/bin/sh +# echo "Do something." + +## ConfigMap with scripts to be run at first boot +## NOTE: This will override initdbScripts +# initdbScriptsConfigMap: + +## Secret with scripts to be run at first boot (in case it contains sensitive information) +## NOTE: This can work along initdbScripts or initdbScriptsConfigMap +# initdbScriptsSecret: + +## Specify the PostgreSQL username and password to execute the initdb scripts +# initdbUser: +# initdbPassword: + +## Audit settings +## https://github.com/bitnami/bitnami-docker-postgresql#auditing +## +audit: + ## Log client hostnames + ## + logHostname: false + ## Log connections to the server + ## + logConnections: false + ## Log disconnections + ## + logDisconnections: false + ## Operation to audit using pgAudit (default if not set) + ## + pgAuditLog: "" + ## Log catalog using pgAudit + ## + pgAuditLogCatalog: "off" + ## Log level for clients + ## + clientMinMessages: error + ## Template for log line prefix (default if not set) + ## + logLinePrefix: "" + ## Log timezone + ## + logTimezone: "" + +## Shared preload libraries +## +postgresqlSharedPreloadLibraries: "pgaudit" + +## Maximum total connections +## +postgresqlMaxConnections: + +## Maximum connections for the postgres user +## +postgresqlPostgresConnectionLimit: + +## Maximum connections for the created user +## +postgresqlDbUserConnectionLimit: + +## TCP keepalives interval +## +postgresqlTcpKeepalivesInterval: + +## TCP keepalives idle +## +postgresqlTcpKeepalivesIdle: + +## TCP keepalives count +## +postgresqlTcpKeepalivesCount: + +## Statement timeout +## +postgresqlStatementTimeout: + +## Remove pg_hba.conf lines with the following comma-separated patterns +## (cannot be used with custom pg_hba.conf) +## +postgresqlPghbaRemoveFilters: + +## Optional duration in seconds the pod needs to terminate gracefully. +## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods +## +# terminationGracePeriodSeconds: 30 + +## LDAP configuration +## +ldap: + enabled: false + url: '' + server: '' + port: '' + prefix: '' + suffix: '' + baseDN: '' + bindDN: '' + bind_password: + search_attr: '' + search_filter: '' + scheme: '' + tls: {} + +## PostgreSQL service configuration +## +service: + ## PosgresSQL service type + ## + type: ClusterIP + # clusterIP: None + port: 5432 + + ## Specify the nodePort value for the LoadBalancer and NodePort service types. + ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport + ## + # nodePort: + + ## Provide any additional annotations which may be required. Evaluated as a template. + ## + annotations: {} + ## Set the LoadBalancer service type to internal only. + ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer + ## + # loadBalancerIP: + ## Load Balancer sources. Evaluated as a template. + ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service + ## + # loadBalancerSourceRanges: + # - 10.10.10.0/24 + +## Start primary and read(s) pod(s) without limitations on shm memory. +## By default docker and containerd (and possibly other container runtimes) +## limit `/dev/shm` to `64M` (see e.g. the +## [docker issue](https://github.com/docker-library/postgres/issues/416) and the +## [containerd issue](https://github.com/containerd/containerd/issues/3654), +## which could be not enough if PostgreSQL uses parallel workers heavily. +## +shmVolume: + ## Set `shmVolume.enabled` to `true` to mount a new tmpfs volume to remove + ## this limitation. + ## + enabled: true + ## Set to `true` to `chmod 777 /dev/shm` on a initContainer. + ## This option is ignored if `volumePermissions.enabled` is `false` + ## + chmod: + enabled: true + +## PostgreSQL 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) +## +persistence: + enabled: true + ## A manually managed Persistent Volume and Claim + ## If defined, PVC must be created manually before volume will be bound + ## The value is evaluated as a template, so, for example, the name can depend on .Release or .Chart + ## + # existingClaim: + + ## The path the volume will be mounted at, useful when using different + ## PostgreSQL images. + ## + mountPath: /bitnami/postgresql + + ## The subdirectory of the volume to mount to, useful in dev environments + ## and one PV for multiple services. + ## + subPath: '' + + # storageClass: "-" + accessModes: + - ReadWriteOnce + size: 8Gi + annotations: {} + ## selector can be used to match an existing PersistentVolume + ## selector: + ## matchLabels: + ## app: my-app + selector: {} + +## updateStrategy for PostgreSQL StatefulSet and its reads StatefulSets +## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies +## +updateStrategy: + type: RollingUpdate + +## +## PostgreSQL Primary parameters +## +primary: + ## PostgreSQL Primary pod affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAffinityPreset: "" + + ## PostgreSQL Primary pod anti-affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAntiAffinityPreset: soft + + ## PostgreSQL Primary node affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity + ## Allowed values: soft, hard + ## + nodeAffinityPreset: + ## Node affinity type + ## Allowed values: soft, hard + type: "" + ## Node label key to match + ## E.g. + ## key: "kubernetes.io/e2e-az-name" + ## + key: "" + ## Node label values to match + ## E.g. + ## values: + ## - e2e-az1 + ## - e2e-az2 + ## + values: [] + + ## Affinity for PostgreSQL primary pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity + ## Note: primary.podAffinityPreset, primary.podAntiAffinityPreset, and primary.nodeAffinityPreset will be ignored when it's set + ## + affinity: {} + + ## Node labels for PostgreSQL primary pods assignment + ## ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Tolerations for PostgreSQL primary pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + ## + tolerations: [] + + labels: {} + annotations: {} + podLabels: {} + podAnnotations: {} + priorityClassName: '' + ## Extra init containers + ## Example + ## + ## extraInitContainers: + ## - name: do-something + ## image: busybox + ## command: ['do', 'something'] + ## + extraInitContainers: [] + + ## Additional PostgreSQL primary Volume mounts + ## + extraVolumeMounts: [] + ## Additional PostgreSQL primary Volumes + ## + extraVolumes: [] + ## Add sidecars to the pod + ## + ## For example: + ## sidecars: + ## - name: your-image-name + ## image: your-image + ## imagePullPolicy: Always + ## ports: + ## - name: portname + ## containerPort: 1234 + ## + sidecars: [] + + ## Override the service configuration for primary + ## + service: {} + # type: + # nodePort: + # clusterIP: + +## +## PostgreSQL read only replica parameters +## +readReplicas: + ## PostgreSQL read only pod affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAffinityPreset: "" + + ## PostgreSQL read only pod anti-affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAntiAffinityPreset: soft + + ## PostgreSQL read only node affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity + ## Allowed values: soft, hard + ## + nodeAffinityPreset: + ## Node affinity type + ## Allowed values: soft, hard + type: "" + ## Node label key to match + ## E.g. + ## key: "kubernetes.io/e2e-az-name" + ## + key: "" + ## Node label values to match + ## E.g. + ## values: + ## - e2e-az1 + ## - e2e-az2 + ## + values: [] + + ## Affinity for PostgreSQL read only pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity + ## Note: readReplicas.podAffinityPreset, readReplicas.podAntiAffinityPreset, and readReplicas.nodeAffinityPreset will be ignored when it's set + ## + affinity: {} + + ## Node labels for PostgreSQL read only pods assignment + ## ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Tolerations for PostgreSQL read only pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + ## + tolerations: [] + labels: {} + annotations: {} + podLabels: {} + podAnnotations: {} + priorityClassName: '' + + ## Extra init containers + ## Example + ## + ## extraInitContainers: + ## - name: do-something + ## image: busybox + ## command: ['do', 'something'] + ## + extraInitContainers: [] + + ## Additional PostgreSQL read replicas Volume mounts + ## + extraVolumeMounts: [] + + ## Additional PostgreSQL read replicas Volumes + ## + extraVolumes: [] + + ## Add sidecars to the pod + ## + ## For example: + ## sidecars: + ## - name: your-image-name + ## image: your-image + ## imagePullPolicy: Always + ## ports: + ## - name: portname + ## containerPort: 1234 + ## + sidecars: [] + + ## Override the service configuration for read + ## + service: {} + # type: + # nodePort: + # clusterIP: + + ## Whether to enable PostgreSQL read replicas data Persistent + ## + persistence: + enabled: true + + # Override the resource configuration for read replicas + resources: {} + # requests: + # memory: 256Mi + # cpu: 250m + +## Configure resource requests and limits +## ref: http://kubernetes.io/docs/user-guide/compute-resources/ +## +resources: + requests: + memory: 256Mi + cpu: 250m + +## Add annotations to all the deployed resources +## +commonAnnotations: {} + +networkPolicy: + ## Enable creation of NetworkPolicy resources. Only Ingress traffic is filtered for now. + ## + enabled: false + + ## The Policy model to apply. When set to false, only pods with the correct + ## client label will have network access to the port PostgreSQL is listening + ## on. When true, PostgreSQL will accept connections from any source + ## (with the correct destination port). + ## + allowExternal: true + + ## 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 DB. + ## But sometimes, we want the DB 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: {} + +## Configure extra options for startup, liveness and readiness probes +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes +## +startupProbe: + enabled: false + initialDelaySeconds: 30 + periodSeconds: 15 + timeoutSeconds: 5 + failureThreshold: 10 + successThreshold: 1 + +livenessProbe: + enabled: true + initialDelaySeconds: 30 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + +readinessProbe: + enabled: true + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + +## Custom Startup probe +## +customStartupProbe: {} + +## Custom Liveness probe +## +customLivenessProbe: {} + +## Custom Rediness probe +## +customReadinessProbe: {} + +## +## TLS configuration +## +tls: + # Enable TLS traffic + enabled: false + # + # Whether to use the server's TLS cipher preferences rather than the client's. + preferServerCiphers: true + # + # Name of the Secret that contains the certificates + certificatesSecret: '' + # + # Certificate filename + certFilename: '' + # + # Certificate Key filename + certKeyFilename: '' + # + # CA Certificate filename + # If provided, PostgreSQL will authenticate TLS/SSL clients by requesting them a certificate + # ref: https://www.postgresql.org/docs/9.6/auth-methods.html + certCAFilename: + # + # File containing a Certificate Revocation List + crlFilename: + +## Configure metrics exporter +## +metrics: + enabled: false + # resources: {} + service: + type: ClusterIP + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '9187' + loadBalancerIP: + serviceMonitor: + enabled: false + additionalLabels: {} + # namespace: monitoring + # interval: 30s + # scrapeTimeout: 10s + ## Custom PrometheusRule to be defined + ## The value is evaluated as a template, so, for example, the value can depend on .Release or .Chart + ## ref: https://github.com/coreos/prometheus-operator#customresourcedefinitions + ## + prometheusRule: + enabled: false + additionalLabels: {} + namespace: '' + ## These are just examples rules, please adapt them to your needs. + ## Make sure to constraint the rules to the current postgresql service. + ## rules: + ## - alert: HugeReplicationLag + ## expr: pg_replication_lag{service="{{ template "common.names.fullname" . }}-metrics"} / 3600 > 1 + ## for: 1m + ## labels: + ## severity: critical + ## annotations: + ## description: replication for {{ template "common.names.fullname" . }} PostgreSQL is lagging by {{ "{{ $value }}" }} hour(s). + ## summary: PostgreSQL replication is lagging by {{ "{{ $value }}" }} hour(s). + ## + rules: [] + + image: + registry: docker.io + repository: bitnami/postgres-exporter + tag: 0.9.0-debian-10-r43 + 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 + ## Define additional custom metrics + ## ref: https://github.com/wrouesnel/postgres_exporter#adding-new-metrics-via-a-config-file + # customMetrics: + # pg_database: + # query: "SELECT d.datname AS name, CASE WHEN pg_catalog.has_database_privilege(d.datname, 'CONNECT') THEN pg_catalog.pg_database_size(d.datname) ELSE 0 END AS size_bytes FROM pg_catalog.pg_database d where datname not in ('template0', 'template1', 'postgres')" + # metrics: + # - name: + # usage: "LABEL" + # description: "Name of the database" + # - size_bytes: + # usage: "GAUGE" + # description: "Size of the database in bytes" + # + ## An array to add extra env vars to configure postgres-exporter + ## see: https://github.com/wrouesnel/postgres_exporter#environment-variables + ## For example: + # extraEnvVars: + # - name: PG_EXPORTER_DISABLE_DEFAULT_METRICS + # value: "true" + extraEnvVars: {} + + ## Pod Security Context + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ + ## + securityContext: + enabled: false + runAsUser: 1001 + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes) + ## Configure extra options for liveness and readiness probes + ## + livenessProbe: + enabled: true + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + + readinessProbe: + enabled: true + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + +## Array with extra yaml to deploy with the chart. Evaluated as a template +## +extraDeploy: [] diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/access-tls-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/access-tls-values.yaml new file mode 100644 index 0000000000..27e24d3468 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/access-tls-values.yaml @@ -0,0 +1,34 @@ +databaseUpgradeReady: true +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +access: + accessConfig: + security: + tls: true + resetAccessCAKeys: true diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/default-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/default-values.yaml new file mode 100644 index 0000000000..020f523352 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/default-values.yaml @@ -0,0 +1,32 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true +## This is an exception here because HA needs masterKey to connect with other node members and it is commented in values to support 6.x to 7.x Migration +## Please refer https://github.com/jfrog/charts/blob/master/stable/artifactory-ha/README.md#special-upgrade-notes-1 +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/global-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/global-values.yaml new file mode 100644 index 0000000000..0987e17ca7 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/global-values.yaml @@ -0,0 +1,255 @@ +databaseUpgradeReady: true +artifactory: + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + customInitContainersBegin: | + - name: "custom-init-begin-local" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in local" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: volume + customInitContainers: | + - name: "custom-init-local" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in local" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: volume + # Add custom volumes + customVolumes: | + - name: custom-script-local + emptyDir: + sizeLimit: 100Mi + # Add custom volumesMounts + customVolumeMounts: | + - name: custom-script-local + mountPath: "/scriptslocal" + # Add custom sidecar containers + customSidecarContainers: | + - name: "sidecar-list-local" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - NET_RAW + command: ["sh","-c","echo 'Sidecar is running in local' >> /scriptslocal/sidecarlocal.txt; cat /scriptslocal/sidecarlocal.txt; while true; do sleep 30; done"] + volumeMounts: + - mountPath: "/scriptslocal" + name: custom-script-local + resources: + requests: + memory: "32Mi" + cpu: "50m" + limits: + memory: "128Mi" + cpu: "100m" + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +global: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + customInitContainersBegin: | + - name: "custom-init-begin-global" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in global" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: volume + customInitContainers: | + - name: "custom-init-global" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in global" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: volume + # Add custom volumes + customVolumes: | + - name: custom-script-global + emptyDir: + sizeLimit: 100Mi + # Add custom volumesMounts + customVolumeMounts: | + - name: custom-script-global + mountPath: "/scripts" + # Add custom sidecar containers + customSidecarContainers: | + - name: "sidecar-list-global" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - NET_RAW + command: ["sh","-c","echo 'Sidecar is running in global' >> /scripts/sidecarglobal.txt; cat /scripts/sidecarglobal.txt; while true; do sleep 30; done"] + volumeMounts: + - mountPath: "/scripts" + name: custom-script-global + resources: + requests: + memory: "32Mi" + cpu: "50m" + limits: + memory: "128Mi" + cpu: "100m" + +nginx: + customInitContainers: | + - name: "custom-init-begin-nginx" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in nginx" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: custom-script-local + customSidecarContainers: | + - name: "sidecar-list-nginx" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - NET_RAW + command: ["sh","-c","echo 'Sidecar is running in local' >> /scriptslocal/sidecarlocal.txt; cat /scriptslocal/sidecarlocal.txt; while true; do sleep 30; done"] + volumeMounts: + - mountPath: "/scriptslocal" + name: custom-script-local + resources: + requests: + memory: "32Mi" + cpu: "50m" + limits: + memory: "128Mi" + cpu: "100m" + # Add custom volumes + customVolumes: | + - name: custom-script-local + emptyDir: + sizeLimit: 100Mi + + artifactoryConf: | + {{- if .Values.nginx.https.enabled }} + ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3; + ssl_certificate {{ .Values.nginx.persistence.mountPath }}/ssl/tls.crt; + ssl_certificate_key {{ .Values.nginx.persistence.mountPath }}/ssl/tls.key; + ssl_session_cache shared:SSL:1m; + ssl_prefer_server_ciphers on; + {{- end }} + ## server configuration + server { + listen 8088; + {{- if .Values.nginx.internalPortHttps }} + listen {{ .Values.nginx.internalPortHttps }} ssl; + {{- else -}} + {{- if .Values.nginx.https.enabled }} + listen {{ .Values.nginx.https.internalPort }} ssl; + {{- end }} + {{- end }} + {{- if .Values.nginx.internalPortHttp }} + listen {{ .Values.nginx.internalPortHttp }}; + {{- else -}} + {{- if .Values.nginx.http.enabled }} + listen {{ .Values.nginx.http.internalPort }}; + {{- end }} + {{- end }} + server_name ~(?.+)\.{{ include "artifactory-ha.fullname" . }} {{ include "artifactory-ha.fullname" . }} + {{- range .Values.ingress.hosts -}} + {{- if contains "." . -}} + {{ "" | indent 0 }} ~(?.+)\.{{ . }} + {{- end -}} + {{- end -}}; + if ($http_x_forwarded_proto = '') { + set $http_x_forwarded_proto $scheme; + } + ## Application specific logs + ## access_log /var/log/nginx/artifactory-access.log timing; + ## error_log /var/log/nginx/artifactory-error.log; + rewrite ^/artifactory/?$ / redirect; + if ( $repo != "" ) { + rewrite ^/(v1|v2)/(.*) /artifactory/api/docker/$repo/$1/$2 break; + } + chunked_transfer_encoding on; + client_max_body_size 0; + + location / { + proxy_read_timeout 900; + proxy_pass_header Server; + proxy_cookie_path ~*^/.* /; + proxy_pass {{ include "artifactory-ha.scheme" . }}://{{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.externalPort }}/; + {{- if .Values.nginx.service.ssloffload}} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host; + {{- else }} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host:$server_port; + proxy_set_header X-Forwarded-Port $server_port; + {{- end }} + proxy_set_header X-Forwarded-Proto $http_x_forwarded_proto; + proxy_set_header Host $http_host; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always; + + location /artifactory/ { + if ( $request_uri ~ ^/artifactory/(.*)$ ) { + proxy_pass {{ include "artifactory-ha.scheme" . }}://{{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/$1; + } + proxy_pass {{ include "artifactory-ha.scheme" . }}://{{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/; + } + } + } + + ## A list of custom ports to expose on the NGINX pod. Follows the conventional Kubernetes yaml syntax for container ports. + customPorts: + - containerPort: 8088 + name: http2 + service: + ## A list of custom ports to expose through the Ingress controller service. Follows the conventional Kubernetes yaml syntax for service ports. + customPorts: + - port: 8088 + targetPort: 8088 + protocol: TCP + name: http2 diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/large-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/large-values.yaml new file mode 100644 index 0000000000..153307aa26 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/large-values.yaml @@ -0,0 +1,85 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + database: + maxOpenConnections: 150 + tomcat: + connector: + maxThreads: 300 + primary: + replicaCount: 4 + resources: + requests: + memory: "6Gi" + cpu: "2" + limits: + memory: "10Gi" + cpu: "8" + javaOpts: + xms: "8g" + xmx: "10g" +access: + database: + maxOpenConnections: 150 + tomcat: + connector: + maxThreads: 100 +router: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + database: + maxOpenConnections: 150 + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +jfconnect: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/loggers-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/loggers-values.yaml new file mode 100644 index 0000000000..03c94be953 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/loggers-values.yaml @@ -0,0 +1,43 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + + loggers: + - access-audit.log + - access-request.log + - access-security-audit.log + - access-service.log + - artifactory-access.log + - artifactory-event.log + - artifactory-import-export.log + - artifactory-request.log + - artifactory-service.log + - frontend-request.log + - frontend-service.log + - metadata-request.log + - metadata-service.log + - router-request.log + - router-service.log + - router-traefik.log + + catalinaLoggers: + - tomcat-catalina.log + - tomcat-localhost.log diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/medium-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/medium-values.yaml new file mode 100644 index 0000000000..115e7d4602 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/medium-values.yaml @@ -0,0 +1,85 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + database: + maxOpenConnections: 100 + tomcat: + connector: + maxThreads: 200 + primary: + replicaCount: 3 + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "8Gi" + cpu: "6" + javaOpts: + xms: "6g" + xmx: "8g" +access: + database: + maxOpenConnections: 100 + tomcat: + connector: + maxThreads: 50 +router: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + database: + maxOpenConnections: 100 + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +jfconnect: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/migration-disabled-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/migration-disabled-values.yaml new file mode 100644 index 0000000000..44895a3731 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/migration-disabled-values.yaml @@ -0,0 +1,31 @@ +databaseUpgradeReady: true +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + migration: + enabled: false + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/nginx-autoreload-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/nginx-autoreload-values.yaml new file mode 100644 index 0000000000..a6f4e8001f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/nginx-autoreload-values.yaml @@ -0,0 +1,53 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true +## This is an exception here because HA needs masterKey to connect with other node members and it is commented in values to support 6.x to 7.x Migration +## Please refer https://github.com/jfrog/charts/blob/master/stable/artifactory-ha/README.md#special-upgrade-notes-1 +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false + +nginx: + customVolumes: | + - name: scripts + configMap: + name: {{ template "artifactory-ha.fullname" . }}-nginx-scripts + defaultMode: 0550 + customVolumeMounts: | + - name: scripts + mountPath: /var/opt/jfrog/nginx/scripts/ + customCommand: + - /bin/sh + - -c + - | + # watch for configmap changes + /sbin/inotifyd /var/opt/jfrog/nginx/scripts/configreloader.sh {{ .Values.nginx.persistence.mountPath -}}/conf.d:n & + {{ if .Values.nginx.https.enabled -}} + # watch for tls secret changes + /sbin/inotifyd /var/opt/jfrog/nginx/scripts/configreloader.sh {{ .Values.nginx.persistence.mountPath -}}/ssl:n & + {{ end -}} + nginx -g 'daemon off;' diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-access-tls-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-access-tls-values.yaml new file mode 100644 index 0000000000..6f3b13cb14 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-access-tls-values.yaml @@ -0,0 +1,106 @@ +databaseUpgradeReady: true +artifactory: + replicaCount: 3 + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + +access: + accessConfig: + security: + tls: true + resetAccessCAKeys: true + +postgresql: + postgresqlPassword: password + postgresqlExtendedConf: + maxConnections: 102 + persistence: + enabled: false + +rbac: + create: true +serviceAccount: + create: true + automountServiceAccountToken: true + +ingress: + enabled: true + className: "testclass" + hosts: + - demonow.xyz +nginx: + enabled: false +jfconnect: + enabled: true + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +mc: + enabled: true +splitServicesToContainers: true + +router: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-values.yaml new file mode 100644 index 0000000000..87832a5051 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/rtsplit-values.yaml @@ -0,0 +1,155 @@ +databaseUpgradeReady: true +artifactory: + replicaCount: 3 + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + + # Add lifecycle hooks for artifactory container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the artifactory postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the artifactory postStart handler >> /tmp/message"] + +postgresql: + postgresqlPassword: password + postgresqlExtendedConf: + maxConnections: 102 + persistence: + enabled: false + +rbac: + create: true +serviceAccount: + create: true + automountServiceAccountToken: true + +ingress: + enabled: true + className: "testclass" + hosts: + - demonow.xyz +nginx: + enabled: false +jfconnect: + enabled: true + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + # Add lifecycle hooks for jfconect container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the jfconnect postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the jfconnect postStart handler >> /tmp/message"] + +mc: + enabled: true +splitServicesToContainers: true + +router: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + # Add lifecycle hooks for router container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the router postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the router postStart handler >> /tmp/message"] +frontend: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + # Add lifecycle hooks for frontend container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the frontend postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the frontend postStart handler >> /tmp/message"] +metadata: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the metadata postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the metadata postStart handler >> /tmp/message"] +event: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the event postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the event postStart handler >> /tmp/message"] +observability: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the observability postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the observability postStart handler >> /tmp/message"] diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/small-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/small-values.yaml new file mode 100644 index 0000000000..b4557289ef --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/small-values.yaml @@ -0,0 +1,87 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + database: + maxOpenConnections: 80 + tomcat: + connector: + maxThreads: 200 + primary: + replicaCount: 1 + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "6g" + node: + replicaCount: 2 +access: + database: + maxOpenConnections: 80 + tomcat: + connector: + maxThreads: 50 +router: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + database: + maxOpenConnections: 80 + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +jfconnect: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-ha/107.90.14/ci/test-values.yaml b/charts/jfrog/artifactory-ha/107.90.14/ci/test-values.yaml new file mode 100644 index 0000000000..8bbbb5b3e5 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/ci/test-values.yaml @@ -0,0 +1,85 @@ +databaseUpgradeReady: true +artifactory: + metrics: + enabled: true + podSecurityContext: + fsGroupChangePolicy: "OnRootMismatch" + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + unifiedSecretInstallation: false + persistence: + enabled: false + primary: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + node: + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + statefulset: + annotations: + artifactory: test + +postgresql: + postgresqlPassword: "password" + postgresqlExtendedConf: + maxConnections: "102" + persistence: + enabled: false +rbac: + create: true +serviceAccount: + create: true + automountServiceAccountToken: true +ingress: + enabled: true + className: "testclass" + hosts: + - demonow.xyz +nginx: + enabled: false + +jfconnect: + enabled: false + +## filebeat sidecar +filebeat: + enabled: true + filebeatYml: | + logging.level: info + path.data: {{ .Values.artifactory.persistence.mountPath }}/log/filebeat + name: artifactory-filebeat + queue.spool: + file: + permissions: 0760 + filebeat.inputs: + - type: log + enabled: true + close_eof: ${CLOSE:false} + paths: + - {{ .Values.artifactory.persistence.mountPath }}/log/*.log + fields: + service: "jfrt" + log_type: "artifactory" + output.file: + path: "/tmp/filebeat" + filename: filebeat + readinessProbe: + exec: + command: + - sh + - -c + - | + #!/usr/bin/env bash -e + curl --fail 127.0.0.1:5066 diff --git a/charts/jfrog/artifactory-ha/107.90.14/files/binarystore.xml b/charts/jfrog/artifactory-ha/107.90.14/files/binarystore.xml new file mode 100644 index 0000000000..0e7bc5af0f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/files/binarystore.xml @@ -0,0 +1,439 @@ +{{- if and (eq .Values.artifactory.persistence.type "nfs") (.Values.artifactory.haDataDir.enabled) }} + + + + + + + +{{- end }} +{{- if and (eq .Values.artifactory.persistence.type "nfs") (not .Values.artifactory.haDataDir.enabled) }} + + {{- if (.Values.artifactory.persistence.maxCacheSize) }} + + + + + + {{- else }} + + + + {{- end }} + + {{- if .Values.artifactory.persistence.maxCacheSize }} + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + {{- end }} + + + {{ .Values.artifactory.persistence.nfs.dataDir }}/filestore + + + +{{- end }} + +{{- if eq .Values.artifactory.persistence.type "file-system" }} + +{{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + + + + + + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) -}} + + {{- end }} + + + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + // Specify the read and write strategy and redundancy for the sharding binary provider + + roundRobin + percentageFreeSpace + 2 + + + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) -}} + //For each sub-provider (mount), specify the filestore location + + filestore{{ $sharedClaimNumber }} + + {{- end }} + +{{- else }} + + + + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + 2 + 2 + + + + + + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + + + shard-fs-1 + local + + + + + 30 + tester-remote1 + 10000 + remote + + + +{{- end }} +{{- end }} +{{- if or (eq .Values.artifactory.persistence.type "google-storage") (eq .Values.artifactory.persistence.type "google-storage-v2") (eq .Values.artifactory.persistence.type "google-storage-v2-direct") }} + + + {{- if or (eq .Values.artifactory.persistence.type "google-storage") (eq .Values.artifactory.persistence.type "google-storage-v2") }} + + + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + 2 + + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "google-storage-v2-direct" }} + + + + + + {{- end }} + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + {{- if or (eq .Values.artifactory.persistence.type "google-storage") (eq .Values.artifactory.persistence.type "google-storage-v2") }} + + local + + + + + 30 + 10000 + remote + + {{- end }} + + + + {{- if .Values.artifactory.persistence.googleStorage.useInstanceCredentials }} + true + {{- else }} + false + {{- end }} + {{ .Values.artifactory.persistence.googleStorage.enableSignedUrlRedirect }} + google-cloud-storage + {{ .Values.artifactory.persistence.googleStorage.endpoint }} + {{ .Values.artifactory.persistence.googleStorage.httpsOnly }} + {{ .Values.artifactory.persistence.googleStorage.bucketName }} + {{ .Values.artifactory.persistence.googleStorage.path }} + {{ .Values.artifactory.persistence.googleStorage.bucketExists }} + + +{{- end }} +{{- if or (eq .Values.artifactory.persistence.type "aws-s3-v3") (eq .Values.artifactory.persistence.type "s3-storage-v3-direct") (eq .Values.artifactory.persistence.type "s3-storage-v3-archive") }} + + + {{- if eq .Values.artifactory.persistence.type "aws-s3-v3" }} + + + + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "s3-storage-v3-direct" }} + + + + + + {{- else if eq .Values.artifactory.persistence.type "s3-storage-v3-archive" }} + + + + + + + {{- end }} + + {{- if eq .Values.artifactory.persistence.type "aws-s3-v3" }} + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + + + + + remote + + + + local + + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + {{- end }} + + {{- if eq .Values.artifactory.persistence.type "s3-storage-v3-direct" }} + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + {{- end }} + + {{- with .Values.artifactory.persistence.awsS3V3 }} + + {{ .testConnection }} + {{- if .identity }} + {{ .identity }} + {{- end }} + {{- if .credential }} + {{ .credential }} + {{- end }} + {{ .region }} + {{ .bucketName }} + {{ .path }} + {{ .endpoint }} + {{- with .port }} + {{ . }} + {{- end }} + {{- with .useHttp }} + {{ . }} + {{- end }} + {{- with .maxConnections }} + {{ . }} + {{- end }} + {{- with .connectionTimeout }} + {{ . }} + {{- end }} + {{- with .socketTimeout }} + {{ . }} + {{- end }} + {{- with .kmsServerSideEncryptionKeyId }} + {{ . }} + {{- end }} + {{- with .kmsKeyRegion }} + {{ . }} + {{- end }} + {{- with .kmsCryptoMode }} + {{ . }} + {{- end }} + {{- if .useInstanceCredentials }} + true + {{- else }} + false + {{- end }} + {{ .usePresigning }} + {{ .signatureExpirySeconds }} + {{ .signedUrlExpirySeconds }} + {{- with .cloudFrontDomainName }} + {{ . }} + {{- end }} + {{- with .cloudFrontKeyPairId }} + {{ . }} + {{- end }} + {{- with .cloudFrontPrivateKey }} + {{ . }} + {{- end }} + {{- with .enableSignedUrlRedirect }} + {{ . }} + {{- end }} + {{- with .enablePathStyleAccess }} + {{ . }} + {{- end }} + {{- with .multiPartLimit }} + {{ . | int64 }} + {{- end }} + {{- with .multipartElementSize }} + {{ . | int64 }} + {{- end }} + + {{- end }} + +{{- end }} + +{{- if or (eq .Values.artifactory.persistence.type "azure-blob") (eq .Values.artifactory.persistence.type "azure-blob-storage-direct") }} + + + {{- if eq .Values.artifactory.persistence.type "azure-blob" }} + + + + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "azure-blob-storage-direct" }} + + + + + + {{- end }} + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + {{- if eq .Values.artifactory.persistence.type "azure-blob" }} + + + crossNetworkStrategy + crossNetworkStrategy + 2 + 1 + + + + + remote + + + + local + + {{- end }} + + + + {{ .Values.artifactory.persistence.azureBlob.accountName }} + {{ .Values.artifactory.persistence.azureBlob.accountKey }} + {{ .Values.artifactory.persistence.azureBlob.endpoint }} + {{ .Values.artifactory.persistence.azureBlob.containerName }} + {{ .Values.artifactory.persistence.azureBlob.multiPartLimit | int64 }} + {{ .Values.artifactory.persistence.azureBlob.multipartElementSize | int64 }} + {{ .Values.artifactory.persistence.azureBlob.testConnection }} + + +{{- end }} +{{- if eq .Values.artifactory.persistence.type "azure-blob-storage-v2-direct" -}} + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + {{ .Values.artifactory.persistence.azureBlob.accountName }} + {{ .Values.artifactory.persistence.azureBlob.accountKey }} + {{ .Values.artifactory.persistence.azureBlob.endpoint }} + {{ .Values.artifactory.persistence.azureBlob.containerName }} + {{ .Values.artifactory.persistence.azureBlob.multiPartLimit | int64 }} + {{ .Values.artifactory.persistence.azureBlob.multipartElementSize | int64 }} + {{ .Values.artifactory.persistence.azureBlob.testConnection }} + + +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/files/installer-info.json b/charts/jfrog/artifactory-ha/107.90.14/files/installer-info.json new file mode 100644 index 0000000000..cf6b020fb3 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/files/installer-info.json @@ -0,0 +1,32 @@ +{ + "productId": "Helm_artifactory-ha/{{ .Chart.Version }}", + "features": [ + { + "featureId": "Platform/{{ printf "%s-%s" "kubernetes" .Capabilities.KubeVersion.Version }}" + }, + { + "featureId": "Database/{{ .Values.database.type }}" + }, + { + "featureId": "PostgreSQL_Enabled/{{ .Values.postgresql.enabled }}" + }, + { + "featureId": "Nginx_Enabled/{{ .Values.nginx.enabled }}" + }, + { + "featureId": "ArtifactoryPersistence_Type/{{ .Values.artifactory.persistence.type }}" + }, + { + "featureId": "SplitServicesToContainers_Enabled/{{ .Values.splitServicesToContainers }}" + }, + { + "featureId": "UnifiedSecretInstallation_Enabled/{{ .Values.artifactory.unifiedSecretInstallation }}" + }, + { + "featureId": "Filebeat_Enabled/{{ .Values.filebeat.enabled }}" + }, + { + "featureId": "ReplicaCount/{{ add .Values.artifactory.primary.replicaCount .Values.artifactory.node.replicaCount }}" + } + ] +} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/files/migrate.sh b/charts/jfrog/artifactory-ha/107.90.14/files/migrate.sh new file mode 100644 index 0000000000..ba44160f47 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/files/migrate.sh @@ -0,0 +1,4311 @@ +#!/bin/bash + +# Flags +FLAG_Y="y" +FLAG_N="n" +FLAGS_Y_N="$FLAG_Y $FLAG_N" +FLAG_NOT_APPLICABLE="_NA_" + +CURRENT_VERSION=$1 + +WRAPPER_SCRIPT_TYPE_RPMDEB="RPMDEB" +WRAPPER_SCRIPT_TYPE_DOCKER_COMPOSE="DOCKERCOMPOSE" + +SENSITIVE_KEY_VALUE="__sensitive_key_hidden___" + +# Shared system keys +SYS_KEY_SHARED_JFROGURL="shared.jfrogUrl" +SYS_KEY_SHARED_SECURITY_JOINKEY="shared.security.joinKey" +SYS_KEY_SHARED_SECURITY_MASTERKEY="shared.security.masterKey" + +SYS_KEY_SHARED_NODE_ID="shared.node.id" +SYS_KEY_SHARED_JAVAHOME="shared.javaHome" + +SYS_KEY_SHARED_DATABASE_TYPE="shared.database.type" +SYS_KEY_SHARED_DATABASE_TYPE_VALUE_POSTGRES="postgresql" +SYS_KEY_SHARED_DATABASE_DRIVER="shared.database.driver" +SYS_KEY_SHARED_DATABASE_URL="shared.database.url" +SYS_KEY_SHARED_DATABASE_USERNAME="shared.database.username" +SYS_KEY_SHARED_DATABASE_PASSWORD="shared.database.password" + +SYS_KEY_SHARED_ELASTICSEARCH_URL="shared.elasticsearch.url" +SYS_KEY_SHARED_ELASTICSEARCH_USERNAME="shared.elasticsearch.username" +SYS_KEY_SHARED_ELASTICSEARCH_PASSWORD="shared.elasticsearch.password" +SYS_KEY_SHARED_ELASTICSEARCH_CLUSTERSETUP="shared.elasticsearch.clusterSetup" +SYS_KEY_SHARED_ELASTICSEARCH_UNICASTFILE="shared.elasticsearch.unicastFile" +SYS_KEY_SHARED_ELASTICSEARCH_CLUSTERSETUP_VALUE="YES" + +# Define this in product specific script. Should contain the path to unitcast file +# File used by insight server to write cluster active nodes info. This will be read by elasticsearch +#SYS_KEY_SHARED_ELASTICSEARCH_UNICASTFILE_VALUE="" + +SYS_KEY_RABBITMQ_ACTIVE_NODE_NAME="shared.rabbitMq.active.node.name" +SYS_KEY_RABBITMQ_ACTIVE_NODE_IP="shared.rabbitMq.active.node.ip" + +# Filenames +FILE_NAME_SYSTEM_YAML="system.yaml" +FILE_NAME_JOIN_KEY="join.key" +FILE_NAME_MASTER_KEY="master.key" +FILE_NAME_INSTALLER_YAML="installer.yaml" + +# Global constants used in business logic +NODE_TYPE_STANDALONE="standalone" +NODE_TYPE_CLUSTER_NODE="node" +NODE_TYPE_DATABASE="database" + +# External(isable) databases +DATABASE_POSTGRES="POSTGRES" +DATABASE_ELASTICSEARCH="ELASTICSEARCH" +DATABASE_RABBITMQ="RABBITMQ" + +POSTGRES_LABEL="PostgreSQL" +ELASTICSEARCH_LABEL="Elasticsearch" +RABBITMQ_LABEL="Rabbitmq" + +ARTIFACTORY_LABEL="Artifactory" +JFMC_LABEL="Mission Control" +DISTRIBUTION_LABEL="Distribution" +XRAY_LABEL="Xray" + +POSTGRES_CONTAINER="postgres" +ELASTICSEARCH_CONTAINER="elasticsearch" +RABBITMQ_CONTAINER="rabbitmq" +REDIS_CONTAINER="redis" + +#Adding a small timeout before a read ensures it is positioned correctly in the screen +read_timeout=0.5 + +# Options related to data directory location +PROMPT_DATA_DIR_LOCATION="Installation Directory" +KEY_DATA_DIR_LOCATION="installer.data_dir" + +SYS_KEY_SHARED_NODE_HAENABLED="shared.node.haEnabled" +PROMPT_ADD_TO_CLUSTER="Are you adding an additional node to an existing product cluster?" +KEY_ADD_TO_CLUSTER="installer.ha" +VALID_VALUES_ADD_TO_CLUSTER="$FLAGS_Y_N" + +MESSAGE_POSTGRES_INSTALL="The installer can install a $POSTGRES_LABEL database, or you can connect to an existing compatible $POSTGRES_LABEL database\n(compatible databases: https://www.jfrog.com/confluence/display/JFROG/System+Requirements#SystemRequirements-RequirementsMatrix)" +PROMPT_POSTGRES_INSTALL="Do you want to install $POSTGRES_LABEL?" +KEY_POSTGRES_INSTALL="installer.install_postgresql" +VALID_VALUES_POSTGRES_INSTALL="$FLAGS_Y_N" + +# Postgres connection details +RPM_DEB_POSTGRES_HOME_DEFAULT="/var/opt/jfrog/postgres" +RPM_DEB_MESSAGE_STANDALONE_POSTGRES_DATA="$POSTGRES_LABEL home will have data and its configuration" +RPM_DEB_PROMPT_STANDALONE_POSTGRES_DATA="Type desired $POSTGRES_LABEL home location" +RPM_DEB_KEY_STANDALONE_POSTGRES_DATA="installer.postgresql.home" + +MESSAGE_DATABASE_URL="Provide the database connection details" +PROMPT_DATABASE_URL(){ + local databaseURlExample= + case "$PRODUCT_NAME" in + $ARTIFACTORY_LABEL) + databaseURlExample="jdbc:postgresql://:/artifactory" + ;; + $JFMC_LABEL) + databaseURlExample="postgresql://:/mission_control?sslmode=disable" + ;; + $DISTRIBUTION_LABEL) + databaseURlExample="jdbc:postgresql://:/distribution?sslmode=disable" + ;; + $XRAY_LABEL) + databaseURlExample="postgres://:/xraydb?sslmode=disable" + ;; + esac + if [ -z "$databaseURlExample" ]; then + echo -n "$POSTGRES_LABEL URL" # For consistency with username and password + return + fi + echo -n "$POSTGRES_LABEL url. Example: [$databaseURlExample]" +} +REGEX_DATABASE_URL(){ + local databaseURlExample= + case "$PRODUCT_NAME" in + $ARTIFACTORY_LABEL) + databaseURlExample="jdbc:postgresql://.*/artifactory.*" + ;; + $JFMC_LABEL) + databaseURlExample="postgresql://.*/mission_control.*" + ;; + $DISTRIBUTION_LABEL) + databaseURlExample="jdbc:postgresql://.*/distribution.*" + ;; + $XRAY_LABEL) + databaseURlExample="postgres://.*/xraydb.*" + ;; + esac + echo -n "^$databaseURlExample\$" +} +ERROR_MESSAGE_DATABASE_URL="Invalid $POSTGRES_LABEL URL" +KEY_DATABASE_URL="$SYS_KEY_SHARED_DATABASE_URL" +#NOTE: It is important to display the label. Since the message may be hidden if URL is known +PROMPT_DATABASE_USERNAME="$POSTGRES_LABEL username" +KEY_DATABASE_USERNAME="$SYS_KEY_SHARED_DATABASE_USERNAME" +#NOTE: It is important to display the label. Since the message may be hidden if URL is known +PROMPT_DATABASE_PASSWORD="$POSTGRES_LABEL password" +KEY_DATABASE_PASSWORD="$SYS_KEY_SHARED_DATABASE_PASSWORD" +IS_SENSITIVE_DATABASE_PASSWORD="$FLAG_Y" + +MESSAGE_STANDALONE_ELASTICSEARCH_INSTALL="The installer can install a $ELASTICSEARCH_LABEL database or you can connect to an existing compatible $ELASTICSEARCH_LABEL database" +PROMPT_STANDALONE_ELASTICSEARCH_INSTALL="Do you want to install $ELASTICSEARCH_LABEL?" +KEY_STANDALONE_ELASTICSEARCH_INSTALL="installer.install_elasticsearch" +VALID_VALUES_STANDALONE_ELASTICSEARCH_INSTALL="$FLAGS_Y_N" + +# Elasticsearch connection details +MESSAGE_ELASTICSEARCH_DETAILS="Provide the $ELASTICSEARCH_LABEL connection details" +PROMPT_ELASTICSEARCH_URL="$ELASTICSEARCH_LABEL URL" +KEY_ELASTICSEARCH_URL="$SYS_KEY_SHARED_ELASTICSEARCH_URL" + +PROMPT_ELASTICSEARCH_USERNAME="$ELASTICSEARCH_LABEL username" +KEY_ELASTICSEARCH_USERNAME="$SYS_KEY_SHARED_ELASTICSEARCH_USERNAME" + +PROMPT_ELASTICSEARCH_PASSWORD="$ELASTICSEARCH_LABEL password" +KEY_ELASTICSEARCH_PASSWORD="$SYS_KEY_SHARED_ELASTICSEARCH_PASSWORD" +IS_SENSITIVE_ELASTICSEARCH_PASSWORD="$FLAG_Y" + +# Cluster related questions +MESSAGE_CLUSTER_MASTER_KEY="Provide the cluster's master key. It can be found in the data directory of the first node under /etc/security/master.key" +PROMPT_CLUSTER_MASTER_KEY="Master Key" +KEY_CLUSTER_MASTER_KEY="$SYS_KEY_SHARED_SECURITY_MASTERKEY" +IS_SENSITIVE_CLUSTER_MASTER_KEY="$FLAG_Y" + +MESSAGE_JOIN_KEY="The Join key is the secret key used to establish trust between services in the JFrog Platform.\n(You can copy the Join Key from Admin > User Management > Settings)" +PROMPT_JOIN_KEY="Join Key" +KEY_JOIN_KEY="$SYS_KEY_SHARED_SECURITY_JOINKEY" +IS_SENSITIVE_JOIN_KEY="$FLAG_Y" +REGEX_JOIN_KEY="^[a-zA-Z0-9]{16,}\$" +ERROR_MESSAGE_JOIN_KEY="Invalid Join Key" + +# Rabbitmq related cluster information +MESSAGE_RABBITMQ_ACTIVE_NODE_NAME="Provide an active ${RABBITMQ_LABEL} node name. Run the command [ hostname -s ] on any of the existing nodes in the product cluster to get this" +PROMPT_RABBITMQ_ACTIVE_NODE_NAME="${RABBITMQ_LABEL} active node name" +KEY_RABBITMQ_ACTIVE_NODE_NAME="$SYS_KEY_RABBITMQ_ACTIVE_NODE_NAME" + +# Rabbitmq related cluster information (necessary only for docker-compose) +PROMPT_RABBITMQ_ACTIVE_NODE_IP="${RABBITMQ_LABEL} active node ip" +KEY_RABBITMQ_ACTIVE_NODE_IP="$SYS_KEY_RABBITMQ_ACTIVE_NODE_IP" + +MESSAGE_JFROGURL(){ + echo -e "The JFrog URL allows ${PRODUCT_NAME} to connect to a JFrog Platform Instance.\n(You can copy the JFrog URL from Administration > User Management > Settings > Connection details)" +} +PROMPT_JFROGURL="JFrog URL" +KEY_JFROGURL="$SYS_KEY_SHARED_JFROGURL" +REGEX_JFROGURL="^https?://.*:{0,}[0-9]{0,4}\$" +ERROR_MESSAGE_JFROGURL="Invalid JFrog URL" + + +# Set this to FLAG_Y on upgrade +IS_UPGRADE="${FLAG_N}" + +# This belongs in JFMC but is the ONLY one that needs it so keeping it here for now. Can be made into a method and overridden if necessary +MESSAGE_MULTIPLE_PG_SCHEME="Please setup $POSTGRES_LABEL with schema as described in https://www.jfrog.com/confluence/display/JFROG/Installing+Mission+Control" + +_getMethodOutputOrVariableValue() { + unset EFFECTIVE_MESSAGE + local keyToSearch=$1 + local effectiveMessage= + local result="0" + # logSilly "Searching for method: [$keyToSearch]" + LC_ALL=C type "$keyToSearch" > /dev/null 2>&1 || result="$?" + if [[ "$result" == "0" ]]; then + # logSilly "Found method for [$keyToSearch]" + EFFECTIVE_MESSAGE="$($keyToSearch)" + return + fi + eval EFFECTIVE_MESSAGE=\${$keyToSearch} + if [ ! -z "$EFFECTIVE_MESSAGE" ]; then + return + fi + # logSilly "Didn't find method or variable for [$keyToSearch]" +} + + +# REF https://misc.flogisoft.com/bash/tip_colors_and_formatting +cClear="\e[0m" +cBlue="\e[38;5;69m" +cRedDull="\e[1;31m" +cYellow="\e[1;33m" +cRedBright="\e[38;5;197m" +cBold="\e[1m" + + +_loggerGetModeRaw() { + local MODE="$1" + case $MODE in + INFO) + printf "" + ;; + DEBUG) + printf "%s" "[${MODE}] " + ;; + WARN) + printf "${cRedDull}%s%s${cClear}" "[" "${MODE}" "] " + ;; + ERROR) + printf "${cRedBright}%s%s${cClear}" "[" "${MODE}" "] " + ;; + esac +} + + +_loggerGetMode() { + local MODE="$1" + case $MODE in + INFO) + printf "${cBlue}%s%-5s%s${cClear}" "[" "${MODE}" "]" + ;; + DEBUG) + printf "%-7s" "[${MODE}]" + ;; + WARN) + printf "${cRedDull}%s%-5s%s${cClear}" "[" "${MODE}" "]" + ;; + ERROR) + printf "${cRedBright}%s%-5s%s${cClear}" "[" "${MODE}" "]" + ;; + esac +} + +# Capitalises the first letter of the message +_loggerGetMessage() { + local originalMessage="$*" + local firstChar=$(echo "${originalMessage:0:1}" | awk '{ print toupper($0) }') + local resetOfMessage="${originalMessage:1}" + echo "$firstChar$resetOfMessage" +} + +# The spec also says content should be left-trimmed but this is not necessary in our case. We don't reach the limit. +_loggerGetStackTrace() { + printf "%s%-30s%s" "[" "$1:$2" "]" +} + +_loggerGetThread() { + printf "%s" "[main]" +} + +_loggerGetServiceType() { + printf "%s%-5s%s" "[" "shell" "]" +} + +#Trace ID is not applicable to scripts +_loggerGetTraceID() { + printf "%s" "[]" +} + +logRaw() { + echo "" + printf "$1" + echo "" +} + +logBold(){ + echo "" + printf "${cBold}$1${cClear}" + echo "" +} + +# The date binary works differently based on whether it is GNU/BSD +is_date_supported=0 +date --version > /dev/null 2>&1 || is_date_supported=1 +IS_GNU=$(echo $is_date_supported) + +_loggerGetTimestamp() { + if [ "${IS_GNU}" == "0" ]; then + echo -n $(date -u +%FT%T.%3NZ) + else + echo -n $(date -u +%FT%T.000Z) + fi +} + +# https://www.shellscript.sh/tips/spinner/ +_spin() +{ + spinner="/|\\-/|\\-" + while : + do + for i in `seq 0 7` + do + echo -n "${spinner:$i:1}" + echo -en "\010" + sleep 1 + done + done +} + +showSpinner() { + # Start the Spinner: + _spin & + # Make a note of its Process ID (PID): + SPIN_PID=$! + # Kill the spinner on any signal, including our own exit. + trap "kill -9 $SPIN_PID" `seq 0 15` &> /dev/null || return 0 +} + +stopSpinner() { + local occurrences=$(ps -ef | grep -wc "${SPIN_PID}") + let "occurrences+=0" + # validate that it is present (2 since this search itself will show up in the results) + if [ $occurrences -gt 1 ]; then + kill -9 $SPIN_PID &>/dev/null || return 0 + wait $SPIN_ID &>/dev/null + fi +} + +_getEffectiveMessage(){ + local MESSAGE="$1" + local MODE=${2-"INFO"} + + if [ -z "$CONTEXT" ]; then + CONTEXT=$(caller) + fi + + _EFFECTIVE_MESSAGE= + if [ -z "$LOG_BEHAVIOR_ADD_META" ]; then + _EFFECTIVE_MESSAGE="$(_loggerGetModeRaw $MODE)$(_loggerGetMessage $MESSAGE)" + else + local SERVICE_TYPE="script" + local TRACE_ID="" + local THREAD="main" + + local CONTEXT_LINE=$(echo "$CONTEXT" | awk '{print $1}') + local CONTEXT_FILE=$(echo "$CONTEXT" | awk -F"/" '{print $NF}') + + _EFFECTIVE_MESSAGE="$(_loggerGetTimestamp) $(_loggerGetServiceType) $(_loggerGetMode $MODE) $(_loggerGetTraceID) $(_loggerGetStackTrace $CONTEXT_FILE $CONTEXT_LINE) $(_loggerGetThread) - $(_loggerGetMessage $MESSAGE)" + fi + CONTEXT= +} + +# Important - don't call any log method from this method. Will become an infinite loop. Use echo to debug +_logToFile() { + local MODE=${1-"INFO"} + local targetFile="$LOG_BEHAVIOR_ADD_REDIRECTION" + # IF the file isn't passed, abort + if [ -z "$targetFile" ]; then + return + fi + # IF this is not being run in verbose mode and mode is debug or lower, abort + if [ "${VERBOSE_MODE}" != "$FLAG_Y" ] && [ "${VERBOSE_MODE}" != "true" ] && [ "${VERBOSE_MODE}" != "debug" ]; then + if [ "$MODE" == "DEBUG" ] || [ "$MODE" == "SILLY" ]; then + return + fi + fi + + # Create the file if it doesn't exist + if [ ! -f "${targetFile}" ]; then + return + # touch $targetFile > /dev/null 2>&1 || true + fi + # # Make it readable + # chmod 640 $targetFile > /dev/null 2>&1 || true + + # Log contents + printf "%s\n" "$_EFFECTIVE_MESSAGE" >> "$targetFile" || true +} + +logger() { + if [ "$LOG_BEHAVIOR_ADD_NEW_LINE" == "$FLAG_Y" ]; then + echo "" + fi + _getEffectiveMessage "$@" + local MODE=${2-"INFO"} + printf "%s\n" "$_EFFECTIVE_MESSAGE" + _logToFile "$MODE" +} + +logDebug(){ + VERBOSE_MODE=${VERBOSE_MODE-"false"} + CONTEXT=$(caller) + if [ "${VERBOSE_MODE}" == "$FLAG_Y" ] || [ "${VERBOSE_MODE}" == "true" ] || [ "${VERBOSE_MODE}" == "debug" ];then + logger "$1" "DEBUG" + else + logger "$1" "DEBUG" >&6 + fi + CONTEXT= +} + +logSilly(){ + VERBOSE_MODE=${VERBOSE_MODE-"false"} + CONTEXT=$(caller) + if [ "${VERBOSE_MODE}" == "silly" ];then + logger "$1" "DEBUG" + else + logger "$1" "DEBUG" >&6 + fi + CONTEXT= +} + +logError() { + CONTEXT=$(caller) + logger "$1" "ERROR" + CONTEXT= +} + +errorExit () { + CONTEXT=$(caller) + logger "$1" "ERROR" + CONTEXT= + exit 1 +} + +warn () { + CONTEXT=$(caller) + logger "$1" "WARN" + CONTEXT= +} + +note () { + CONTEXT=$(caller) + logger "$1" "NOTE" + CONTEXT= +} + +bannerStart() { + title=$1 + echo + echo -e "\033[1m${title}\033[0m" + echo +} + +bannerSection() { + title=$1 + echo + echo -e "******************************** ${title} ********************************" + echo +} + +bannerSubSection() { + title=$1 + echo + echo -e "************** ${title} *******************" + echo +} + +bannerMessge() { + title=$1 + echo + echo -e "********************************" + echo -e "${title}" + echo -e "********************************" + echo +} + +setRed () { + local input="$1" + echo -e \\033[31m${input}\\033[0m +} +setGreen () { + local input="$1" + echo -e \\033[32m${input}\\033[0m +} +setYellow () { + local input="$1" + echo -e \\033[33m${input}\\033[0m +} + +logger_addLinebreak () { + echo -e "---\n" +} + +bannerImportant() { + title=$1 + local bold="\033[1m" + local noColour="\033[0m" + echo + echo -e "${bold}######################################## IMPORTANT ########################################${noColour}" + echo -e "${bold}${title}${noColour}" + echo -e "${bold}###########################################################################################${noColour}" + echo +} + +bannerEnd() { + #TODO pass a title and calculate length dynamically so that start and end look alike + echo + echo "*****************************************************************************" + echo +} + +banner() { + title=$1 + content=$2 + bannerStart "${title}" + echo -e "$content" +} + +# The logic below helps us redirect content we'd normally hide to the log file. + # + # We have several commands which clutter the console with output and so use + # `cmd > /dev/null` - this redirects the command's output to null. + # + # However, the information we just hid maybe useful for support. Using the code pattern + # `cmd >&6` (instead of `cmd> >/dev/null` ), the command's output is hidden from the console + # but redirected to the installation log file + # + +#Default value of 6 is just null +exec 6>>/dev/null +redirectLogsToFile() { + echo "" + # local file=$1 + + # [ ! -z "${file}" ] || return 0 + + # local logDir=$(dirname "$file") + + # if [ ! -f "${file}" ]; then + # [ -d "${logDir}" ] || mkdir -p ${logDir} || \ + # ( echo "WARNING : Could not create parent directory (${logDir}) to redirect console log : ${file}" ; return 0 ) + # fi + + # #6 now points to the log file + # exec 6>>${file} + # #reference https://unix.stackexchange.com/questions/145651/using-exec-and-tee-to-redirect-logs-to-stdout-and-a-log-file-in-the-same-time + # exec 2>&1 > >(tee -a "${file}") +} + +# Check if a give key contains any sensitive string as part of it +# Based on the result, the caller can decide its value can be displayed or not +# Sample usage : isKeySensitive "${key}" && displayValue="******" || displayValue=${value} +isKeySensitive(){ + local key=$1 + local sensitiveKeys="password|secret|key|token" + + if [ -z "${key}" ]; then + return 1 + else + local lowercaseKey=$(echo "${key}" | tr '[:upper:]' '[:lower:]' 2>/dev/null) + [[ "${lowercaseKey}" =~ ${sensitiveKeys} ]] && return 0 || return 1 + fi +} + +getPrintableValueOfKey(){ + local displayValue= + local key="$1" + if [ -z "$key" ]; then + # This is actually an incorrect usage of this method but any logging will cause unexpected content in the caller + echo -n "" + return + fi + + local value="$2" + isKeySensitive "${key}" && displayValue="$SENSITIVE_KEY_VALUE" || displayValue="${value}" + echo -n $displayValue +} + +_createConsoleLog(){ + if [ -z "${JF_PRODUCT_HOME}" ]; then + return + fi + local targetFile="${JF_PRODUCT_HOME}/var/log/console.log" + mkdir -p "${JF_PRODUCT_HOME}/var/log" || true + if [ ! -f ${targetFile} ]; then + touch $targetFile > /dev/null 2>&1 || true + fi + chmod 640 $targetFile > /dev/null 2>&1 || true +} + +# Output from application's logs are piped to this method. It checks a configuration variable to determine if content should be logged to +# the common console.log file +redirectServiceLogsToFile() { + + local result="0" + # check if the function getSystemValue exists + LC_ALL=C type getSystemValue > /dev/null 2>&1 || result="$?" + if [[ "$result" != "0" ]]; then + warn "Couldn't find the systemYamlHelper. Skipping log redirection" + return 0 + fi + + getSystemValue "shared.consoleLog" "NOT_SET" + if [[ "${YAML_VALUE}" == "false" ]]; then + logger "Redirection is set to false. Skipping log redirection" + return 0; + fi + + if [ -z "${JF_PRODUCT_HOME}" ] || [ "${JF_PRODUCT_HOME}" == "" ]; then + warn "JF_PRODUCT_HOME is unavailable. Skipping log redirection" + return 0 + fi + + local targetFile="${JF_PRODUCT_HOME}/var/log/console.log" + + _createConsoleLog + + while read -r line; do + printf '%s\n' "${line}" >> $targetFile || return 0 # Don't want to log anything - might clutter the screen + done +} + +## Display environment variables starting with JF_ along with its value +## Value of sensitive keys will be displayed as "******" +## +## Sample Display : +## +## ======================== +## JF Environment variables +## ======================== +## +## JF_SHARED_NODE_ID : locahost +## JF_SHARED_JOINKEY : ****** +## +## +displayEnv() { + local JFEnv=$(printenv | grep ^JF_ 2>/dev/null) + local key= + local value= + + if [ -z "${JFEnv}" ]; then + return + fi + + cat << ENV_START_MESSAGE + +======================== +JF Environment variables +======================== +ENV_START_MESSAGE + + for entry in ${JFEnv}; do + key=$(echo "${entry}" | awk -F'=' '{print $1}') + value=$(echo "${entry}" | awk -F'=' '{print $2}') + + isKeySensitive "${key}" && value="******" || value=${value} + + printf "\n%-35s%s" "${key}" " : ${value}" + done + echo; +} + +_addLogRotateConfiguration() { + logDebug "Method ${FUNCNAME[0]}" + # mandatory inputs + local confFile="$1" + local logFile="$2" + + # Method available in _ioOperations.sh + LC_ALL=C type io_setYQPath > /dev/null 2>&1 || return 1 + + io_setYQPath + + # Method available in _systemYamlHelper.sh + LC_ALL=C type getSystemValue > /dev/null 2>&1 || return 1 + + local frequency="daily" + local archiveFolder="archived" + + local compressLogFiles= + getSystemValue "shared.logging.rotation.compress" "true" + if [[ "${YAML_VALUE}" == "true" ]]; then + compressLogFiles="compress" + fi + + getSystemValue "shared.logging.rotation.maxFiles" "10" + local noOfBackupFiles="${YAML_VALUE}" + + getSystemValue "shared.logging.rotation.maxSizeMb" "25" + local sizeOfFile="${YAML_VALUE}M" + + logDebug "Adding logrotate configuration for [$logFile] to [$confFile]" + + # Add configuration to file + local confContent=$(cat << LOGROTATECONF +$logFile { + $frequency + missingok + rotate $noOfBackupFiles + $compressLogFiles + notifempty + olddir $archiveFolder + dateext + extension .log + dateformat -%Y-%m-%d + size ${sizeOfFile} +} +LOGROTATECONF +) + echo "${confContent}" > ${confFile} || return 1 +} + +_operationIsBySameUser() { + local targetUser="$1" + local currentUserID=$(id -u) + local currentUserName=$(id -un) + + if [ $currentUserID == $targetUser ] || [ $currentUserName == $targetUser ]; then + echo -n "yes" + else + echo -n "no" + fi +} + +_addCronJobForLogrotate() { + logDebug "Method ${FUNCNAME[0]}" + + # Abort if logrotate is not available + [ "$(io_commandExists 'crontab')" != "yes" ] && warn "cron is not available" && return 1 + + # mandatory inputs + local productHome="$1" + local confFile="$2" + local cronJobOwner="$3" + + # We want to use our binary if possible. It may be more recent than the one in the OS + local logrotateBinary="$productHome/app/third-party/logrotate/logrotate" + + if [ ! -f "$logrotateBinary" ]; then + logrotateBinary="logrotate" + [ "$(io_commandExists 'logrotate')" != "yes" ] && warn "logrotate is not available" && return 1 + fi + local cmd="$logrotateBinary ${confFile} --state $productHome/var/etc/logrotate/logrotate-state" #--verbose + + id -u $cronJobOwner > /dev/null 2>&1 || { warn "User $cronJobOwner does not exist. Aborting logrotate configuration" && return 1; } + + # Remove the existing line + removeLogRotation "$productHome" "$cronJobOwner" || true + + # Run logrotate daily at 23:55 hours + local cronInterval="55 23 * * * $cmd" + + local standaloneMode=$(_operationIsBySameUser "$cronJobOwner") + + # If this is standalone mode, we cannot use -u - the user running this process may not have the necessary privileges + if [ "$standaloneMode" == "no" ]; then + (crontab -l -u $cronJobOwner 2>/dev/null; echo "$cronInterval") | crontab -u $cronJobOwner - + else + (crontab -l 2>/dev/null; echo "$cronInterval") | crontab - + fi +} + +## Configure logrotate for a product +## Failure conditions: +## If logrotation could not be setup for some reason +## Parameters: +## $1: The product name +## $2: The product home +## Depends on global: none +## Updates global: none +## Returns: NA + +configureLogRotation() { + logDebug "Method ${FUNCNAME[0]}" + + # mandatory inputs + local productName="$1" + if [ -z $productName ]; then + warn "Incorrect usage. A product name is necessary for configuring log rotation" && return 1 + fi + + local productHome="$2" + if [ -z $productHome ]; then + warn "Incorrect usage. A product home folder is necessary for configuring log rotation" && return 1 + fi + + local logFile="${productHome}/var/log/console.log" + if [[ $(uname) == "Darwin" ]]; then + logger "Log rotation for [$logFile] has not been configured. Please setup manually" + return 0 + fi + + local userID="$3" + if [ -z $userID ]; then + warn "Incorrect usage. A userID is necessary for configuring log rotation" && return 1 + fi + + local groupID=${4:-$userID} + local logConfigOwner=${5:-$userID} + + logDebug "Configuring log rotation as user [$userID], group [$groupID], effective cron User [$logConfigOwner]" + + local errorMessage="Could not configure logrotate. Please configure log rotation of the file: [$logFile] manually" + + local confFile="${productHome}/var/etc/logrotate/logrotate.conf" + + # TODO move to recursive method + createDir "${productHome}" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var/log" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var/log/archived" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + + # TODO move to recursive method + createDir "${productHome}/var/etc" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var/etc/logrotate" "$logConfigOwner" || { warn "${errorMessage}" && return 1; } + + # conf file should be owned by the user running the script + createFile "${confFile}" "${logConfigOwner}" || { warn "Could not create configuration file [$confFile]" return 1; } + + _addLogRotateConfiguration "${confFile}" "${logFile}" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + _addCronJobForLogrotate "${productHome}" "${confFile}" "${logConfigOwner}" || { warn "${errorMessage}" && return 1; } +} + +_pauseExecution() { + if [ "${VERBOSE_MODE}" == "debug" ]; then + + local breakPoint="$1" + if [ ! -z "$breakPoint" ]; then + printf "${cBlue}Breakpoint${cClear} [$breakPoint] " + echo "" + fi + printf "${cBlue}Press enter once you are ready to continue${cClear}" + read -s choice + echo "" + fi +} + +# removeLogRotation "$productHome" "$cronJobOwner" || true +removeLogRotation() { + logDebug "Method ${FUNCNAME[0]}" + if [[ $(uname) == "Darwin" ]]; then + logDebug "Not implemented for Darwin." + return 0 + fi + local productHome="$1" + local cronJobOwner="$2" + local standaloneMode=$(_operationIsBySameUser "$cronJobOwner") + + local confFile="${productHome}/var/etc/logrotate/logrotate.conf" + + if [ "$standaloneMode" == "no" ]; then + crontab -l -u $cronJobOwner 2>/dev/null | grep -v "$confFile" | crontab -u $cronJobOwner - + else + crontab -l 2>/dev/null | grep -v "$confFile" | crontab - + fi +} + +# NOTE: This method does not check the configuration to see if redirection is necessary. +# This is intentional. If we don't redirect, tomcat logs might get redirected to a folder/file +# that does not exist, causing the service itself to not start +setupTomcatRedirection() { + logDebug "Method ${FUNCNAME[0]}" + local consoleLog="${JF_PRODUCT_HOME}/var/log/console.log" + _createConsoleLog + export CATALINA_OUT="${consoleLog}" +} + +setupScriptLogsRedirection() { + logDebug "Method ${FUNCNAME[0]}" + if [ -z "${JF_PRODUCT_HOME}" ]; then + logDebug "No JF_PRODUCT_HOME. Returning" + return + fi + # Create the console.log file if it is not already present + # _createConsoleLog || true + # # Ensure any logs (logger/logError/warn) also get redirected to the console.log + # # Using installer.log as a temparory fix. Please change this to console.log once INST-291 is fixed + export LOG_BEHAVIOR_ADD_REDIRECTION="${JF_PRODUCT_HOME}/var/log/console.log" + export LOG_BEHAVIOR_ADD_META="$FLAG_Y" +} + +# Returns Y if this method is run inside a container +isRunningInsideAContainer() { + local check1=$(grep -sq 'docker\|kubepods' /proc/1/cgroup; echo $?) + local check2=$(grep -sq 'containers' /proc/self/mountinfo; echo $?) + if [[ $check1 == 0 || $check2 == 0 || -f "/.dockerenv" ]]; then + echo -n "$FLAG_Y" + else + echo -n "$FLAG_N" + fi +} + +POSTGRES_USER=999 +NGINX_USER=104 +NGINX_GROUP=107 +ES_USER=1000 +REDIS_USER=999 +MONGO_USER=999 +RABBITMQ_USER=999 +LOG_FILE_PERMISSION=640 +PID_FILE_PERMISSION=644 + +# Copy file +copyFile(){ + local source=$1 + local target=$2 + local mode=${3:-overwrite} + local enableVerbose=${4:-"${FLAG_N}"} + local verboseFlag="" + + if [ ! -z "${enableVerbose}" ] && [ "${enableVerbose}" == "${FLAG_Y}" ]; then + verboseFlag="-v" + fi + + if [[ ! ( $source && $target ) ]]; then + warn "Source and target is mandatory to copy file" + return 1 + fi + + if [[ -f "${target}" ]]; then + [[ "$mode" = "overwrite" ]] && ( cp ${verboseFlag} -f "$source" "$target" || errorExit "Unable to copy file, command : cp -f ${source} ${target}") || true + else + cp ${verboseFlag} -f "$source" "$target" || errorExit "Unable to copy file, command : cp -f ${source} ${target}" + fi +} + +# Copy files recursively from given source directory to destination directory +# This method wil copy but will NOT overwrite +# Destination will be created if its not available +copyFilesNoOverwrite(){ + local src=$1 + local dest=$2 + local enableVerboseCopy="${3:-${FLAG_Y}}" + + if [[ -z "${src}" || -z "${dest}" ]]; then + return + fi + + if [ -d "${src}" ] && [ "$(ls -A ${src})" ]; then + local relativeFilePath="" + local targetFilePath="" + + for file in $(find ${src} -type f 2>/dev/null) ; do + # Derive relative path and attach it to destination + # Example : + # src=/extra_config + # dest=/var/opt/jfrog/artifactory/etc + # file=/extra_config/config.xml + # relativeFilePath=config.xml + # targetFilePath=/var/opt/jfrog/artifactory/etc/config.xml + relativeFilePath=${file/${src}/} + targetFilePath=${dest}${relativeFilePath} + + createDir "$(dirname "$targetFilePath")" + copyFile "${file}" "${targetFilePath}" "no_overwrite" "${enableVerboseCopy}" + done + fi +} + +# TODO : WINDOWS ? +# Check the max open files and open processes set on the system +checkULimits () { + local minMaxOpenFiles=${1:-32000} + local minMaxOpenProcesses=${2:-1024} + local setValue=${3:-true} + local warningMsgForFiles=${4} + local warningMsgForProcesses=${5} + + logger "Checking open files and processes limits" + + local currentMaxOpenFiles=$(ulimit -n) + logger "Current max open files is $currentMaxOpenFiles" + if [ ${currentMaxOpenFiles} != "unlimited" ] && [ "$currentMaxOpenFiles" -lt "$minMaxOpenFiles" ]; then + if [ "${setValue}" ]; then + ulimit -n "${minMaxOpenFiles}" >/dev/null 2>&1 || warn "Max number of open files $currentMaxOpenFiles is low!" + [ -z "${warningMsgForFiles}" ] || warn "${warningMsgForFiles}" + else + errorExit "Max number of open files $currentMaxOpenFiles, is too low. Cannot run the application!" + fi + fi + + local currentMaxOpenProcesses=$(ulimit -u) + logger "Current max open processes is $currentMaxOpenProcesses" + if [ "$currentMaxOpenProcesses" != "unlimited" ] && [ "$currentMaxOpenProcesses" -lt "$minMaxOpenProcesses" ]; then + if [ "${setValue}" ]; then + ulimit -u "${minMaxOpenProcesses}" >/dev/null 2>&1 || warn "Max number of open files $currentMaxOpenFiles is low!" + [ -z "${warningMsgForProcesses}" ] || warn "${warningMsgForProcesses}" + else + errorExit "Max number of open files $currentMaxOpenProcesses, is too low. Cannot run the application!" + fi + fi +} + +createDirs() { + local appDataDir=$1 + local serviceName=$2 + local folders="backup bootstrap data etc logs work" + + [ -z "${appDataDir}" ] && errorExit "An application directory is mandatory to create its data structure" || true + [ -z "${serviceName}" ] && errorExit "A service name is mandatory to create service data structure" || true + + for folder in ${folders} + do + folder=${appDataDir}/${folder}/${serviceName} + if [ ! -d "${folder}" ]; then + logger "Creating folder : ${folder}" + mkdir -p "${folder}" || errorExit "Failed to create ${folder}" + fi + done +} + + +testReadWritePermissions () { + local dir_to_check=$1 + local error=false + + [ -d ${dir_to_check} ] || errorExit "'${dir_to_check}' is not a directory" + + local test_file=${dir_to_check}/test-permissions + + # Write file + if echo test > ${test_file} 1> /dev/null 2>&1; then + # Write succeeded. Testing read... + if cat ${test_file} > /dev/null; then + rm -f ${test_file} + else + error=true + fi + else + error=true + fi + + if [ ${error} == true ]; then + return 1 + else + return 0 + fi +} + +# Test directory has read/write permissions for current user +testDirectoryPermissions () { + local dir_to_check=$1 + local error=false + + [ -d ${dir_to_check} ] || errorExit "'${dir_to_check}' is not a directory" + + local u_id=$(id -u) + local id_str="id ${u_id}" + + logger "Testing directory ${dir_to_check} has read/write permissions for user ${id_str}" + + if ! testReadWritePermissions ${dir_to_check}; then + error=true + fi + + if [ "${error}" == true ]; then + local stat_data=$(stat -Lc "Directory: %n, permissions: %a, owner: %U, group: %G" ${dir_to_check}) + logger "###########################################################" + logger "${dir_to_check} DOES NOT have proper permissions for user ${id_str}" + logger "${stat_data}" + logger "Mounted directory must have read/write permissions for user ${id_str}" + logger "###########################################################" + errorExit "Directory ${dir_to_check} has bad permissions for user ${id_str}" + fi + logger "Permissions for ${dir_to_check} are good" +} + +# Utility method to create a directory path recursively with chown feature as +# Failure conditions: +## Exits if unable to create a directory +# Parameters: +## $1: Root directory from where the path can be created +## $2: List of recursive child directories separated by space +## $3: user who should own the directory. Optional +## $4: group who should own the directory. Optional +# Depends on global: none +# Updates global: none +# Returns: NA +# +# Usage: +# createRecursiveDir "/opt/jfrog/product/var" "bootstrap tomcat lib" "user_name" "group_name" +createRecursiveDir(){ + local rootDir=$1 + local pathDirs=$2 + local user=$3 + local group=${4:-${user}} + local fullPath= + + [ ! -z "${rootDir}" ] || return 0 + + createDir "${rootDir}" "${user}" "${group}" + + [ ! -z "${pathDirs}" ] || return 0 + + fullPath=${rootDir} + + for dir in ${pathDirs}; do + fullPath=${fullPath}/${dir} + createDir "${fullPath}" "${user}" "${group}" + done +} + +# Utility method to create a directory +# Failure conditions: +## Exits if unable to create a directory +# Parameters: +## $1: directory to create +## $2: user who should own the directory. Optional +## $3: group who should own the directory. Optional +# Depends on global: none +# Updates global: none +# Returns: NA + +createDir(){ + local dirName="$1" + local printMessage=no + logSilly "Method ${FUNCNAME[0]} invoked with [$dirName]" + [ -z "${dirName}" ] && return + + logDebug "Attempting to create ${dirName}" + mkdir -p "${dirName}" || errorExit "Unable to create directory: [${dirName}]" + local userID="$2" + local groupID=${3:-$userID} + + # If UID/GID is passed, chown the folder + if [ ! -z "$userID" ] && [ ! -z "$groupID" ]; then + # Earlier, this line would have returned 1 if it failed. Now it just warns. + # This is intentional. Earlier, this line would NOT be reached if the folder already existed. + # Since it will always come to this line and the script may be running as a non-root user, this method will just warn if + # setting permissions fails (so as to not affect any existing flows) + io_setOwnershipNonRecursive "$dirName" "$userID" "$groupID" || warn "Could not set owner of [$dirName] to [$userID:$groupID]" + fi + # logging message to print created dir with user and group + local logMessage=${4:-$printMessage} + if [[ "${logMessage}" == "yes" ]]; then + logger "Successfully created directory [${dirName}]. Owner: [${userID}:${groupID}]" + fi +} + +removeSoftLinkAndCreateDir () { + local dirName="$1" + local userID="$2" + local groupID="$3" + local logMessage="$4" + removeSoftLink "${dirName}" + createDir "${dirName}" "${userID}" "${groupID}" "${logMessage}" +} + +# Utility method to remove a soft link +removeSoftLink () { + local dirName="$1" + if [[ -L "${dirName}" ]]; then + targetLink=$(readlink -f "${dirName}") + logger "Removing the symlink [${dirName}] pointing to [${targetLink}]" + rm -f "${dirName}" + fi +} + +# Check Directory exist in the path +checkDirExists () { + local directoryPath="$1" + + [[ -d "${directoryPath}" ]] && echo -n "true" || echo -n "false" +} + + +# Utility method to create a file +# Failure conditions: +# Parameters: +## $1: file to create +# Depends on global: none +# Updates global: none +# Returns: NA + +createFile(){ + local fileName="$1" + logSilly "Method ${FUNCNAME[0]} [$fileName]" + [ -f "${fileName}" ] && return 0 + touch "${fileName}" || return 1 + + local userID="$2" + local groupID=${3:-$userID} + + # If UID/GID is passed, chown the folder + if [ ! -z "$userID" ] && [ ! -z "$groupID" ]; then + io_setOwnership "$fileName" "$userID" "$groupID" || return 1 + fi +} + +# Check File exist in the filePath +# IMPORTANT- DON'T ADD LOGGING to this method +checkFileExists () { + local filePath="$1" + + [[ -f "${filePath}" ]] && echo -n "true" || echo -n "false" +} + +# Check for directories contains any (files or sub directories) +# IMPORTANT- DON'T ADD LOGGING to this method +checkDirContents () { + local directoryPath="$1" + if [[ "$(ls -1 "${directoryPath}" | wc -l)" -gt 0 ]]; then + echo -n "true" + else + echo -n "false" + fi +} + +# Check contents exist in directory +# IMPORTANT- DON'T ADD LOGGING to this method +checkContentExists () { + local source="$1" + + if [[ "$(checkDirContents "${source}")" != "true" ]]; then + echo -n "false" + else + echo -n "true" + fi +} + +# Resolve the variable +# IMPORTANT- DON'T ADD LOGGING to this method +evalVariable () { + local output="$1" + local input="$2" + + eval "${output}"=\${"${input}"} + eval echo \${"${output}"} +} + +# Usage: if [ "$(io_commandExists 'curl')" == "yes" ] +# IMPORTANT- DON'T ADD LOGGING to this method +io_commandExists() { + local commandToExecute="$1" + hash "${commandToExecute}" 2>/dev/null + local rt=$? + if [ "$rt" == 0 ]; then echo -n "yes"; else echo -n "no"; fi +} + +# Usage: if [ "$(io_curlExists)" != "yes" ] +# IMPORTANT- DON'T ADD LOGGING to this method +io_curlExists() { + io_commandExists "curl" +} + + +io_hasMatch() { + logSilly "Method ${FUNCNAME[0]}" + local result=0 + logDebug "Executing [echo \"$1\" | grep \"$2\" >/dev/null 2>&1]" + echo "$1" | grep "$2" >/dev/null 2>&1 || result=1 + return $result +} + +# Utility method to check if the string passed (usually a connection url) corresponds to this machine itself +# Failure conditions: None +# Parameters: +## $1: string to check against +# Depends on global: none +# Updates global: IS_LOCALHOST with value "yes/no" +# Returns: NA + +io_getIsLocalhost() { + logSilly "Method ${FUNCNAME[0]}" + IS_LOCALHOST="$FLAG_N" + local inputString="$1" + logDebug "Parsing [$inputString] to check if we are dealing with this machine itself" + + io_hasMatch "$inputString" "localhost" && { + logDebug "Found localhost. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for localhost" + + local hostIP=$(io_getPublicHostIP) + io_hasMatch "$inputString" "$hostIP" && { + logDebug "Found $hostIP. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for $hostIP" + + local hostID=$(io_getPublicHostID) + io_hasMatch "$inputString" "$hostID" && { + logDebug "Found $hostID. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for $hostID" + + local hostName=$(io_getPublicHostName) + io_hasMatch "$inputString" "$hostName" && { + logDebug "Found $hostName. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for $hostName" + +} + +# Usage: if [ "$(io_tarExists)" != "yes" ] +# IMPORTANT- DON'T ADD LOGGING to this method +io_tarExists() { + io_commandExists "tar" +} + +# IMPORTANT- DON'T ADD LOGGING to this method +io_getPublicHostIP() { + local OS_TYPE=$(uname) + local publicHostIP= + if [ "${OS_TYPE}" == "Darwin" ]; then + ipStatus=$(ifconfig en0 | grep "status" | awk '{print$2}') + if [ "${ipStatus}" == "active" ]; then + publicHostIP=$(ifconfig en0 | grep inet | grep -v inet6 | awk '{print $2}') + else + errorExit "Host IP could not be resolved!" + fi + elif [ "${OS_TYPE}" == "Linux" ]; then + publicHostIP=$(hostname -i 2>/dev/null || echo "127.0.0.1") + fi + publicHostIP=$(echo "${publicHostIP}" | awk '{print $1}') + echo -n "${publicHostIP}" +} + +# Will return the short host name (up to the first dot) +# IMPORTANT- DON'T ADD LOGGING to this method +io_getPublicHostName() { + echo -n "$(hostname -s)" +} + +# Will return the full host name (use this as much as possible) +# IMPORTANT- DON'T ADD LOGGING to this method +io_getPublicHostID() { + echo -n "$(hostname)" +} + +# Utility method to backup a file +# Failure conditions: NA +# Parameters: filePath +# Depends on global: none, +# Updates global: none +# Returns: NA +io_backupFile() { + logSilly "Method ${FUNCNAME[0]}" + fileName="$1" + if [ ! -f "${filePath}" ]; then + logDebug "No file: [${filePath}] to backup" + return + fi + dateTime=$(date +"%Y-%m-%d-%H-%M-%S") + targetFileName="${fileName}.backup.${dateTime}" + yes | \cp -f "$fileName" "${targetFileName}" + logger "File [${fileName}] backedup as [${targetFileName}]" +} + +# Reference https://stackoverflow.com/questions/4023830/how-to-compare-two-strings-in-dot-separated-version-format-in-bash/4025065#4025065 +is_number() { + case "$BASH_VERSION" in + 3.1.*) + PATTERN='\^\[0-9\]+\$' + ;; + *) + PATTERN='^[0-9]+$' + ;; + esac + + [[ "$1" =~ $PATTERN ]] +} + +io_compareVersions() { + if [[ $# != 2 ]] + then + echo "Usage: min_version current minimum" + return + fi + + A="${1%%.*}" + B="${2%%.*}" + + if [[ "$A" != "$1" && "$B" != "$2" && "$A" == "$B" ]] + then + io_compareVersions "${1#*.}" "${2#*.}" + else + if is_number "$A" && is_number "$B" + then + if [[ "$A" -eq "$B" ]]; then + echo "0" + elif [[ "$A" -gt "$B" ]]; then + echo "1" + elif [[ "$A" -lt "$B" ]]; then + echo "-1" + fi + fi + fi +} + +# Reference https://stackoverflow.com/questions/369758/how-to-trim-whitespace-from-a-bash-variable +# Strip all leading and trailing spaces +# IMPORTANT- DON'T ADD LOGGING to this method +io_trim() { + local var="$1" + # remove leading whitespace characters + var="${var#"${var%%[![:space:]]*}"}" + # remove trailing whitespace characters + var="${var%"${var##*[![:space:]]}"}" + echo -n "$var" +} + +# temporary function will be removing it ASAP +# search for string and replace text in file +replaceText_migration_hook () { + local regexString="$1" + local replaceText="$2" + local file="$3" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e "s/${regexString}/${replaceText}/" "${file}" || warn "Failed to replace the text in ${file}" + else + sed -i -e "s/${regexString}/${replaceText}/" "${file}" || warn "Failed to replace the text in ${file}" + fi +} + +# search for string and replace text in file +replaceText () { + local regexString="$1" + local replaceText="$2" + local file="$3" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e "s#${regexString}#${replaceText}#" "${file}" || warn "Failed to replace the text in ${file}" + else + sed -i -e "s#${regexString}#${replaceText}#" "${file}" || warn "Failed to replace the text in ${file}" + logDebug "Replaced [$regexString] with [$replaceText] in [$file]" + fi +} + +# search for string and prepend text in file +prependText () { + local regexString="$1" + local text="$2" + local file="$3" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e '/'"${regexString}"'/i\'$'\n\\'"${text}"''$'\n' "${file}" || warn "Failed to prepend the text in ${file}" + else + sed -i -e '/'"${regexString}"'/i\'$'\n\\'"${text}"''$'\n' "${file}" || warn "Failed to prepend the text in ${file}" + fi +} + +# add text to beginning of the file +addText () { + local text="$1" + local file="$2" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e '1s/^/'"${text}"'\'$'\n/' "${file}" || warn "Failed to add the text in ${file}" + else + sed -i -e '1s/^/'"${text}"'\'$'\n/' "${file}" || warn "Failed to add the text in ${file}" + fi +} + +io_replaceString () { + local value="$1" + local firstString="$2" + local secondString="$3" + local separator=${4:-"/"} + local updateValue= + if [[ $(uname) == "Darwin" ]]; then + updateValue=$(echo "${value}" | sed "s${separator}${firstString}${separator}${secondString}${separator}") + else + updateValue=$(echo "${value}" | sed "s${separator}${firstString}${separator}${secondString}${separator}") + fi + echo -n "${updateValue}" +} + +_findYQ() { + # logSilly "Method ${FUNCNAME[0]}" (Intentionally not logging. Does not add value) + local parentDir="$1" + if [ -z "$parentDir" ]; then + return + fi + logDebug "Executing command [find "${parentDir}" -name third-party -type d]" + local yq=$(find "${parentDir}" -name third-party -type d) + if [ -d "${yq}/yq" ]; then + export YQ_PATH="${yq}/yq" + fi +} + + +io_setYQPath() { + # logSilly "Method ${FUNCNAME[0]}" (Intentionally not logging. Does not add value) + if [ "$(io_commandExists 'yq')" == "yes" ]; then + return + fi + + if [ ! -z "${JF_PRODUCT_HOME}" ] && [ -d "${JF_PRODUCT_HOME}" ]; then + _findYQ "${JF_PRODUCT_HOME}" + fi + + if [ -z "${YQ_PATH}" ] && [ ! -z "${COMPOSE_HOME}" ] && [ -d "${COMPOSE_HOME}" ]; then + _findYQ "${COMPOSE_HOME}" + fi + # TODO We can remove this block after all the code is restructured. + if [ -z "${YQ_PATH}" ] && [ ! -z "${SCRIPT_HOME}" ] && [ -d "${SCRIPT_HOME}" ]; then + _findYQ "${SCRIPT_HOME}" + fi + +} + +io_getLinuxDistribution() { + LINUX_DISTRIBUTION= + + # Make sure running on Linux + [ $(uname -s) != "Linux" ] && return + + # Find out what Linux distribution we are on + + cat /etc/*-release | grep -i Red >/dev/null 2>&1 && LINUX_DISTRIBUTION=RedHat || true + + # OS 6.x + cat /etc/issue.net | grep Red >/dev/null 2>&1 && LINUX_DISTRIBUTION=RedHat || true + + # OS 7.x + cat /etc/*-release | grep -i centos >/dev/null 2>&1 && LINUX_DISTRIBUTION=CentOS && LINUX_DISTRIBUTION_VER="7" || true + + # OS 8.x + grep -q -i "release 8" /etc/redhat-release >/dev/null 2>&1 && LINUX_DISTRIBUTION_VER="8" || true + + # OS 7.x + grep -q -i "release 7" /etc/redhat-release >/dev/null 2>&1 && LINUX_DISTRIBUTION_VER="7" || true + + # OS 6.x + grep -q -i "release 6" /etc/redhat-release >/dev/null 2>&1 && LINUX_DISTRIBUTION_VER="6" || true + + cat /etc/*-release | grep -i Red | grep -i 'VERSION=7' >/dev/null 2>&1 && LINUX_DISTRIBUTION=RedHat && LINUX_DISTRIBUTION_VER="7" || true + + cat /etc/*-release | grep -i debian >/dev/null 2>&1 && LINUX_DISTRIBUTION=Debian || true + + cat /etc/*-release | grep -i ubuntu >/dev/null 2>&1 && LINUX_DISTRIBUTION=Ubuntu || true +} + +## Utility method to check ownership of folders/files +## Failure conditions: + ## If invoked with incorrect inputs - FATAL + ## If file is not owned by the user & group +## Parameters: + ## user + ## group + ## folder to chown +## Globals: none +## Returns: none +## NOTE: The method does NOTHING if the OS is Mac +io_checkOwner () { + logSilly "Method ${FUNCNAME[0]}" + local osType=$(uname) + + if [ "${osType}" != "Linux" ]; then + logDebug "Unsupported OS. Skipping check" + return 0 + fi + + local file_to_check=$1 + local user_id_to_check=$2 + + + if [ -z "$user_id_to_check" ] || [ -z "$file_to_check" ]; then + errorExit "Invalid invocation of method. Missing mandatory inputs" + fi + + local group_id_to_check=${3:-$user_id_to_check} + local check_user_name=${4:-"no"} + + logDebug "Checking permissions on [$file_to_check] for user [$user_id_to_check] & group [$group_id_to_check]" + + local stat= + + if [ "${check_user_name}" == "yes" ]; then + stat=( $(stat -Lc "%U %G" ${file_to_check}) ) + else + stat=( $(stat -Lc "%u %g" ${file_to_check}) ) + fi + + local user_id=${stat[0]} + local group_id=${stat[1]} + + if [[ "${user_id}" != "${user_id_to_check}" ]] || [[ "${group_id}" != "${group_id_to_check}" ]] ; then + logDebug "Ownership mismatch. [${file_to_check}] is not owned by [${user_id_to_check}:${group_id_to_check}]" + return 1 + else + return 0 + fi +} + +## Utility method to change ownership of a file/folder - NON recursive +## Failure conditions: + ## If invoked with incorrect inputs - FATAL + ## If chown operation fails - returns 1 +## Parameters: + ## user + ## group + ## file to chown +## Globals: none +## Returns: none +## NOTE: The method does NOTHING if the OS is Mac + +io_setOwnershipNonRecursive() { + + local osType=$(uname) + if [ "${osType}" != "Linux" ]; then + return + fi + + local targetFile=$1 + local user=$2 + + if [ -z "$user" ] || [ -z "$targetFile" ]; then + errorExit "Invalid invocation of method. Missing mandatory inputs" + fi + + local group=${3:-$user} + logDebug "Method ${FUNCNAME[0]}. Executing [chown ${user}:${group} ${targetFile}]" + chown ${user}:${group} ${targetFile} || return 1 +} + +## Utility method to change ownership of a file. +## IMPORTANT +## If being called on a folder, should ONLY be called for fresh folders or may cause performance issues +## Failure conditions: + ## If invoked with incorrect inputs - FATAL + ## If chown operation fails - returns 1 +## Parameters: + ## user + ## group + ## file to chown +## Globals: none +## Returns: none +## NOTE: The method does NOTHING if the OS is Mac + +io_setOwnership() { + + local osType=$(uname) + if [ "${osType}" != "Linux" ]; then + return + fi + + local targetFile=$1 + local user=$2 + + if [ -z "$user" ] || [ -z "$targetFile" ]; then + errorExit "Invalid invocation of method. Missing mandatory inputs" + fi + + local group=${3:-$user} + logDebug "Method ${FUNCNAME[0]}. Executing [chown -R ${user}:${group} ${targetFile}]" + chown -R ${user}:${group} ${targetFile} || return 1 +} + +## Utility method to create third party folder structure necessary for Postgres +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## POSTGRESQL_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createPostgresDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${POSTGRESQL_DATA_ROOT}" ] && return 0 + + logDebug "Property [${POSTGRESQL_DATA_ROOT}] exists. Proceeding" + + createDir "${POSTGRESQL_DATA_ROOT}/data" + io_setOwnership "${POSTGRESQL_DATA_ROOT}" "${POSTGRES_USER}" "${POSTGRES_USER}" || errorExit "Setting ownership of [${POSTGRESQL_DATA_ROOT}] to [${POSTGRES_USER}:${POSTGRES_USER}] failed" +} + +## Utility method to create third party folder structure necessary for Nginx +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## NGINX_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createNginxDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${NGINX_DATA_ROOT}" ] && return 0 + + logDebug "Property [${NGINX_DATA_ROOT}] exists. Proceeding" + + createDir "${NGINX_DATA_ROOT}" + io_setOwnership "${NGINX_DATA_ROOT}" "${NGINX_USER}" "${NGINX_GROUP}" || errorExit "Setting ownership of [${NGINX_DATA_ROOT}] to [${NGINX_USER}:${NGINX_GROUP}] failed" +} + +## Utility method to create third party folder structure necessary for ElasticSearch +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## ELASTIC_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createElasticSearchDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${ELASTIC_DATA_ROOT}" ] && return 0 + + logDebug "Property [${ELASTIC_DATA_ROOT}] exists. Proceeding" + + createDir "${ELASTIC_DATA_ROOT}/data" + io_setOwnership "${ELASTIC_DATA_ROOT}" "${ES_USER}" "${ES_USER}" || errorExit "Setting ownership of [${ELASTIC_DATA_ROOT}] to [${ES_USER}:${ES_USER}] failed" +} + +## Utility method to create third party folder structure necessary for Redis +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## REDIS_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createRedisDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${REDIS_DATA_ROOT}" ] && return 0 + + logDebug "Property [${REDIS_DATA_ROOT}] exists. Proceeding" + + createDir "${REDIS_DATA_ROOT}" + io_setOwnership "${REDIS_DATA_ROOT}" "${REDIS_USER}" "${REDIS_USER}" || errorExit "Setting ownership of [${REDIS_DATA_ROOT}] to [${REDIS_USER}:${REDIS_USER}] failed" +} + +## Utility method to create third party folder structure necessary for Mongo +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## MONGODB_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createMongoDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${MONGODB_DATA_ROOT}" ] && return 0 + + logDebug "Property [${MONGODB_DATA_ROOT}] exists. Proceeding" + + createDir "${MONGODB_DATA_ROOT}/logs" + createDir "${MONGODB_DATA_ROOT}/configdb" + createDir "${MONGODB_DATA_ROOT}/db" + io_setOwnership "${MONGODB_DATA_ROOT}" "${MONGO_USER}" "${MONGO_USER}" || errorExit "Setting ownership of [${MONGODB_DATA_ROOT}] to [${MONGO_USER}:${MONGO_USER}] failed" +} + +## Utility method to create third party folder structure necessary for RabbitMQ +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## RABBITMQ_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createRabbitMQDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${RABBITMQ_DATA_ROOT}" ] && return 0 + + logDebug "Property [${RABBITMQ_DATA_ROOT}] exists. Proceeding" + + createDir "${RABBITMQ_DATA_ROOT}" + io_setOwnership "${RABBITMQ_DATA_ROOT}" "${RABBITMQ_USER}" "${RABBITMQ_USER}" || errorExit "Setting ownership of [${RABBITMQ_DATA_ROOT}] to [${RABBITMQ_USER}:${RABBITMQ_USER}] failed" +} + +# Add or replace a property in provided properties file +addOrReplaceProperty() { + local propertyName=$1 + local propertyValue=$2 + local propertiesPath=$3 + local delimiter=${4:-"="} + + # Return if any of the inputs are empty + [[ -z "$propertyName" || "$propertyName" == "" ]] && return + [[ -z "$propertyValue" || "$propertyValue" == "" ]] && return + [[ -z "$propertiesPath" || "$propertiesPath" == "" ]] && return + + grep "^${propertyName}\s*${delimiter}.*$" ${propertiesPath} > /dev/null 2>&1 + [ $? -ne 0 ] && echo -e "\n${propertyName}${delimiter}${propertyValue}" >> ${propertiesPath} + sed -i -e "s|^${propertyName}\s*${delimiter}.*$|${propertyName}${delimiter}${propertyValue}|g;" ${propertiesPath} +} + +# Set property only if its not set +io_setPropertyNoOverride(){ + local propertyName=$1 + local propertyValue=$2 + local propertiesPath=$3 + + # Return if any of the inputs are empty + [[ -z "$propertyName" || "$propertyName" == "" ]] && return + [[ -z "$propertyValue" || "$propertyValue" == "" ]] && return + [[ -z "$propertiesPath" || "$propertiesPath" == "" ]] && return + + grep "^${propertyName}:" ${propertiesPath} > /dev/null 2>&1 + if [ $? -ne 0 ]; then + echo -e "${propertyName}: ${propertyValue}" >> ${propertiesPath} || warn "Setting property ${propertyName}: ${propertyValue} in [ ${propertiesPath} ] failed" + else + logger "Skipping update of property : ${propertyName}" >&6 + fi +} + +# Add a line to a file if it doesn't already exist +addLine() { + local line_to_add=$1 + local target_file=$2 + logger "Trying to add line $1 to $2" >&6 2>&1 + cat "$target_file" | grep -F "$line_to_add" -wq >&6 2>&1 + if [ $? != 0 ]; then + logger "Line does not exist and will be added" >&6 2>&1 + echo $line_to_add >> $target_file || errorExit "Could not update $target_file" + fi +} + +# Utility method to check if a value (first parameter) exists in an array (2nd parameter) +# 1st parameter "value to find" +# 2nd parameter "The array to search in. Please pass a string with each value separated by space" +# Example: containsElement "y" "y Y n N" +containsElement () { + local searchElement=$1 + local searchArray=($2) + local found=1 + for elementInIndex in "${searchArray[@]}";do + if [[ $elementInIndex == $searchElement ]]; then + found=0 + fi + done + return $found +} + +# Utility method to get user's choice +# 1st parameter "what to ask the user" +# 2nd parameter "what choices to accept, separated by spaces" +# 3rd parameter "what is the default choice (to use if the user simply presses Enter)" +# Example 'getUserChoice "Are you feeling lucky? Punk!" "y n Y N" "y"' +getUserChoice(){ + configureLogOutput + read_timeout=${read_timeout:-0.5} + local choice="na" + local text_to_display=$1 + local choices=$2 + local default_choice=$3 + users_choice= + + until containsElement "$choice" "$choices"; do + echo "";echo ""; + sleep $read_timeout #This ensures correct placement of the question. + read -p "$text_to_display :" choice + : ${choice:=$default_choice} + done + users_choice=$choice + echo -e "\n$text_to_display: $users_choice" >&6 + sleep $read_timeout #This ensures correct logging +} + +setFilePermission () { + local permission=$1 + local file=$2 + chmod "${permission}" "${file}" || warn "Setting permission ${permission} to file [ ${file} ] failed" +} + + +#setting required paths +setAppDir (){ + SCRIPT_DIR=$(dirname $0) + SCRIPT_HOME="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" + APP_DIR="`cd "${SCRIPT_HOME}";pwd`" +} + +ZIP_TYPE="zip" +COMPOSE_TYPE="compose" +HELM_TYPE="helm" +RPM_TYPE="rpm" +DEB_TYPE="debian" + +sourceScript () { + local file="$1" + + [ ! -z "${file}" ] || errorExit "target file is not passed to source a file" + + if [ ! -f "${file}" ]; then + errorExit "${file} file is not found" + else + source "${file}" || errorExit "Unable to source ${file}, please check if the user ${USER} has permissions to perform this action" + fi +} +# Source required helpers +initHelpers () { + local systemYamlHelper="${APP_DIR}/systemYamlHelper.sh" + local thirdPartyDir=$(find ${APP_DIR}/.. -name third-party -type d) + export YQ_PATH="${thirdPartyDir}/yq" + LIBXML2_PATH="${thirdPartyDir}/libxml2/bin/xmllint" + export LD_LIBRARY_PATH="${thirdPartyDir}/libxml2/lib" + sourceScript "${systemYamlHelper}" +} +# Check migration info yaml file available in the path +checkMigrationInfoYaml () { + + if [[ -f "${APP_DIR}/migrationHelmInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationHelmInfo.yaml" + INSTALLER="${HELM_TYPE}" + elif [[ -f "${APP_DIR}/migrationZipInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationZipInfo.yaml" + INSTALLER="${ZIP_TYPE}" + elif [[ -f "${APP_DIR}/migrationRpmInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationRpmInfo.yaml" + INSTALLER="${RPM_TYPE}" + elif [[ -f "${APP_DIR}/migrationDebInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationDebInfo.yaml" + INSTALLER="${DEB_TYPE}" + elif [[ -f "${APP_DIR}/migrationComposeInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationComposeInfo.yaml" + INSTALLER="${COMPOSE_TYPE}" + else + errorExit "File migration Info yaml does not exist in [${APP_DIR}]" + fi +} + +retrieveYamlValue () { + local yamlPath="$1" + local value="$2" + local output="$3" + local message="$4" + + [[ -z "${yamlPath}" ]] && errorExit "yamlPath is mandatory to get value from ${MIGRATION_SYSTEM_YAML_INFO}" + + getYamlValue "${yamlPath}" "${MIGRATION_SYSTEM_YAML_INFO}" "false" + value="${YAML_VALUE}" + if [[ -z "${value}" ]]; then + if [[ "${output}" == "Warning" ]]; then + warn "Empty value for ${yamlPath} in [${MIGRATION_SYSTEM_YAML_INFO}]" + elif [[ "${output}" == "Skip" ]]; then + return + else + errorExit "${message}" + fi + fi +} + +checkEnv () { + + if [[ "${INSTALLER}" == "${ZIP_TYPE}" ]]; then + # check Environment JF_PRODUCT_HOME is set before migration + NEW_DATA_DIR="$(evalVariable "NEW_DATA_DIR" "JF_PRODUCT_HOME")" + if [[ -z "${NEW_DATA_DIR}" ]]; then + errorExit "Environment variable JF_PRODUCT_HOME is not set, this is required to perform Migration" + fi + # appending var directory to $JF_PRODUCT_HOME + NEW_DATA_DIR="${NEW_DATA_DIR}/var" + elif [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + getCustomDataDir_hook + NEW_DATA_DIR="${OLD_DATA_DIR}" + if [[ -z "${NEW_DATA_DIR}" ]] && [[ -z "${OLD_DATA_DIR}" ]]; then + errorExit "Could not find ${PROMPT_DATA_DIR_LOCATION} to perform Migration" + fi + else + # check Environment JF_ROOT_DATA_DIR is set before migration + OLD_DATA_DIR="$(evalVariable "OLD_DATA_DIR" "JF_ROOT_DATA_DIR")" + # check Environment JF_ROOT_DATA_DIR is set before migration + NEW_DATA_DIR="$(evalVariable "NEW_DATA_DIR" "JF_ROOT_DATA_DIR")" + if [[ -z "${NEW_DATA_DIR}" ]] && [[ -z "${OLD_DATA_DIR}" ]]; then + errorExit "Could not find ${PROMPT_DATA_DIR_LOCATION} to perform Migration" + fi + # appending var directory to $JF_PRODUCT_HOME + NEW_DATA_DIR="${NEW_DATA_DIR}/var" + fi + +} + +getDataDir () { + + if [[ "${INSTALLER}" == "${ZIP_TYPE}" || "${INSTALLER}" == "${COMPOSE_TYPE}"|| "${INSTALLER}" == "${HELM_TYPE}" ]]; then + checkEnv + else + getCustomDataDir_hook + NEW_DATA_DIR="`cd "${APP_DIR}"/../../;pwd`" + NEW_DATA_DIR="${NEW_DATA_DIR}/var" + fi +} + +# Retrieve Product name from MIGRATION_SYSTEM_YAML_INFO +getProduct () { + retrieveYamlValue "migration.product" "${YAML_VALUE}" "Fail" "Empty value under ${yamlPath} in [${MIGRATION_SYSTEM_YAML_INFO}]" + PRODUCT="${YAML_VALUE}" + PRODUCT=$(echo "${PRODUCT}" | tr '[:upper:]' '[:lower:]' 2>/dev/null) + if [[ "${PRODUCT}" != "artifactory" && "${PRODUCT}" != "distribution" && "${PRODUCT}" != "xray" ]]; then + errorExit "migration.product in [${MIGRATION_SYSTEM_YAML_INFO}] is not correct, please set based on product as ARTIFACTORY or DISTRIBUTION" + fi + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + JF_USER="${PRODUCT}" + fi +} +# Compare product version with minProductVersion and maxProductVersion +migrateCheckVersion () { + local productVersion="$1" + local minProductVersion="$2" + local maxProductVersion="$3" + local productVersion618="6.18.0" + local unSupportedProductVersions7=("7.2.0 7.2.1") + + if [[ "$(io_compareVersions "${productVersion}" "${maxProductVersion}")" -eq 0 || "$(io_compareVersions "${productVersion}" "${maxProductVersion}")" -eq 1 ]]; then + logger "Migration not necessary. ${PRODUCT} is already ${productVersion}" + exit 11 + elif [[ "$(io_compareVersions "${productVersion}" "${minProductVersion}")" -eq 0 || "$(io_compareVersions "${productVersion}" "${minProductVersion}")" -eq 1 ]]; then + if [[ ("$(io_compareVersions "${productVersion}" "${productVersion618}")" -eq 0 || "$(io_compareVersions "${productVersion}" "${productVersion618}")" -eq 1) && " ${unSupportedProductVersions7[@]} " =~ " ${CURRENT_VERSION} " ]]; then + touch /tmp/error; + errorExit "Current ${PRODUCT} version (${productVersion}) does not support migration to ${CURRENT_VERSION}" + else + bannerStart "Detected ${PRODUCT} ${productVersion}, initiating migration" + fi + else + logger "Current ${PRODUCT} ${productVersion} version is not supported for migration" + exit 1 + fi +} + +getProductVersion () { + local minProductVersion="$1" + local maxProductVersion="$2" + local newfilePath="$3" + local oldfilePath="$4" + local propertyInDocker="$5" + local property="$6" + local productVersion= + local status= + + if [[ "$INSTALLER" == "${COMPOSE_TYPE}" ]]; then + if [[ -f "${oldfilePath}" ]]; then + if [[ "${PRODUCT}" == "artifactory" ]]; then + productVersion="$(readKey "${property}" "${oldfilePath}")" + else + productVersion="$(cat "${oldfilePath}")" + fi + status="success" + elif [[ -f "${newfilePath}" ]]; then + productVersion="$(readKey "${propertyInDocker}" "${newfilePath}")" + status="fail" + else + logger "File [${oldfilePath}] or [${newfilePath}] not found to get current version." + exit 0 + fi + elif [[ "$INSTALLER" == "${HELM_TYPE}" ]]; then + if [[ -f "${oldfilePath}" ]]; then + if [[ "${PRODUCT}" == "artifactory" ]]; then + productVersion="$(readKey "${property}" "${oldfilePath}")" + else + productVersion="$(cat "${oldfilePath}")" + fi + status="success" + else + productVersion="${CURRENT_VERSION}" + [[ -z "${productVersion}" || "${productVersion}" == "" ]] && logger "${PRODUCT} CURRENT_VERSION is not set" && exit 0 + fi + else + if [[ -f "${newfilePath}" ]]; then + productVersion="$(readKey "${property}" "${newfilePath}")" + status="fail" + elif [[ -f "${oldfilePath}" ]]; then + productVersion="$(readKey "${property}" "${oldfilePath}")" + status="success" + else + if [[ "${INSTALLER}" == "${ZIP_TYPE}" ]]; then + logger "File [${newfilePath}] not found to get current version." + else + logger "File [${oldfilePath}] or [${newfilePath}] not found to get current version." + fi + exit 0 + fi + fi + if [[ -z "${productVersion}" || "${productVersion}" == "" ]]; then + [[ "${status}" == "success" ]] && logger "No version found in file [${oldfilePath}]." + [[ "${status}" == "fail" ]] && logger "No version found in file [${newfilePath}]." + exit 0 + fi + + migrateCheckVersion "${productVersion}" "${minProductVersion}" "${maxProductVersion}" +} + +readKey () { + local property="$1" + local file="$2" + local version= + + while IFS='=' read -r key value || [ -n "${key}" ]; + do + [[ ! "${key}" =~ \#.* && ! -z "${key}" && ! -z "${value}" ]] + key="$(io_trim "${key}")" + if [[ "${key}" == "${property}" ]]; then + version="${value}" && check=true && break + else + check=false + fi + done < "${file}" + if [[ "${check}" == "false" ]]; then + return + fi + echo "${version}" +} + +# create Log directory +createLogDir () { + if [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + getUserAndGroupFromFile + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/log" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" + fi +} + +# Creating migration log file +creationMigrateLog () { + local LOG_FILE_NAME="migration.log" + createLogDir + local MIGRATION_LOG_FILE="${NEW_DATA_DIR}/log/${LOG_FILE_NAME}" + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + MIGRATION_LOG_FILE="${SCRIPT_HOME}/${LOG_FILE_NAME}" + fi + touch "${MIGRATION_LOG_FILE}" + setFilePermission "${LOG_FILE_PERMISSION}" "${MIGRATION_LOG_FILE}" + exec &> >(tee -a "${MIGRATION_LOG_FILE}") +} +# Set path where system.yaml should create +setSystemYamlPath () { + SYSTEM_YAML_PATH="${NEW_DATA_DIR}/etc/system.yaml" + if [[ "${INSTALLER}" != "${HELM_TYPE}" ]]; then + logger "system.yaml will be created in path [${SYSTEM_YAML_PATH}]" + fi +} +# Create directory +createDirectory () { + local directory="$1" + local output="$2" + local check=false + local message="Could not create directory ${directory}, please check if the user ${USER} has permissions to perform this action" + removeSoftLink "${directory}" + mkdir -p "${directory}" && check=true || check=false + if [[ "${check}" == "false" ]]; then + if [[ "${output}" == "Warning" ]]; then + warn "${message}" + else + errorExit "${message}" + fi + fi + setOwnershipBasedOnInstaller "${directory}" +} + +setOwnershipBasedOnInstaller () { + local directory="$1" + if [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + getUserAndGroupFromFile + chown -R ${USER_TO_CHECK}:${GROUP_TO_CHECK} "${directory}" || warn "Setting ownership on $directory failed" + elif [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + io_setOwnership "${directory}" "${JF_USER}" "${JF_USER}" + fi +} + +getUserAndGroup () { + local file="$1" + read uid gid <<<$(stat -c '%U %G' ${file}) + USER_TO_CHECK="${uid}" + GROUP_TO_CHECK="${gid}" +} + +# set ownership +getUserAndGroupFromFile () { + case $PRODUCT in + artifactory) + getUserAndGroup "/etc/opt/jfrog/artifactory/artifactory.properties" + ;; + distribution) + getUserAndGroup "${OLD_DATA_DIR}/etc/versions.properties" + ;; + xray) + getUserAndGroup "${OLD_DATA_DIR}/security/master.key" + ;; + esac +} + +# creating required directories +createRequiredDirs () { + bannerSubSection "CREATING REQUIRED DIRECTORIES" + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/etc/security" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/data" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/log/archived" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/work" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/backup" "${JF_USER}" "${JF_USER}" "yes" + io_setOwnership "${NEW_DATA_DIR}" "${JF_USER}" "${JF_USER}" + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" ]]; then + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/data/postgres" "${POSTGRES_USER}" "${POSTGRES_USER}" "yes" + fi + elif [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + getUserAndGroupFromFile + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/etc" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/etc/security" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/data" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/log/archived" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/work" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/backup" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + fi +} + +# Check entry in map is format +checkMapEntry () { + local entry="$1" + + [[ "${entry}" != *"="* ]] && echo -n "false" || echo -n "true" +} +# Check value Empty and warn +warnIfEmpty () { + local filePath="$1" + local yamlPath="$2" + local check= + + if [[ -z "${filePath}" ]]; then + warn "Empty value in yamlpath [${yamlPath} in [${MIGRATION_SYSTEM_YAML_INFO}]" + check=false + else + check=true + fi + echo "${check}" +} + +logCopyStatus () { + local status="$1" + local logMessage="$2" + local warnMessage="$3" + + [[ "${status}" == "success" ]] && logger "${logMessage}" + [[ "${status}" == "fail" ]] && warn "${warnMessage}" +} +# copy contents from source to destination +copyCmd () { + local source="$1" + local target="$2" + local mode="$3" + local status= + + case $mode in + unique) + cp -up "${source}"/* "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied directory contents from [${source}] to [${target}]" "Failed to copy directory contents from [${source}] to [${target}]" + ;; + specific) + cp -pf "${source}" "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied file [${source}] to [${target}]" "Failed to copy file [${source}] to [${target}]" + ;; + patternFiles) + cp -pf "${source}"* "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied files matching [${source}*] to [${target}]" "Failed to copy files matching [${source}*] to [${target}]" + ;; + full) + cp -prf "${source}"/* "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied directory contents from [${source}] to [${target}]" "Failed to copy directory contents from [${source}] to [${target}]" + ;; + esac +} +# Check contents exist in source before copying +copyOnContentExist () { + local source="$1" + local target="$2" + local mode="$3" + + if [[ "$(checkContentExists "${source}")" == "true" ]]; then + copyCmd "${source}" "${target}" "${mode}" + else + logger "No contents to copy from [${source}]" + fi +} + +# move source to destination +moveCmd () { + local source="$1" + local target="$2" + local status= + + mv -f "${source}" "${target}" && status="success" || status="fail" + [[ "${status}" == "success" ]] && logger "Successfully moved directory [${source}] to [${target}]" + [[ "${status}" == "fail" ]] && warn "Failed to move directory [${source}] to [${target}]" +} + +# symlink target to source +symlinkCmd () { + local source="$1" + local target="$2" + local symlinkSubDir="$3" + local check=false + + if [[ "${symlinkSubDir}" == "subDir" ]]; then + ln -sf "${source}"/* "${target}" && check=true || check=false + else + ln -sf "${source}" "${target}" && check=true || check=false + fi + + [[ "${check}" == "true" ]] && logger "Successfully symlinked directory [${target}] to old [${source}]" + [[ "${check}" == "false" ]] && warn "Symlink operation failed" +} +# Check contents exist in source before symlinking +symlinkOnExist () { + local source="$1" + local target="$2" + local symlinkSubDir="$3" + + if [[ "$(checkContentExists "${source}")" == "true" ]]; then + if [[ "${symlinkSubDir}" == "subDir" ]]; then + symlinkCmd "${source}" "${target}" "subDir" + else + symlinkCmd "${source}" "${target}" + fi + else + logger "No contents to symlink from [${source}]" + fi +} + +prependDir () { + local absolutePath="$1" + local fullPath="$2" + local sourcePath= + + if [[ "${absolutePath}" = \/* ]]; then + sourcePath="${absolutePath}" + else + sourcePath="${fullPath}" + fi + echo "${sourcePath}" +} + +getFirstEntry (){ + local entry="$1" + + [[ -z "${entry}" ]] && return + echo "${entry}" | awk -F"=" '{print $1}' +} + +getSecondEntry () { + local entry="$1" + + [[ -z "${entry}" ]] && return + echo "${entry}" | awk -F"=" '{print $2}' +} +# To get absolutePath +pathResolver () { + local directoryPath="$1" + local dataDir= + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + retrieveYamlValue "migration.oldDataDir" "oldDataDir" "Warning" + dataDir="${YAML_VALUE}" + cd "${dataDir}" + else + cd "${OLD_DATA_DIR}" + fi + absoluteDir="`cd "${directoryPath}";pwd`" + echo "${absoluteDir}" +} + +checkPathResolver () { + local value="$1" + + if [[ "${value}" == \/* ]]; then + value="${value}" + else + value="$(pathResolver "${value}")" + fi + echo "${value}" +} + +propertyMigrate () { + local entry="$1" + local filePath="$2" + local fileName="$3" + local check=false + + local yamlPath="$(getFirstEntry "${entry}")" + local property="$(getSecondEntry "${entry}")" + if [[ -z "${property}" ]]; then + warn "Property is empty in map [${entry}] in the file [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + if [[ -z "${yamlPath}" ]]; then + warn "yamlPath is empty for [${property}] in [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + local keyValues=$(cat "${NEW_DATA_DIR}/${filePath}/${fileName}" | grep "^[^#]" | grep "[*=*]") + for i in ${keyValues}; do + key=$(echo "${i}" | awk -F"=" '{print $1}') + value=$(echo "${i}" | cut -f 2- -d '=') + [ -z "${key}" ] && continue + [ -z "${value}" ] && continue + if [[ "${key}" == "${property}" ]]; then + if [[ "${PRODUCT}" == "artifactory" ]]; then + value="$(migrateResolveDerbyPath "${key}" "${value}")" + value="$(migrateResolveHaDirPath "${key}" "${value}")" + if [[ "${INSTALLER}" != "${DOCKER_TYPE}" ]]; then + value="$(updatePostgresUrlString_Hook "${yamlPath}" "${value}")" + fi + fi + if [[ "${key}" == "context.url" ]]; then + local ip=$(echo "${value}" | awk -F/ '{print $3}' | sed 's/:.*//') + setSystemValue "shared.node.ip" "${ip}" "${SYSTEM_YAML_PATH}" + logger "Setting [shared.node.ip] with [${ip}] in system.yaml" + fi + setSystemValue "${yamlPath}" "${value}" "${SYSTEM_YAML_PATH}" && logger "Setting [${yamlPath}] with value of the property [${property}] in system.yaml" && check=true && break || check=false + fi + done + [[ "${check}" == "false" ]] && logger "Property [${property}] not found in file [${fileName}]" +} + +setHaEnabled_hook () { + echo "" +} + +migratePropertiesFiles () { + local fileList= + local filePath= + local fileName= + local map= + + retrieveYamlValue "migration.propertyFiles.files" "fileList" "Skip" + fileList="${YAML_VALUE}" + if [[ -z "${fileList}" ]]; then + return + fi + bannerSection "PROCESSING MIGRATION OF PROPERTY FILES" + for file in ${fileList}; + do + bannerSubSection "Processing Migration of $file" + retrieveYamlValue "migration.propertyFiles.$file.filePath" "filePath" "Warning" + filePath="${YAML_VALUE}" + retrieveYamlValue "migration.propertyFiles.$file.fileName" "fileName" "Warning" + fileName="${YAML_VALUE}" + [[ -z "${filePath}" && -z "${fileName}" ]] && continue + if [[ "$(checkFileExists "${NEW_DATA_DIR}/${filePath}/${fileName}")" == "true" ]]; then + logger "File [${fileName}] found in path [${NEW_DATA_DIR}/${filePath}]" + # setting haEnabled with true only if ha-node.properties is present + setHaEnabled_hook "${filePath}" + retrieveYamlValue "migration.propertyFiles.$file.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + propertyMigrate "${entry}" "${filePath}" "${fileName}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e yamlPath=property" + fi + done + else + logger "File [${fileName}] was not found in path [${NEW_DATA_DIR}/${filePath}] to migrate" + fi + done +} + +createTargetDir () { + local mountDir="$1" + local target="$2" + + logger "Target directory not found [${mountDir}/${target}], creating it" + createDirectoryRecursive "${mountDir}" "${target}" "Warning" +} + +createDirectoryRecursive () { + local mountDir="$1" + local target="$2" + local output="$3" + local check=false + local message="Could not create directory ${directory}, please check if the user ${USER} has permissions to perform this action" + removeSoftLink "${mountDir}/${target}" + local directory=$(echo "${target}" | tr '/' ' ' ) + local targetDir="${mountDir}" + for dir in ${directory}; + do + targetDir="${targetDir}/${dir}" + mkdir -p "${targetDir}" && check=true || check=false + setOwnershipBasedOnInstaller "${targetDir}" + done + if [[ "${check}" == "false" ]]; then + if [[ "${output}" == "Warning" ]]; then + warn "${message}" + else + errorExit "${message}" + fi + fi +} + +copyOperation () { + local source="$1" + local target="$2" + local mode="$3" + local check=false + local targetDataDir= + local targetLink= + local date= + + # prepend OLD_DATA_DIR only if source is relative path + source="$(prependDir "${source}" "${OLD_DATA_DIR}/${source}")" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + copyLogMessage "${mode}" + #remove source if it is a symlink + if [[ -L "${source}" ]]; then + targetLink=$(readlink -f "${source}") + logger "Removing the symlink [${source}] pointing to [${targetLink}]" + rm -f "${source}" + source=${targetLink} + fi + if [[ "$(checkDirExists "${source}")" != "true" ]]; then + logger "Source [${source}] directory not found in path" + return + fi + if [[ "$(checkDirContents "${source}")" != "true" ]]; then + logger "No contents to copy from [${source}]" + return + fi + if [[ "$(checkDirExists "${targetDataDir}/${target}")" != "true" ]]; then + createTargetDir "${targetDataDir}" "${target}" + fi + copyOnContentExist "${source}" "${targetDataDir}/${target}" "${mode}" +} + +copySpecificFiles () { + local source="$1" + local target="$2" + local mode="$3" + + # prepend OLD_DATA_DIR only if source is relative path + source="$(prependDir "${source}" "${OLD_DATA_DIR}/${source}")" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + copyLogMessage "${mode}" + if [[ "$(checkFileExists "${source}")" != "true" ]]; then + logger "Source file [${source}] does not exist in path" + return + fi + if [[ "$(checkDirExists "${targetDataDir}/${target}")" != "true" ]]; then + createTargetDir "${targetDataDir}" "${target}" + fi + copyCmd "${source}" "${targetDataDir}/${target}" "${mode}" +} + +copyPatternMatchingFiles () { + local source="$1" + local target="$2" + local mode="$3" + local sourcePath="${4}" + + # prepend OLD_DATA_DIR only if source is relative path + sourcePath="$(prependDir "${sourcePath}" "${OLD_DATA_DIR}/${sourcePath}")" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + copyLogMessage "${mode}" + if [[ "$(checkDirExists "${sourcePath}")" != "true" ]]; then + logger "Source [${sourcePath}] directory not found in path" + return + fi + if ls "${sourcePath}/${source}"* 1> /dev/null 2>&1; then + if [[ "$(checkDirExists "${targetDataDir}/${target}")" != "true" ]]; then + createTargetDir "${targetDataDir}" "${target}" + fi + copyCmd "${sourcePath}/${source}" "${targetDataDir}/${target}" "${mode}" + else + logger "Source file [${sourcePath}/${source}*] does not exist in path" + fi +} + +copyLogMessage () { + local mode="$1" + case $mode in + specific) + logger "Copy file [${source}] to target [${targetDataDir}/${target}]" + ;; + patternFiles) + logger "Copy files matching [${sourcePath}/${source}*] to target [${targetDataDir}/${target}]" + ;; + full) + logger "Copy directory contents from source [${source}] to target [${targetDataDir}/${target}]" + ;; + unique) + logger "Copy directory contents from source [${source}] to target [${targetDataDir}/${target}]" + ;; + esac +} + +copyBannerMessages () { + local mode="$1" + local textMode="$2" + case $mode in + specific) + bannerSection "COPY ${textMode} FILES" + ;; + patternFiles) + bannerSection "COPY MATCHING ${textMode}" + ;; + full) + bannerSection "COPY ${textMode} DIRECTORIES CONTENTS" + ;; + unique) + bannerSection "COPY ${textMode} DIRECTORIES CONTENTS" + ;; + esac +} + +invokeCopyFunctions () { + local mode="$1" + local source="$2" + local target="$3" + + case $mode in + specific) + copySpecificFiles "${source}" "${target}" "${mode}" + ;; + patternFiles) + retrieveYamlValue "migration.${copyFormat}.sourcePath" "map" "Warning" + local sourcePath="${YAML_VALUE}" + copyPatternMatchingFiles "${source}" "${target}" "${mode}" "${sourcePath}" + ;; + full) + copyOperation "${source}" "${target}" "${mode}" + ;; + unique) + copyOperation "${source}" "${target}" "${mode}" + ;; + esac +} +# Copies contents from source directory and target directory +copyDataDirectories () { + local copyFormat="$1" + local mode="$2" + local map= + local source= + local target= + local textMode= + local targetDataDir= + local copyFormatValue= + + retrieveYamlValue "migration.${copyFormat}" "${copyFormat}" "Skip" + copyFormatValue="${YAML_VALUE}" + if [[ -z "${copyFormatValue}" ]]; then + return + fi + textMode=$(echo "${mode}" | tr '[:lower:]' '[:upper:]' 2>/dev/null) + copyBannerMessages "${mode}" "${textMode}" + retrieveYamlValue "migration.${copyFormat}.map" "map" "Warning" + map="${YAML_VALUE}" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + source="$(getSecondEntry "${entry}")" + target="$(getFirstEntry "${entry}")" + [[ -z "${source}" ]] && warn "source value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${target}" ]] && warn "target value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + invokeCopyFunctions "${mode}" "${source}" "${target}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e target=source" + fi + echo ""; + done +} + +invokeMoveFunctions () { + local source="$1" + local target="$2" + local sourceDataDir= + local targetBasename= + # prepend OLD_DATA_DIR only if source is relative path + sourceDataDir=$(prependDir "${source}" "${OLD_DATA_DIR}/${source}") + targetBasename=$(dirname "${target}") + logger "Moving directory source [${sourceDataDir}] to target [${NEW_DATA_DIR}/${target}]" + if [[ "$(checkDirExists "${sourceDataDir}")" != "true" ]]; then + logger "Directory [${sourceDataDir}] not found in path to move" + return + fi + if [[ "$(checkDirExists "${NEW_DATA_DIR}/${targetBasename}")" != "true" ]]; then + createTargetDir "${NEW_DATA_DIR}" "${targetBasename}" + moveCmd "${sourceDataDir}" "${NEW_DATA_DIR}/${target}" + else + moveCmd "${sourceDataDir}" "${NEW_DATA_DIR}/tempDir" + moveCmd "${NEW_DATA_DIR}/tempDir" "${NEW_DATA_DIR}/${target}" + fi +} + +# Move source directory and target directory +moveDirectories () { + local moveDataDirectories= + local map= + local source= + local target= + + retrieveYamlValue "migration.moveDirectories" "moveDirectories" "Skip" + moveDirectories="${YAML_VALUE}" + if [[ -z "${moveDirectories}" ]]; then + return + fi + bannerSection "MOVE DIRECTORIES" + retrieveYamlValue "migration.moveDirectories.map" "map" "Warning" + map="${YAML_VALUE}" + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + source="$(getSecondEntry "${entry}")" + target="$(getFirstEntry "${entry}")" + [[ -z "${source}" ]] && warn "source value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${target}" ]] && warn "target value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + invokeMoveFunctions "${source}" "${target}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e target=source" + fi + echo ""; + done +} + +# Trim masterKey if its generated using hex 32 +trimMasterKey () { + local masterKeyDir=/opt/jfrog/artifactory/var/etc/security + local oldMasterKey=$(<${masterKeyDir}/master.key) + local oldMasterKey_Length=$(echo ${#oldMasterKey}) + local newMasterKey= + if [[ ${oldMasterKey_Length} -gt 32 ]]; then + bannerSection "TRIM MASTERKEY" + newMasterKey=$(echo ${oldMasterKey:0:32}) + cp ${masterKeyDir}/master.key ${masterKeyDir}/backup_master.key + logger "Original masterKey is backed up : ${masterKeyDir}/backup_master.key" + rm -rf ${masterKeyDir}/master.key + echo ${newMasterKey} > ${masterKeyDir}/master.key + logger "masterKey is trimmed : ${masterKeyDir}/master.key" + fi +} + +copyDirectories () { + + copyDataDirectories "copyFiles" "full" + copyDataDirectories "copyUniqueFiles" "unique" + copyDataDirectories "copySpecificFiles" "specific" + copyDataDirectories "copyPatternMatchingFiles" "patternFiles" +} + +symlinkDir () { + local source="$1" + local target="$2" + local targetDir= + local basename= + local targetParentDir= + + targetDir="$(dirname "${target}")" + if [[ "${targetDir}" == "${source}" ]]; then + # symlink the sub directories + createDirectory "${NEW_DATA_DIR}/${target}" "Warning" + if [[ "$(checkDirExists "${NEW_DATA_DIR}/${target}")" == "true" ]]; then + symlinkOnExist "${OLD_DATA_DIR}/${source}" "${NEW_DATA_DIR}/${target}" "subDir" + basename="$(basename "${target}")" + cd "${NEW_DATA_DIR}/${target}" && rm -f "${basename}" + fi + else + targetParentDir="$(dirname "${NEW_DATA_DIR}/${target}")" + createDirectory "${targetParentDir}" "Warning" + if [[ "$(checkDirExists "${targetParentDir}")" == "true" ]]; then + symlinkOnExist "${OLD_DATA_DIR}/${source}" "${NEW_DATA_DIR}/${target}" + fi + fi +} + +symlinkOperation () { + local source="$1" + local target="$2" + local check=false + local targetLink= + local date= + + # Check if source is a link and do symlink + if [[ -L "${OLD_DATA_DIR}/${source}" ]]; then + targetLink=$(readlink -f "${OLD_DATA_DIR}/${source}") + symlinkOnExist "${targetLink}" "${NEW_DATA_DIR}/${target}" + else + # check if source is directory and do symlink + if [[ "$(checkDirExists "${OLD_DATA_DIR}/${source}")" != "true" ]]; then + logger "Source [${source}] directory not found in path to symlink" + return + fi + if [[ "$(checkDirContents "${OLD_DATA_DIR}/${source}")" != "true" ]]; then + logger "No contents found in [${OLD_DATA_DIR}/${source}] to symlink" + return + fi + if [[ "$(checkDirExists "${NEW_DATA_DIR}/${target}")" != "true" ]]; then + logger "Target directory [${NEW_DATA_DIR}/${target}] does not exist to create symlink, creating it" + symlinkDir "${source}" "${target}" + else + rm -rf "${NEW_DATA_DIR}/${target}" && check=true || check=false + [[ "${check}" == "false" ]] && warn "Failed to remove contents in [${NEW_DATA_DIR}/${target}/]" + symlinkDir "${source}" "${target}" + fi + fi +} +# Creates a symlink path - Source directory to which the symbolic link should point. +symlinkDirectories () { + local linkFiles= + local map= + local source= + local target= + + retrieveYamlValue "migration.linkFiles" "linkFiles" "Skip" + linkFiles="${YAML_VALUE}" + if [[ -z "${linkFiles}" ]]; then + return + fi + bannerSection "SYMLINK DIRECTORIES" + retrieveYamlValue "migration.linkFiles.map" "map" "Warning" + map="${YAML_VALUE}" + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + source="$(getSecondEntry "${entry}")" + target="$(getFirstEntry "${entry}")" + logger "Symlink directory [${NEW_DATA_DIR}/${target}] to old [${OLD_DATA_DIR}/${source}]" + [[ -z "${source}" ]] && warn "source value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${target}" ]] && warn "target value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + symlinkOperation "${source}" "${target}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e target=source" + fi + echo ""; + done +} + +updateConnectionString () { + local yamlPath="$1" + local value="$2" + local mongoPath="shared.mongo.url" + local rabbitmqPath="shared.rabbitMq.url" + local postgresPath="shared.database.url" + local redisPath="shared.redis.connectionString" + local mongoConnectionString="mongo.connectionString" + local sourceKey= + local hostIp=$(io_getPublicHostIP) + local hostKey= + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + # Replace @postgres:,@mongodb:,@rabbitmq:,@redis: to @{hostIp}: (Compose Installer) + hostKey="@${hostIp}:" + case $yamlPath in + ${postgresPath}) + sourceKey="@postgres:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${mongoPath}) + sourceKey="@mongodb:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${rabbitmqPath}) + sourceKey="@rabbitmq:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${redisPath}) + sourceKey="@redis:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${mongoConnectionString}) + sourceKey="@mongodb:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + esac + fi + echo -n "${value}" +} + +yamlMigrate () { + local entry="$1" + local sourceFile="$2" + local value= + local yamlPath= + local key= + yamlPath="$(getFirstEntry "${entry}")" + key="$(getSecondEntry "${entry}")" + if [[ -z "${key}" ]]; then + warn "key is empty in map [${entry}] in the file [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + if [[ -z "${yamlPath}" ]]; then + warn "yamlPath is empty for [${key}] in [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + getYamlValue "${key}" "${sourceFile}" "false" + value="${YAML_VALUE}" + if [[ ! -z "${value}" ]]; then + value=$(updateConnectionString "${yamlPath}" "${value}") + fi + if [[ -z "${value}" ]]; then + logger "No value for [${key}] in [${sourceFile}]" + else + setSystemValue "${yamlPath}" "${value}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value of the key [${key}] in system.yaml" + fi +} + +migrateYamlFile () { + local files= + local filePath= + local fileName= + local sourceFile= + local map= + retrieveYamlValue "migration.yaml.files" "files" "Skip" + files="${YAML_VALUE}" + if [[ -z "${files}" ]]; then + return + fi + bannerSection "MIGRATION OF YAML FILES" + for file in $files; + do + bannerSubSection "Processing Migration of $file" + retrieveYamlValue "migration.yaml.$file.filePath" "filePath" "Warning" + filePath="${YAML_VALUE}" + retrieveYamlValue "migration.yaml.$file.fileName" "fileName" "Warning" + fileName="${YAML_VALUE}" + [[ -z "${filePath}" && -z "${fileName}" ]] && continue + sourceFile="${NEW_DATA_DIR}/${filePath}/${fileName}" + if [[ "$(checkFileExists "${sourceFile}")" == "true" ]]; then + logger "File [${fileName}] found in path [${NEW_DATA_DIR}/${filePath}]" + retrieveYamlValue "migration.yaml.$file.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + yamlMigrate "${entry}" "${sourceFile}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e yamlPath=key" + fi + done + else + logger "File [${fileName}] is not found in path [${NEW_DATA_DIR}/${filePath}] to migrate" + fi + done +} +# updates the key and value in system.yaml +updateYamlKeyValue () { + local entry="$1" + local value= + local yamlPath= + local key= + + yamlPath="$(getFirstEntry "${entry}")" + value="$(getSecondEntry "${entry}")" + if [[ -z "${value}" ]]; then + warn "value is empty in map [${entry}] in the file [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + if [[ -z "${yamlPath}" ]]; then + warn "yamlPath is empty for [${key}] in [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + setSystemValue "${yamlPath}" "${value}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value [${value}] in system.yaml" +} + +updateSystemYamlFile () { + local updateYaml= + local map= + + retrieveYamlValue "migration.updateSystemYaml" "updateYaml" "Skip" + updateSystemYaml="${YAML_VALUE}" + if [[ -z "${updateSystemYaml}" ]]; then + return + fi + bannerSection "UPDATE SYSTEM YAML FILE WITH KEY AND VALUES" + retrieveYamlValue "migration.updateSystemYaml.map" "map" "Warning" + map="${YAML_VALUE}" + if [[ -z "${map}" ]]; then + return + fi + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + updateYamlKeyValue "${entry}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e yamlPath=key" + fi + done +} + +backupFiles_hook () { + logSilly "Method ${FUNCNAME[0]}" +} + +backupDirectory () { + local backupDir="$1" + local dir="$2" + local targetDir="$3" + local effectiveUser= + local effectiveGroup= + + if [[ "${dir}" = \/* ]]; then + dir=$(echo "${dir/\//}") + fi + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + effectiveUser="${JF_USER}" + effectiveGroup="${JF_USER}" + elif [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + effectiveUser="${USER_TO_CHECK}" + effectiveGroup="${GROUP_TO_CHECK}" + fi + + removeSoftLinkAndCreateDir "${backupDir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local backupDirectory="${backupDir}/${PRODUCT}" + removeSoftLinkAndCreateDir "${backupDirectory}" "${effectiveUser}" "${effectiveGroup}" "yes" + removeSoftLinkAndCreateDir "${backupDirectory}/${dir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local outputCheckDirExists="$(checkDirExists "${backupDirectory}/${dir}")" + if [[ "${outputCheckDirExists}" == "true" ]]; then + copyOnContentExist "${targetDir}" "${backupDirectory}/${dir}" "full" + fi +} + +removeOldDirectory () { + local backupDir="$1" + local entry="$2" + local check=false + + # prepend OLD_DATA_DIR only if entry is relative path + local targetDir="$(prependDir "${entry}" "${OLD_DATA_DIR}/${entry}")" + local outputCheckDirExists="$(checkDirExists "${targetDir}")" + if [[ "${outputCheckDirExists}" != "true" ]]; then + logger "No [${targetDir}] directory found to delete" + echo ""; + return + fi + backupDirectory "${backupDir}" "${entry}" "${targetDir}" + rm -rf "${targetDir}" && check=true || check=false + [[ "${check}" == "true" ]] && logger "Successfully removed directory [${targetDir}]" + [[ "${check}" == "false" ]] && warn "Failed to remove directory [${targetDir}]" + echo ""; +} + +cleanUpOldDataDirectories () { + local cleanUpOldDataDir= + local map= + local entry= + + retrieveYamlValue "migration.cleanUpOldDataDir" "cleanUpOldDataDir" "Skip" + cleanUpOldDataDir="${YAML_VALUE}" + if [[ -z "${cleanUpOldDataDir}" ]]; then + return + fi + bannerSection "CLEAN UP OLD DATA DIRECTORIES" + retrieveYamlValue "migration.cleanUpOldDataDir.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + date="$(date +%Y%m%d%H%M)" + backupDir="${NEW_DATA_DIR}/backup/backup-${date}" + bannerImportant "****** Old data configurations are backedup in [${backupDir}] directory ******" + backupFiles_hook "${backupDir}/${PRODUCT}" + for entry in $map; + do + removeOldDirectory "${backupDir}" "${entry}" + done +} + +backupFiles () { + local backupDir="$1" + local dir="$2" + local targetDir="$3" + local fileName="$4" + local effectiveUser= + local effectiveGroup= + + if [[ "${dir}" = \/* ]]; then + dir=$(echo "${dir/\//}") + fi + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + effectiveUser="${JF_USER}" + effectiveGroup="${JF_USER}" + elif [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + effectiveUser="${USER_TO_CHECK}" + effectiveGroup="${GROUP_TO_CHECK}" + fi + + removeSoftLinkAndCreateDir "${backupDir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local backupDirectory="${backupDir}/${PRODUCT}" + removeSoftLinkAndCreateDir "${backupDirectory}" "${effectiveUser}" "${effectiveGroup}" "yes" + removeSoftLinkAndCreateDir "${backupDirectory}/${dir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local outputCheckDirExists="$(checkDirExists "${backupDirectory}/${dir}")" + if [[ "${outputCheckDirExists}" == "true" ]]; then + copyCmd "${targetDir}/${fileName}" "${backupDirectory}/${dir}" "specific" + fi +} + +removeOldFiles () { + local backupDir="$1" + local directoryName="$2" + local fileName="$3" + local check=false + + # prepend OLD_DATA_DIR only if entry is relative path + local targetDir="$(prependDir "${directoryName}" "${OLD_DATA_DIR}/${directoryName}")" + local outputCheckFileExists="$(checkFileExists "${targetDir}/${fileName}")" + if [[ "${outputCheckFileExists}" != "true" ]]; then + logger "No [${targetDir}/${fileName}] file found to delete" + return + fi + backupFiles "${backupDir}" "${directoryName}" "${targetDir}" "${fileName}" + rm -f "${targetDir}/${fileName}" && check=true || check=false + [[ "${check}" == "true" ]] && logger "Successfully removed file [${targetDir}/${fileName}]" + [[ "${check}" == "false" ]] && warn "Failed to remove file [${targetDir}/${fileName}]" + echo ""; +} + +cleanUpOldFiles () { + local cleanUpFiles= + local map= + local entry= + + retrieveYamlValue "migration.cleanUpOldFiles" "cleanUpOldFiles" "Skip" + cleanUpOldFiles="${YAML_VALUE}" + if [[ -z "${cleanUpOldFiles}" ]]; then + return + fi + bannerSection "CLEAN UP OLD FILES" + retrieveYamlValue "migration.cleanUpOldFiles.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + date="$(date +%Y%m%d%H%M)" + backupDir="${NEW_DATA_DIR}/backup/backup-${date}" + bannerImportant "****** Old files are backedup in [${backupDir}] directory ******" + for entry in $map; + do + local outputCheckMapEntry="$(checkMapEntry "${entry}")" + if [[ "${outputCheckMapEntry}" != "true" ]]; then + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e directoryName=fileName" + fi + local fileName="$(getSecondEntry "${entry}")" + local directoryName="$(getFirstEntry "${entry}")" + [[ -z "${fileName}" ]] && warn "File name value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${directoryName}" ]] && warn "Directory name value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + removeOldFiles "${backupDir}" "${directoryName}" "${fileName}" + echo ""; + done +} + +startMigration () { + bannerSection "STARTING MIGRATION" +} + +endMigration () { + bannerSection "MIGRATION COMPLETED SUCCESSFULLY" +} + +initialize () { + setAppDir + _pauseExecution "setAppDir" + initHelpers + _pauseExecution "initHelpers" + checkMigrationInfoYaml + _pauseExecution "checkMigrationInfoYaml" + getProduct + _pauseExecution "getProduct" + getDataDir + _pauseExecution "getDataDir" +} + +main () { + case $PRODUCT in + artifactory) + migrateArtifactory + ;; + distribution) + migrateDistribution + ;; + xray) + migrationXray + ;; + esac + exit 0 +} + +# Ensures meta data is logged +LOG_BEHAVIOR_ADD_META="$FLAG_Y" + + +migrateResolveDerbyPath () { + local key="$1" + local value="$2" + + if [[ "${key}" == "url" && "${value}" == *"db.home"* ]]; then + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" ]]; then + derbyPath="/opt/jfrog/artifactory/var/data/artifactory/derby" + value=$(echo "${value}" | sed "s|{db.home}|$derbyPath|") + else + derbyPath="${NEW_DATA_DIR}/data/artifactory/derby" + value=$(echo "${value}" | sed "s|{db.home}|$derbyPath|") + fi + fi + echo "${value}" +} + +migrateResolveHaDirPath () { + local key="$1" + local value="$2" + + if [[ "${INSTALLER}" == "${RPM_TYPE}" || "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" || "${INSTALLER}" == "${DEB_TYPE}" ]]; then + if [[ "${key}" == "artifactory.ha.data.dir" || "${key}" == "artifactory.ha.backup.dir" ]]; then + value=$(checkPathResolver "${value}") + fi + fi + echo "${value}" +} +updatePostgresUrlString_Hook () { + local yamlPath="$1" + local value="$2" + local hostIp=$(io_getPublicHostIP) + local sourceKey="//postgresql:" + if [[ "${yamlPath}" == "shared.database.url" ]]; then + value=$(io_replaceString "${value}" "${sourceKey}" "//${hostIp}:" "#") + fi + echo "${value}" +} +# Check Artifactory product version +checkArtifactoryVersion () { + local minProductVersion="6.0.0" + local maxProductVersion="7.0.0" + local propertyInDocker="ARTIFACTORY_VERSION" + local property="artifactory.version" + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" ]]; then + local newfilePath="${APP_DIR}/../.env" + local oldfilePath="${OLD_DATA_DIR}/etc/artifactory.properties" + elif [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + local oldfilePath="${OLD_DATA_DIR}/etc/artifactory.properties" + elif [[ "${INSTALLER}" == "${ZIP_TYPE}" ]]; then + local newfilePath="${NEW_DATA_DIR}/etc/artifactory/artifactory.properties" + local oldfilePath="${OLD_DATA_DIR}/etc/artifactory.properties" + else + local newfilePath="${NEW_DATA_DIR}/etc/artifactory/artifactory.properties" + local oldfilePath="/etc/opt/jfrog/artifactory/artifactory.properties" + fi + + getProductVersion "${minProductVersion}" "${maxProductVersion}" "${newfilePath}" "${oldfilePath}" "${propertyInDocker}" "${property}" +} + +getCustomDataDir_hook () { + retrieveYamlValue "migration.oldDataDir" "oldDataDir" "Fail" + OLD_DATA_DIR="${YAML_VALUE}" +} + +# Get protocol value of connector +getXmlConnectorProtocol () { + local i="$1" + local filePath="$2" + local fileName="$3" + local protocolValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@protocol' ${filePath}/${fileName} 2>/dev/null |awk -F"=" '{print $2}' | tr -d '"') + echo -e "${protocolValue}" +} + +# Get all attributes of connector +getXmlConnectorAttributes () { + local i="$1" + local filePath="$2" + local fileName="$3" + local connectorAttributes=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@*' ${filePath}/${fileName} 2>/dev/null) + # strip leading and trailing spaces + connectorAttributes=$(io_trim "${connectorAttributes}") + echo "${connectorAttributes}" +} + +# Get port value of connector +getXmlConnectorPort () { + local i="$1" + local filePath="$2" + local fileName="$3" + local portValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@port' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + echo -e "${portValue}" +} + +# Get maxThreads value of connector +getXmlConnectorMaxThreads () { + local i="$1" + local filePath="$2" + local fileName="$3" + local maxThreadValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@maxThreads' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + echo -e "${maxThreadValue}" +} +# Get sendReasonPhrase value of connector +getXmlConnectorSendReasonPhrase () { + local i="$1" + local filePath="$2" + local fileName="$3" + local sendReasonPhraseValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@sendReasonPhrase' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + echo -e "${sendReasonPhraseValue}" +} +# Get relaxedPathChars value of connector +getXmlConnectorRelaxedPathChars () { + local i="$1" + local filePath="$2" + local fileName="$3" + local relaxedPathCharsValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@relaxedPathChars' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + # strip leading and trailing spaces + relaxedPathCharsValue=$(io_trim "${relaxedPathCharsValue}") + echo -e "${relaxedPathCharsValue}" +} +# Get relaxedQueryChars value of connector +getXmlConnectorRelaxedQueryChars () { + local i="$1" + local filePath="$2" + local fileName="$3" + local relaxedQueryCharsValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@relaxedQueryChars' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + # strip leading and trailing spaces + relaxedQueryCharsValue=$(io_trim "${relaxedQueryCharsValue}") + echo -e "${relaxedQueryCharsValue}" +} + +# Updating system.yaml with Connector port +setConnectorPort () { + local yamlPath="$1" + local valuePort="$2" + local portYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${valuePort}" ]]; then + warn "port value is empty, could not migrate to system.yaml" + return + fi + ## Getting port yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" portYamlPath "Warning" + portYamlPath="${YAML_VALUE}" + if [[ -z "${portYamlPath}" ]]; then + return + fi + setSystemValue "${portYamlPath}" "${valuePort}" "${SYSTEM_YAML_PATH}" + logger "Setting [${portYamlPath}] with value [${valuePort}] in system.yaml" +} + +# Updating system.yaml with Connector maxThreads +setConnectorMaxThread () { + local yamlPath="$1" + local threadValue="$2" + local maxThreadYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${threadValue}" ]]; then + return + fi + ## Getting max Threads yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" maxThreadYamlPath "Warning" + maxThreadYamlPath="${YAML_VALUE}" + if [[ -z "${maxThreadYamlPath}" ]]; then + return + fi + setSystemValue "${maxThreadYamlPath}" "${threadValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${maxThreadYamlPath}] with value [${threadValue}] in system.yaml" +} + +# Updating system.yaml with Connector sendReasonPhrase +setConnectorSendReasonPhrase () { + local yamlPath="$1" + local sendReasonPhraseValue="$2" + local sendReasonPhraseYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${sendReasonPhraseValue}" ]]; then + return + fi + ## Getting sendReasonPhrase yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" sendReasonPhraseYamlPath "Warning" + sendReasonPhraseYamlPath="${YAML_VALUE}" + if [[ -z "${sendReasonPhraseYamlPath}" ]]; then + return + fi + setSystemValue "${sendReasonPhraseYamlPath}" "${sendReasonPhraseValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${sendReasonPhraseYamlPath}] with value [${sendReasonPhraseValue}] in system.yaml" +} + +# Updating system.yaml with Connector relaxedPathChars +setConnectorRelaxedPathChars () { + local yamlPath="$1" + local relaxedPathCharsValue="$2" + local relaxedPathCharsYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${relaxedPathCharsValue}" ]]; then + return + fi + ## Getting relaxedPathChars yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" relaxedPathCharsYamlPath "Warning" + relaxedPathCharsYamlPath="${YAML_VALUE}" + if [[ -z "${relaxedPathCharsYamlPath}" ]]; then + return + fi + setSystemValue "${relaxedPathCharsYamlPath}" "${relaxedPathCharsValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${relaxedPathCharsYamlPath}] with value [${relaxedPathCharsValue}] in system.yaml" +} + +# Updating system.yaml with Connector relaxedQueryChars +setConnectorRelaxedQueryChars () { + local yamlPath="$1" + local relaxedQueryCharsValue="$2" + local relaxedQueryCharsYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${relaxedQueryCharsValue}" ]]; then + return + fi + ## Getting relaxedQueryChars yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" relaxedQueryCharsYamlPath "Warning" + relaxedQueryCharsYamlPath="${YAML_VALUE}" + if [[ -z "${relaxedQueryCharsYamlPath}" ]]; then + return + fi + setSystemValue "${relaxedQueryCharsYamlPath}" "${relaxedQueryCharsValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${relaxedQueryCharsYamlPath}] with value [${relaxedQueryCharsValue}] in system.yaml" +} + +# Updating system.yaml with Connectors configurations +setConnectorExtraConfig () { + local yamlPath="$1" + local connectorAttributes="$2" + local extraConfigPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${connectorAttributes}" ]]; then + return + fi + ## Getting extraConfig yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" extraConfig "Warning" + extraConfigPath="${YAML_VALUE}" + if [[ -z "${extraConfigPath}" ]]; then + return + fi + # strip leading and trailing spaces + connectorAttributes=$(io_trim "${connectorAttributes}") + setSystemValue "${extraConfigPath}" "${connectorAttributes}" "${SYSTEM_YAML_PATH}" + logger "Setting [${extraConfigPath}] with connector attributes in system.yaml" +} + +# Updating system.yaml with extra Connectors +setExtraConnector () { + local yamlPath="$1" + local extraConnector="$2" + local extraConnectorYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${extraConnector}" ]]; then + return + fi + ## Getting extraConnecotr yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" extraConnectorYamlPath "Warning" + extraConnectorYamlPath="${YAML_VALUE}" + if [[ -z "${extraConnectorYamlPath}" ]]; then + return + fi + getYamlValue "${extraConnectorYamlPath}" "${SYSTEM_YAML_PATH}" "false" + local connectorExtra="${YAML_VALUE}" + if [[ -z "${connectorExtra}" ]]; then + setSystemValue "${extraConnectorYamlPath}" "${extraConnector}" "${SYSTEM_YAML_PATH}" + logger "Setting [${extraConnectorYamlPath}] with extra connectors in system.yaml" + else + setSystemValue "${extraConnectorYamlPath}" "\"${connectorExtra} ${extraConnector}\"" "${SYSTEM_YAML_PATH}" + logger "Setting [${extraConnectorYamlPath}] with extra connectors in system.yaml" + fi +} + +# Migrate extra connectors to system.yaml +migrateExtraConnectors () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local excludeDefaultPort="$4" + local i="$5" + local extraConfig= + local extraConnector= + if [[ "${excludeDefaultPort}" == "yes" ]]; then + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + [[ "${portValue}" != "${DEFAULT_ACCESS_PORT}" && "${portValue}" != "${DEFAULT_RT_PORT}" ]] || continue + extraConnector=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']' ${filePath}/${fileName} 2>/dev/null) + setExtraConnector "${EXTRA_CONFIG_YAMLPATH}" "${extraConnector}" + done + else + extraConnector=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']' ${filePath}/${fileName} 2>/dev/null) + setExtraConnector "${EXTRA_CONFIG_YAMLPATH}" "${extraConnector}" + fi +} + +# Migrate connector configurations +migrateConnectorConfig () { + local i="$1" + local protocolType="$2" + local portValue="$3" + local connectorPortYamlPath="$4" + local connectorMaxThreadYamlPath="$5" + local connectorAttributesYamlPath="$6" + local filePath="$7" + local fileName="$8" + local connectorSendReasonPhraseYamlPath="$9" + local connectorRelaxedPathCharsYamlPath="${10}" + local connectorRelaxedQueryCharsYamlPath="${11}" + + # migrate port + setConnectorPort "${connectorPortYamlPath}" "${portValue}" + + # migrate maxThreads + local maxThreadValue=$(getXmlConnectorMaxThreads "$i" "${filePath}" "${fileName}") + setConnectorMaxThread "${connectorMaxThreadYamlPath}" "${maxThreadValue}" + + # migrate sendReasonPhrase + local sendReasonPhraseValue=$(getXmlConnectorSendReasonPhrase "$i" "${filePath}" "${fileName}") + setConnectorSendReasonPhrase "${connectorSendReasonPhraseYamlPath}" "${sendReasonPhraseValue}" + + # migrate relaxedPathChars + local relaxedPathCharsValue=$(getXmlConnectorRelaxedPathChars "$i" "${filePath}" "${fileName}") + setConnectorRelaxedPathChars "${connectorRelaxedPathCharsYamlPath}" "\"${relaxedPathCharsValue}\"" + # migrate relaxedQueryChars + local relaxedQueryCharsValue=$(getXmlConnectorRelaxedQueryChars "$i" "${filePath}" "${fileName}") + setConnectorRelaxedQueryChars "${connectorRelaxedQueryCharsYamlPath}" "\"${relaxedQueryCharsValue}\"" + + # migrate all attributes to extra config except port , maxThread , sendReasonPhrase ,relaxedPathChars and relaxedQueryChars + local connectorAttributes=$(getXmlConnectorAttributes "$i" "${filePath}" "${fileName}") + connectorAttributes=$(echo "${connectorAttributes}" | sed 's/port="'${portValue}'"//g' | sed 's/maxThreads="'${maxThreadValue}'"//g' | sed 's/sendReasonPhrase="'${sendReasonPhraseValue}'"//g' | sed 's/relaxedPathChars="\'${relaxedPathCharsValue}'\"//g' | sed 's/relaxedQueryChars="\'${relaxedQueryCharsValue}'\"//g') + # strip leading and trailing spaces + connectorAttributes=$(io_trim "${connectorAttributes}") + setConnectorExtraConfig "${connectorAttributesYamlPath}" "${connectorAttributes}" +} + +# Check for default port 8040 and 8081 in connectors and migrate +migrateConnectorPort () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local defaultPort="$4" + local connectorPortYamlPath="$5" + local connectorMaxThreadYamlPath="$6" + local connectorAttributesYamlPath="$7" + local connectorSendReasonPhraseYamlPath="$8" + local connectorRelaxedPathCharsYamlPath="$9" + local connectorRelaxedQueryCharsYamlPath="${10}" + local portYamlPath= + local maxThreadYamlPath= + local status= + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${protocolType}" == *AJP* ]] && continue + [[ "${portValue}" != "${defaultPort}" ]] && continue + if [[ "${portValue}" == "${DEFAULT_RT_PORT}" ]]; then + RT_DEFAULTPORT_STATUS=success + else + AC_DEFAULTPORT_STATUS=success + fi + migrateConnectorConfig "${i}" "${protocolType}" "${portValue}" "${connectorPortYamlPath}" "${connectorMaxThreadYamlPath}" "${connectorAttributesYamlPath}" "${filePath}" "${fileName}" "${connectorSendReasonPhraseYamlPath}" "${connectorRelaxedPathCharsYamlPath}" "${connectorRelaxedQueryCharsYamlPath}" + done +} + +# migrate to extra, connector having default port and protocol is AJP +migrateDefaultPortIfAjp () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local defaultPort="$4" + + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${protocolType}" != *AJP* ]] && continue + [[ "${portValue}" != "${defaultPort}" ]] && continue + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "no" "${i}" + done + +} + +# Comparing max threads in connectors +compareMaxThreads () { + local firstConnectorMaxThread="$1" + local firstConnectorNode="$2" + local secondConnectorMaxThread="$3" + local secondConnectorNode="$4" + local filePath="$5" + local fileName="$6" + + # choose higher maxThreads connector as Artifactory. + if [[ "${firstConnectorMaxThread}" -gt ${secondConnectorMaxThread} || "${firstConnectorMaxThread}" -eq ${secondConnectorMaxThread} ]]; then + # maxThread is higher in firstConnector, + # Taking firstConnector as Artifactory and SecondConnector as Access + # maxThread is equal in both connector,considering firstConnector as Artifactory and SecondConnector as Access + local rtPortValue=$(getXmlConnectorPort "${firstConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${firstConnectorNode}" "${protocolType}" "${rtPortValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + local acPortValue=$(getXmlConnectorPort "${secondConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${secondConnectorNode}" "${protocolType}" "${acPortValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + else + # maxThread is higher in SecondConnector, + # Taking SecondConnector as Artifactory and firstConnector as Access + local rtPortValue=$(getXmlConnectorPort "${secondConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${secondConnectorNode}" "${protocolType}" "${rtPortValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + local acPortValue=$(getXmlConnectorPort "${firstConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${firstConnectorNode}" "${protocolType}" "${acPortValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + fi +} + +# Check max threads exist to compare +maxThreadsExistToCompare () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local firstConnectorMaxThread= + local secondConnectorMaxThread= + local firstConnectorNode= + local secondConnectorNode= + local status=success + local firstnode=fail + + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + if [[ ${protocolType} == *AJP* ]]; then + # Migrate Connectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "no" "${i}" + continue + fi + # store maxthreads value of each connector + if [[ ${firstnode} == "fail" ]]; then + firstConnectorMaxThread=$(getXmlConnectorMaxThreads "${i}" "${filePath}" "${fileName}") + firstConnectorNode="${i}" + firstnode=success + else + secondConnectorMaxThread=$(getXmlConnectorMaxThreads "${i}" "${filePath}" "${fileName}") + secondConnectorNode="${i}" + fi + done + [[ -z "${firstConnectorMaxThread}" ]] && status=fail + [[ -z "${secondConnectorMaxThread}" ]] && status=fail + # maxThreads is set, now compare MaxThreads + if [[ "${status}" == "success" ]]; then + compareMaxThreads "${firstConnectorMaxThread}" "${firstConnectorNode}" "${secondConnectorMaxThread}" "${secondConnectorNode}" "${filePath}" "${fileName}" + else + # Assume first connector is RT, maxThreads is not set in both connectors + local rtPortValue=$(getXmlConnectorPort "${firstConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${firstConnectorNode}" "${protocolType}" "${rtPortValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + local acPortValue=$(getXmlConnectorPort "${secondConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${secondConnectorNode}" "${protocolType}" "${acPortValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + fi +} + +migrateExtraBasedOnNonAjpCount () { + local nonAjpCount="$1" + local filePath="$2" + local fileName="$3" + local connectorCount="$4" + local i="$5" + + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + if [[ "${protocolType}" == *AJP* ]]; then + if [[ "${nonAjpCount}" -eq 1 ]]; then + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "no" "${i}" + continue + else + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + continue + fi + fi +} + +# find RT and AC Connector +findRtAndAcConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local initialAjpCount=0 + local nonAjpCount=0 + + # get the count of non AJP + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${protocolType}" != *AJP* ]] || continue + nonAjpCount=$((initialAjpCount+1)) + initialAjpCount="${nonAjpCount}" + done + if [[ "${nonAjpCount}" -eq 1 ]]; then + # Add the connector found as access and artifactory connectors + # Mark port as 8040 for access + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + done + elif [[ "${nonAjpCount}" -eq 2 ]]; then + # compare maxThreads in both connectors + maxThreadsExistToCompare "${filePath}" "${fileName}" "${connectorCount}" + elif [[ "${nonAjpCount}" -gt 2 ]]; then + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + elif [[ "${nonAjpCount}" -eq 0 ]]; then + # setting with default port in system.yaml + setConnectorPort "${RT_PORT_YAMLPATH}" "${DEFAULT_RT_PORT}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + fi +} + +# get the count of non AJP +getCountOfNonAjp () { + local port="$1" + local connectorCount="$2" + local filePath=$3 + local fileName=$4 + local initialNonAjpCount=0 + + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${portValue}" != "${port}" ]] || continue + [[ "${protocolType}" != *AJP* ]] || continue + local nonAjpCount=$((initialNonAjpCount+1)) + initialNonAjpCount="${nonAjpCount}" + done + echo -e "${nonAjpCount}" +} + +# Find for access connector +findAcConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + + # get the count of non AJP + local nonAjpCount=$(getCountOfNonAjp "${DEFAULT_RT_PORT}" "${connectorCount}" "${filePath}" "${fileName}") + if [[ "${nonAjpCount}" -eq 1 ]]; then + # Add the connector found as access connector and mark port as that of connector + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" != "${DEFAULT_RT_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + fi + done + elif [[ "${nonAjpCount}" -gt 1 ]]; then + # Take RT properties into access with 8040 + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" == "${DEFAULT_RT_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + fi + done + elif [[ "${nonAjpCount}" -eq 0 ]]; then + # Add RT connector details as access connector and mark port as 8040 + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_RT_PORT}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${AC_SENDREASONPHRASE_YAMLPATH}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + fi +} + +# Find for artifactory connector +findRtConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + + # get the count of non AJP + local nonAjpCount=$(getCountOfNonAjp "${DEFAULT_ACCESS_PORT}" "${connectorCount}" "${filePath}" "${fileName}") + if [[ "${nonAjpCount}" -eq 1 ]]; then + # Add the connector found as RT connector + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" != "${DEFAULT_ACCESS_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + fi + done + elif [[ "${nonAjpCount}" -gt 1 ]]; then + # Take access properties into artifactory with 8081 + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" == "${DEFAULT_ACCESS_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + setConnectorPort "${RT_PORT_YAMLPATH}" "${DEFAULT_RT_PORT}" + fi + done + elif [[ "${nonAjpCount}" -eq 0 ]]; then + # Add access connector details as RT connector and mark as ${DEFAULT_RT_PORT} + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_ACCESS_PORT}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + setConnectorPort "${RT_PORT_YAMLPATH}" "${DEFAULT_RT_PORT}" + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + fi +} + +checkForTlsConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + local sslProtocolValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@sslProtocol' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + if [[ "${sslProtocolValue}" == "TLS" ]]; then + bannerImportant "NOTE: Ignoring TLS connector during migration, modify the system yaml to enable TLS. Original server.xml is saved in path [${filePath}/${fileName}]" + TLS_CONNECTOR_EXISTS=${FLAG_Y} + continue + fi + done +} + +# set custom tomcat server Listeners to system.yaml +setListenerConnector () { + local filePath="$1" + local fileName="$2" + local listenerCount="$3" + for ((i = 1 ; i <= "${listenerCount}" ; i++)) + do + local listenerConnector=$($LIBXML2_PATH --xpath '//Server/Listener['$i']' ${filePath}/${fileName} 2>/dev/null) + local listenerClassName=$($LIBXML2_PATH --xpath '//Server/Listener['$i']/@className' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + if [[ "${listenerClassName}" == *Apr* ]]; then + setExtraConnector "${EXTRA_LISTENER_CONFIG_YAMLPATH}" "${listenerConnector}" + fi + done +} +# add custom tomcat server Listeners +addTomcatServerListeners () { + local filePath="$1" + local fileName="$2" + local listenerCount="$3" + if [[ "${listenerCount}" == "0" ]]; then + logger "No listener connectors found in the [${filePath}/${fileName}],skipping migration of listener connectors" + else + setListenerConnector "${filePath}" "${fileName}" "${listenerCount}" + setSystemValue "${RT_TOMCAT_HTTPSCONNECTOR_ENABLED}" "true" "${SYSTEM_YAML_PATH}" + logger "Setting [${RT_TOMCAT_HTTPSCONNECTOR_ENABLED}] with value [true] in system.yaml" + fi +} + +# server.xml migration operations +xmlMigrateOperation () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local listenerCount="$4" + RT_DEFAULTPORT_STATUS=fail + AC_DEFAULTPORT_STATUS=fail + TLS_CONNECTOR_EXISTS=${FLAG_N} + + # Check for connector with TLS , if found ignore migrating it + checkForTlsConnector "${filePath}" "${fileName}" "${connectorCount}" + if [[ "${TLS_CONNECTOR_EXISTS}" == "${FLAG_Y}" ]]; then + return + fi + addTomcatServerListeners "${filePath}" "${fileName}" "${listenerCount}" + # Migrate RT default port from connectors + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_RT_PORT}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + # Migrate to extra if RT default ports are AJP + migrateDefaultPortIfAjp "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_RT_PORT}" + # Migrate AC default port from connectors + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_ACCESS_PORT}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${AC_SENDREASONPHRASE_YAMLPATH}" + # Migrate to extra if access default ports are AJP + migrateDefaultPortIfAjp "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_ACCESS_PORT}" + + if [[ "${AC_DEFAULTPORT_STATUS}" == "success" && "${RT_DEFAULTPORT_STATUS}" == "success" ]]; then + # RT and AC default port found + logger "Artifactory 8081 and Access 8040 default port are found" + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + elif [[ "${AC_DEFAULTPORT_STATUS}" == "success" && "${RT_DEFAULTPORT_STATUS}" == "fail" ]]; then + # Only AC default port found,find RT connector + logger "Found Access default 8040 port" + findRtConnector "${filePath}" "${fileName}" "${connectorCount}" + elif [[ "${AC_DEFAULTPORT_STATUS}" == "fail" && "${RT_DEFAULTPORT_STATUS}" == "success" ]]; then + # Only RT default port found,find AC connector + logger "Found Artifactory default 8081 port" + findAcConnector "${filePath}" "${fileName}" "${connectorCount}" + elif [[ "${AC_DEFAULTPORT_STATUS}" == "fail" && "${RT_DEFAULTPORT_STATUS}" == "fail" ]]; then + # RT and AC default port not found, find connector + logger "Artifactory 8081 and Access 8040 default port are not found" + findRtAndAcConnector "${filePath}" "${fileName}" "${connectorCount}" + fi +} + +# get count of connectors +getXmlConnectorCount () { + local filePath="$1" + local fileName="$2" + local count=$($LIBXML2_PATH --xpath 'count(/Server/Service/Connector)' ${filePath}/${fileName}) + echo -e "${count}" +} + +# get count of listener connectors +getTomcatServerListenersCount () { + local filePath="$1" + local fileName="$2" + local count=$($LIBXML2_PATH --xpath 'count(/Server/Listener)' ${filePath}/${fileName}) + echo -e "${count}" +} + +# Migrate server.xml configuration to system.yaml +migrateXmlFile () { + local xmlFiles= + local fileName= + local filePath= + local sourceFilePath= + DEFAULT_ACCESS_PORT="8040" + DEFAULT_RT_PORT="8081" + AC_PORT_YAMLPATH="migration.xmlFiles.serverXml.access.port" + AC_MAXTHREADS_YAMLPATH="migration.xmlFiles.serverXml.access.maxThreads" + AC_SENDREASONPHRASE_YAMLPATH="migration.xmlFiles.serverXml.access.sendReasonPhrase" + AC_EXTRACONFIG_YAMLPATH="migration.xmlFiles.serverXml.access.extraConfig" + RT_PORT_YAMLPATH="migration.xmlFiles.serverXml.artifactory.port" + RT_MAXTHREADS_YAMLPATH="migration.xmlFiles.serverXml.artifactory.maxThreads" + RT_SENDREASONPHRASE_YAMLPATH='migration.xmlFiles.serverXml.artifactory.sendReasonPhrase' + RT_RELAXEDPATHCHARS_YAMLPATH='migration.xmlFiles.serverXml.artifactory.relaxedPathChars' + RT_RELAXEDQUERYCHARS_YAMLPATH='migration.xmlFiles.serverXml.artifactory.relaxedQueryChars' + RT_EXTRACONFIG_YAMLPATH="migration.xmlFiles.serverXml.artifactory.extraConfig" + ROUTER_PORT_YAMLPATH="migration.xmlFiles.serverXml.router.port" + EXTRA_CONFIG_YAMLPATH="migration.xmlFiles.serverXml.extra.config" + EXTRA_LISTENER_CONFIG_YAMLPATH="migration.xmlFiles.serverXml.extra.listener" + RT_TOMCAT_HTTPSCONNECTOR_ENABLED="artifactory.tomcat.httpsConnector.enabled" + + retrieveYamlValue "migration.xmlFiles" "xmlFiles" "Skip" + xmlFiles="${YAML_VALUE}" + if [[ -z "${xmlFiles}" ]]; then + return + fi + bannerSection "PROCESSING MIGRATION OF XML FILES" + retrieveYamlValue "migration.xmlFiles.serverXml.fileName" "fileName" "Warning" + fileName="${YAML_VALUE}" + if [[ -z "${fileName}" ]]; then + return + fi + bannerSubSection "Processing Migration of $fileName" + retrieveYamlValue "migration.xmlFiles.serverXml.filePath" "filePath" "Warning" + filePath="${YAML_VALUE}" + if [[ -z "${filePath}" ]]; then + return + fi + # prepend NEW_DATA_DIR only if filePath is relative path + sourceFilePath=$(prependDir "${filePath}" "${NEW_DATA_DIR}/${filePath}") + if [[ "$(checkFileExists "${sourceFilePath}/${fileName}")" == "true" ]]; then + logger "File [${fileName}] is found in path [${sourceFilePath}]" + local connectorCount=$(getXmlConnectorCount "${sourceFilePath}" "${fileName}") + if [[ "${connectorCount}" == "0" ]]; then + logger "No connectors found in the [${filePath}/${fileName}],skipping migration of xml configuration" + return + fi + local listenerCount=$(getTomcatServerListenersCount "${sourceFilePath}" "${fileName}") + xmlMigrateOperation "${sourceFilePath}" "${fileName}" "${connectorCount}" "${listenerCount}" + else + logger "File [${fileName}] is not found in path [${sourceFilePath}] to migrate" + fi +} + +compareArtifactoryUser () { + local property="$1" + local oldPropertyValue="$2" + local newPropertyValue="$3" + local yamlPath="$4" + local sourceFile="$5" + + if [[ "${oldPropertyValue}" != "${newPropertyValue}" ]]; then + setSystemValue "${yamlPath}" "${oldPropertyValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value of the property [${property}] in system.yaml" + else + logger "No change in property [${property}] value in [${sourceFile}] to migrate" + fi +} + +migrateReplicator () { + local property="$1" + local oldPropertyValue="$2" + local yamlPath="$3" + + setSystemValue "${yamlPath}" "${oldPropertyValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value of the property [${property}] in system.yaml" +} + +compareJavaOptions () { + local property="$1" + local oldPropertyValue="$2" + local newPropertyValue="$3" + local yamlPath="$4" + local sourceFile="$5" + local oldJavaOption= + local newJavaOption= + local extraJavaOption= + local check=false + local success=true + local status=true + + oldJavaOption=$(echo "${oldPropertyValue}" | awk 'BEGIN{FS=OFS="\""}{for(i=2;i.+)\.{{ include "artifactory-ha.fullname" . }} {{ include "artifactory-ha.fullname" . }} +{{ tpl (include "artifactory.nginx.hosts" .) . }}; + +if ($http_x_forwarded_proto = '') { + set $http_x_forwarded_proto $scheme; +} +set $host_port {{ .Values.nginx.https.externalPort }}; +if ( $scheme = "http" ) { + set $host_port {{ .Values.nginx.http.externalPort }}; +} +## Application specific logs +## access_log /var/log/nginx/artifactory-access.log timing; +## error_log /var/log/nginx/artifactory-error.log; +rewrite ^/artifactory/?$ / redirect; +if ( $repo != "" ) { + rewrite ^/(v1|v2)/(.*) /artifactory/api/docker/$repo/$1/$2 break; +} +chunked_transfer_encoding on; +client_max_body_size 0; + +location / { + proxy_read_timeout 900; + proxy_pass_header Server; + proxy_cookie_path ~*^/.* /; + proxy_pass {{ include "artifactory-ha.scheme" . }}://{{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.externalPort }}/; + {{- if .Values.nginx.service.ssloffload}} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host; + {{- else }} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host:$host_port; + proxy_set_header X-Forwarded-Port $server_port; + {{- end }} + proxy_set_header X-Forwarded-Proto $http_x_forwarded_proto; + proxy_set_header Host $http_host; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + {{- if .Values.nginx.disableProxyBuffering}} + proxy_http_version 1.1; + proxy_request_buffering off; + proxy_buffering off; + {{- end }} + add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always; + location /artifactory/ { + if ( $request_uri ~ ^/artifactory/(.*)$ ) { + proxy_pass http://{{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/$1; + } + proxy_pass http://{{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/; + } + location /pipelines/ { + proxy_http_version 1.1; + proxy_set_header Upgrade $http_upgrade; + proxy_set_header Connection "upgrade"; + proxy_set_header Host $http_host; + {{- if .Values.router.tlsEnabled }} + proxy_pass https://{{ include "artifactory-ha.fullname" . }}:{{ .Values.router.internalPort }}; + {{- else }} + proxy_pass http://{{ include "artifactory-ha.fullname" . }}:{{ .Values.router.internalPort }}; + {{- end }} + } +} +} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/files/nginx-main-conf.yaml b/charts/jfrog/artifactory-ha/107.90.14/files/nginx-main-conf.yaml new file mode 100644 index 0000000000..78cecea6a1 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/files/nginx-main-conf.yaml @@ -0,0 +1,83 @@ +# Main Nginx configuration file +worker_processes 4; + +{{- if .Values.nginx.logs.stderr }} +error_log stderr {{ .Values.nginx.logs.level }}; +{{- else -}} +error_log {{ .Values.nginx.persistence.mountPath }}/logs/error.log {{ .Values.nginx.logs.level }}; +{{- end }} +pid /var/run/nginx.pid; + +{{- if .Values.artifactory.ssh.enabled }} +## SSH Server Configuration +stream { + server { + {{- if .Values.nginx.singleStackIPv6Cluster }} + listen [::]:{{ .Values.nginx.ssh.internalPort }}; + {{- else -}} + listen {{ .Values.nginx.ssh.internalPort }}; + {{- end }} + proxy_pass {{ include "artifactory-ha.fullname" . }}:{{ .Values.artifactory.ssh.externalPort }}; + } +} +{{- end }} + +events { + worker_connections 1024; +} + +http { + include /etc/nginx/mime.types; + default_type application/octet-stream; + + variables_hash_max_size 1024; + variables_hash_bucket_size 64; + server_names_hash_max_size 4096; + server_names_hash_bucket_size 128; + types_hash_max_size 2048; + types_hash_bucket_size 64; + proxy_read_timeout 2400s; + client_header_timeout 2400s; + client_body_timeout 2400s; + proxy_connect_timeout 75s; + proxy_send_timeout 2400s; + proxy_buffer_size 128k; + proxy_buffers 40 128k; + proxy_busy_buffers_size 128k; + proxy_temp_file_write_size 250m; + proxy_http_version 1.1; + client_body_buffer_size 128k; + + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + + log_format timing 'ip = $remote_addr ' + 'user = \"$remote_user\" ' + 'local_time = \"$time_local\" ' + 'host = $host ' + 'request = \"$request\" ' + 'status = $status ' + 'bytes = $body_bytes_sent ' + 'upstream = \"$upstream_addr\" ' + 'upstream_time = $upstream_response_time ' + 'request_time = $request_time ' + 'referer = \"$http_referer\" ' + 'UA = \"$http_user_agent\"'; + + {{- if .Values.nginx.logs.stdout }} + access_log /dev/stdout timing; + {{- else -}} + access_log {{ .Values.nginx.persistence.mountPath }}/logs/access.log timing; + {{- end }} + + sendfile on; + #tcp_nopush on; + + keepalive_timeout 65; + + #gzip on; + + include /etc/nginx/conf.d/*.conf; + +} diff --git a/charts/jfrog/artifactory-ha/107.90.14/files/system.yaml b/charts/jfrog/artifactory-ha/107.90.14/files/system.yaml new file mode 100644 index 0000000000..3a1d93269d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/files/system.yaml @@ -0,0 +1,163 @@ +router: + serviceRegistry: + insecure: {{ .Values.router.serviceRegistry.insecure }} +shared: +{{- if .Values.artifactory.coldStorage.enabled }} + jfrogColdStorage: + coldInstanceEnabled: true +{{- end }} +{{ tpl (include "artifactory.metrics" .) . }} + logging: + consoleLog: + enabled: {{ .Values.artifactory.consoleLog }} + extraJavaOpts: > + -Dartifactory.graceful.shutdown.max.request.duration.millis={{ mul .Values.artifactory.terminationGracePeriodSeconds 1000 }} + -Dartifactory.access.client.max.connections={{ .Values.access.tomcat.connector.maxThreads }} + {{- with .Values.artifactory.primary.javaOpts }} + {{- if .corePoolSize }} + -Dartifactory.async.corePoolSize={{ .corePoolSize }} + {{- end }} + {{- if .xms }} + -Xms{{ .xms }} + {{- end }} + {{- if .xmx }} + -Xmx{{ .xmx }} + {{- end }} + {{- if .jmx.enabled }} + -Dcom.sun.management.jmxremote + -Dcom.sun.management.jmxremote.port={{ .jmx.port }} + -Dcom.sun.management.jmxremote.rmi.port={{ .jmx.port }} + -Dcom.sun.management.jmxremote.ssl={{ .jmx.ssl }} + {{- if .jmx.host }} + -Djava.rmi.server.hostname={{ tpl .jmx.host $ }} + {{- else }} + -Djava.rmi.server.hostname={{ template "artifactory-ha.fullname" $ }} + {{- end }} + {{- if .jmx.authenticate }} + -Dcom.sun.management.jmxremote.authenticate=true + -Dcom.sun.management.jmxremote.access.file={{ .jmx.accessFile }} + -Dcom.sun.management.jmxremote.password.file={{ .jmx.passwordFile }} + {{- else }} + -Dcom.sun.management.jmxremote.authenticate=false + {{- end }} + {{- end }} + {{- if .other }} + {{ .other }} + {{- end }} + {{- end }} + database: + allowNonPostgresql: {{ .Values.database.allowNonPostgresql }} + {{- if .Values.postgresql.enabled }} + type: postgresql + url: "jdbc:postgresql://{{ .Release.Name }}-postgresql:{{ .Values.postgresql.service.port }}/{{ .Values.postgresql.postgresqlDatabase }}" + host: "" + driver: org.postgresql.Driver + username: "{{ .Values.postgresql.postgresqlUsername }}" + {{ else }} + type: "{{ .Values.database.type }}" + driver: "{{ .Values.database.driver }}" + {{- end }} +artifactory: +{{- if or .Values.artifactory.haDataDir.enabled .Values.artifactory.haBackupDir.enabled }} + node: + {{- if .Values.artifactory.haDataDir.path }} + haDataDir: {{ .Values.artifactory.haDataDir.path }} + {{- end }} + {{- if .Values.artifactory.haBackupDir.path }} + haBackupDir: {{ .Values.artifactory.haBackupDir.path }} + {{- end }} +{{- end }} + database: + maxOpenConnections: {{ .Values.artifactory.database.maxOpenConnections }} + tomcat: + maintenanceConnector: + port: {{ .Values.artifactory.tomcat.maintenanceConnector.port }} + connector: + maxThreads: {{ .Values.artifactory.tomcat.connector.maxThreads }} + sendReasonPhrase: {{ .Values.artifactory.tomcat.connector.sendReasonPhrase }} + extraConfig: {{ .Values.artifactory.tomcat.connector.extraConfig }} +frontend: + session: + timeMinutes: {{ .Values.frontend.session.timeoutMinutes | quote }} +access: + runOnArtifactoryTomcat: {{ .Values.access.runOnArtifactoryTomcat | default false }} + database: + maxOpenConnections: {{ .Values.access.database.maxOpenConnections }} + {{- if not (.Values.access.runOnArtifactoryTomcat | default false) }} + extraJavaOpts: > + {{- if .Values.splitServicesToContainers }} + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=70 + {{- end }} + {{- with .Values.access.javaOpts }} + {{- if .other }} + {{ .other }} + {{- end }} + {{- end }} + {{- end }} + tomcat: + connector: + maxThreads: {{ .Values.access.tomcat.connector.maxThreads }} + sendReasonPhrase: {{ .Values.access.tomcat.connector.sendReasonPhrase }} + extraConfig: {{ .Values.access.tomcat.connector.extraConfig }} + {{- if .Values.access.database.enabled }} + type: "{{ .Values.access.database.type }}" + url: "{{ .Values.access.database.url }}" + driver: "{{ .Values.access.database.driver }}" + username: "{{ .Values.access.database.user }}" + password: "{{ .Values.access.database.password }}" + {{- end }} +{{- if .Values.mc.enabled }} +mc: + enabled: true + database: + maxOpenConnections: {{ .Values.mc.database.maxOpenConnections }} + idgenerator: + maxOpenConnections: {{ .Values.mc.idgenerator.maxOpenConnections }} + tomcat: + connector: + maxThreads: {{ .Values.mc.tomcat.connector.maxThreads }} + sendReasonPhrase: {{ .Values.mc.tomcat.connector.sendReasonPhrase }} + extraConfig: {{ .Values.mc.tomcat.connector.extraConfig }} +{{- end }} +metadata: + database: + maxOpenConnections: {{ .Values.metadata.database.maxOpenConnections }} +{{- if and .Values.jfconnect.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" .Values.artifactory.image.repository)) }} +jfconnect: + enabled: true +{{- else }} +jfconnect: + enabled: false +jfconnect_service: + enabled: false +{{- end }} + +{{- if and .Values.federation.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" .Values.artifactory.image.repository)) }} +federation: + enabled: true + embedded: {{ .Values.federation.embedded }} + extraJavaOpts: {{ .Values.federation.extraJavaOpts }} + port: {{ .Values.federation.internalPort }} +rtfs: + database: + driver: org.postgresql.Driver + type: postgresql + username: {{ .Values.federation.database.username }} + password: {{ .Values.federation.database.password }} + url: "jdbc:postgresql://{{ .Values.federation.database.host }}:{{ .Values.federation.database.port }}/{{ .Values.federation.database.name }}" +{{- else }} +federation: + enabled: false +{{- end }} +{{- if .Values.event.webhooks }} +event: + webhooks: {{ toYaml .Values.event.webhooks | nindent 6 }} +{{- end }} +{{- if .Values.evidence.enabled }} +evidence: + enabled: true +{{- else }} +evidence: + enabled: false +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/logo/artifactory-logo.png b/charts/jfrog/artifactory-ha/107.90.14/logo/artifactory-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..fe6c23c5a7f87edaf49f883ffa99d875073e2285 GIT binary patch literal 82419 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 z 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 GZv9`RA2Y%L literal 0 HcmV?d00001 diff --git a/charts/jfrog/artifactory-ha/107.90.14/questions.yml b/charts/jfrog/artifactory-ha/107.90.14/questions.yml new file mode 100644 index 0000000000..14e9024e6d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/questions.yml @@ -0,0 +1,424 @@ +questions: +# Advance Settings +- variable: artifactory.masterKey + default: "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + description: "Artifactory master key. For security reasons, we strongly recommend you generate your own master key using this command: 'openssl rand -hex 32'" + type: string + label: Artifactory master key + group: "Security Settings" + +# Container Images +- variable: defaultImage + default: true + description: "Use default Docker image" + label: Use Default Image + type: boolean + show_subquestion_if: false + group: "Container Images" + subquestions: + - variable: initContainerImage + default: "docker.bintray.io/alpine:3.12" + description: "Init image name" + type: string + label: Init image name + - variable: artifactory.image.repository + default: "docker.bintray.io/jfrog/artifactory-pro" + description: "Artifactory image name" + type: string + label: Artifactory Image Name + - variable: artifactory.image.version + default: "7.6.3" + description: "Artifactory image tag" + type: string + label: Artifactory Image Tag + - variable: nginx.image.repository + default: "docker.bintray.io/jfrog/nginx-artifactory-pro" + description: "Nginx image name" + type: string + label: Nginx Image Name + - variable: nginx.image.version + default: "7.6.3" + description: "Nginx image tag" + type: string + label: Nginx Image Tag + - variable: imagePullSecrets + description: "Image Pull Secret" + type: string + label: Image Pull Secret + +# Services and LoadBalancing Settings +- variable: artifactory.node.replicaCount + default: "2" + description: "Number of Secondary Nodes" + type: string + label: Number of Secondary Nodes + show_subquestion_if: true + group: "Services and Load Balancing" +- variable: ingress.enabled + default: false + description: "Expose app using Layer 7 Load Balancer - ingress" + type: boolean + label: Expose app using Layer 7 Load Balancer + show_subquestion_if: true + group: "Services and Load Balancing" + required: true + subquestions: + - variable: ingress.hosts[0] + default: "xip.io" + description: "Hostname to your artifactory installation" + type: hostname + required: true + label: Hostname + +# Nginx Settings +- variable: nginx.enabled + default: true + description: "Enable nginx server" + type: boolean + label: Enable Nginx Server + group: "Services and Load Balancing" + required: true + show_if: "ingress.enabled=false" +- variable: nginx.service.type + default: "LoadBalancer" + description: "Nginx service type" + type: enum + required: true + label: Nginx Service Type + show_if: "nginx.enabled=true&&ingress.enabled=false" + group: "Services and Load Balancing" + options: + - "ClusterIP" + - "NodePort" + - "LoadBalancer" +- variable: nginx.service.loadBalancerIP + default: "" + description: "Provide Static IP to configure with Nginx" + type: string + label: Config Nginx LoadBalancer IP + show_if: "nginx.enabled=true&&nginx.service.type=LoadBalancer&&ingress.enabled=false" + group: "Services and Load Balancing" +- variable: nginx.tlsSecretName + default: "" + description: "Provide SSL Secret name to configure with Nginx" + type: string + label: Config Nginx SSL Secret + show_if: "nginx.enabled=true&&ingress.enabled=false" + group: "Services and Load Balancing" +- variable: nginx.customArtifactoryConfigMap + default: "" + description: "Provide configMap name to configure Nginx with custom `artifactory.conf`" + type: string + label: ConfigMap for Nginx Artifactory Config + show_if: "nginx.enabled=true&&ingress.enabled=false" + group: "Services and Load Balancing" + +# Artifactory Storage Settings +- variable: artifactory.persistence.size + default: "50Gi" + description: "Artifactory persistent volume size" + type: string + label: Artifactory Persistent Volume Size + required: true + group: "Artifactory Storage" +- variable: artifactory.persistence.type + default: "file-system" + description: "Artifactory persistent volume size" + type: enum + label: Artifactory Persistent Storage Type + required: true + options: + - "file-system" + - "nfs" + - "google-storage" + - "aws-s3" + group: "Artifactory Storage" + +#Storage Type Settings +- variable: artifactory.persistence.nfs.ip + default: "" + type: string + group: "Artifactory Storage" + label: NFS Server IP + description: "NFS server IP" + show_if: "artifactory.persistence.type=nfs" +- variable: artifactory.persistence.nfs.haDataMount + default: "/data" + type: string + label: NFS Data Directory + description: "NFS data directory" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=nfs" +- variable: artifactory.persistence.nfs.haBackupMount + default: "/backup" + type: string + label: NFS Backup Directory + description: "NFS backup directory" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=nfs" +- variable: artifactory.persistence.nfs.dataDir + default: "/var/opt/jfrog/artifactory-ha" + type: string + label: HA Data Directory + description: "HA data directory" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=nfs" +- variable: artifactory.persistence.nfs.backupDir + default: "/var/opt/jfrog/artifactory-backup" + type: string + label: HA Backup Directory + description: "HA backup directory " + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=nfs" +- variable: artifactory.persistence.nfs.capacity + default: "200Gi" + type: string + label: NFS PVC Size + description: "NFS PVC size " + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=nfs" + +#Google storage settings +- variable: artifactory.persistence.googleStorage.bucketName + default: "artifactory-ha-gcp" + type: string + label: Google Storage Bucket Name + description: "Google storage bucket name" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=google-storage" +- variable: artifactory.persistence.googleStorage.identity + default: "" + type: string + label: Google Storage Service Account ID + description: "Google Storage service account id" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=google-storage" +- variable: artifactory.persistence.googleStorage.credential + default: "" + type: string + label: Google Storage Service Account Key + description: "Google Storage service account key" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=google-storage" +- variable: artifactory.persistence.googleStorage.path + default: "artifactory-ha/filestore" + type: string + label: Google Storage Path In Bucket + description: "Google Storage path in bucket" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=google-storage" +# awsS3 storage settings +- variable: artifactory.persistence.awsS3.bucketName + default: "artifactory-ha-aws" + type: string + label: AWS S3 Bucket Name + description: "AWS S3 bucket name" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=aws-s3" +- variable: artifactory.persistence.awsS3.region + default: "" + type: string + label: AWS S3 Bucket Region + description: "AWS S3 bucket region" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=aws-s3" +- variable: artifactory.persistence.awsS3.identity + default: "" + type: string + label: AWS S3 AWS_ACCESS_KEY_ID + description: "AWS S3 AWS_ACCESS_KEY_ID" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=aws-s3" +- variable: artifactory.persistence.awsS3.credential + default: "" + type: string + label: AWS S3 AWS_SECRET_ACCESS_KEY + description: "AWS S3 AWS_SECRET_ACCESS_KEY" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=aws-s3" +- variable: artifactory.persistence.awsS3.path + default: "artifactory-ha/filestore" + type: string + label: AWS S3 Path In Bucket + description: "AWS S3 path in bucket" + group: "Artifactory Storage" + show_if: "artifactory.persistence.type=aws-s3" + +# Database Settings +- variable: postgresql.enabled + default: true + description: "Enable PostgreSQL" + type: boolean + required: true + label: Enable PostgreSQL + group: "Database Settings" + show_subquestion_if: true + subquestions: + - variable: postgresql.postgresqlPassword + default: "" + description: "PostgreSQL password" + type: password + required: true + label: PostgreSQL Password + group: "Database Settings" + show_if: "postgresql.enabled=true" + - variable: postgresql.persistence.size + default: 20Gi + description: "PostgreSQL persistent volume size" + type: string + label: PostgreSQL Persistent Volume Size + show_if: "postgresql.enabled=true" + - variable: postgresql.persistence.storageClass + default: "" + description: "If undefined or null, uses the default StorageClass. Default to null" + type: storageclass + label: Default StorageClass for PostgreSQL + show_if: "postgresql.enabled=true" + - variable: postgresql.resources.requests.cpu + default: "200m" + description: "PostgreSQL initial cpu request" + type: string + label: PostgreSQL Initial CPU Request + show_if: "postgresql.enabled=true" + - variable: postgresql.resources.requests.memory + default: "500Mi" + description: "PostgreSQL initial memory request" + type: string + label: PostgreSQL Initial Memory Request + show_if: "postgresql.enabled=true" + - variable: postgresql.resources.limits.cpu + default: "1" + description: "PostgreSQL cpu limit" + type: string + label: PostgreSQL CPU Limit + show_if: "postgresql.enabled=true" + - variable: postgresql.resources.limits.memory + default: "1Gi" + description: "PostgreSQL memory limit" + type: string + label: PostgreSQL Memory Limit + show_if: "postgresql.enabled=true" +- variable: database.type + default: "postgresql" + description: "xternal database type (postgresql, mysql, oracle or mssql)" + type: enum + required: true + label: External Database Type + group: "Database Settings" + show_if: "postgresql.enabled=false" + options: + - "postgresql" + - "mysql" + - "oracle" + - "mssql" +- variable: database.url + default: "" + description: "External database URL. If you set the url, leave host and port empty" + type: string + label: External Database URL + group: "Database Settings" + show_if: "postgresql.enabled=false" +- variable: database.host + default: "" + description: "External database hostname" + type: string + label: External Database Hostname + group: "Database Settings" + show_if: "postgresql.enabled=false" +- variable: database.port + default: "" + description: "External database port" + type: string + label: External Database Port + group: "Database Settings" + show_if: "postgresql.enabled=false" +- variable: database.user + default: "" + description: "External database username" + type: string + label: External Database Username + group: "Database Settings" + show_if: "postgresql.enabled=false" +- variable: database.password + default: "" + description: "External database password" + type: password + label: External Database Password + group: "Database Settings" + show_if: "postgresql.enabled=false" + +# Advance Settings +- variable: advancedOptions + default: false + description: "Show advanced configurations" + label: Show Advanced Configurations + type: boolean + show_subquestion_if: true + group: "Advanced Options" + subquestions: + - variable: artifactory.primary.resources.requests.cpu + default: "500m" + description: "Artifactory primary node initial cpu request" + type: string + label: Artifactory Primary Node Initial CPU Request + - variable: artifactory.primary.resources.requests.memory + default: "1Gi" + description: "Artifactory primary node initial memory request" + type: string + label: Artifactory Primary Node Initial Memory Request + - variable: artifactory.primary.javaOpts.xms + default: "1g" + description: "Artifactory primary node java Xms size" + type: string + label: Artifactory Primary Node Java Xms Size + - variable: artifactory.primary.resources.limits.cpu + default: "2" + description: "Artifactory primary node cpu limit" + type: string + label: Artifactory Primary Node CPU Limit + - variable: artifactory.primary.resources.limits.memory + default: "4Gi" + description: "Artifactory primary node memory limit" + type: string + label: Artifactory Primary Node Memory Limit + - variable: artifactory.primary.javaOpts.xmx + default: "4g" + description: "Artifactory primary node java Xmx size" + type: string + label: Artifactory Primary Node Java Xmx Size + - variable: artifactory.node.resources.requests.cpu + default: "500m" + description: "Artifactory member node initial cpu request" + type: string + label: Artifactory Member Node Initial CPU Request + - variable: artifactory.node.resources.requests.memory + default: "2Gi" + description: "Artifactory member node initial memory request" + type: string + label: Artifactory Member Node Initial Memory Request + - variable: artifactory.node.javaOpts.xms + default: "1g" + description: "Artifactory member node java Xms size" + type: string + label: Artifactory Member Node Java Xms Size + - variable: artifactory.node.resources.limits.cpu + default: "2" + description: "Artifactory member node cpu limit" + type: string + label: Artifactory Member Node CPU Limit + - variable: artifactory.node.resources.limits.memory + default: "4Gi" + description: "Artifactory member node memory limit" + type: string + label: Artifactory Member Node Memory Limit + - variable: artifactory.node.javaOpts.xmx + default: "4g" + description: "Artifactory member node java Xmx size" + type: string + label: Artifactory Member Node Java Xmx Size + +# Internal Settings +- variable: installerInfo + default: '\{\"productId\": \"RancherHelm_artifactory-ha/7.17.5\", \"features\": \[\{\"featureId\": \"Partner/ACC-007246\"\}\]\}' + type: string + group: "Internal Settings (Do not modify)" diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge-extra-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge-extra-config.yaml new file mode 100644 index 0000000000..6afc491dc8 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge-extra-config.yaml @@ -0,0 +1,44 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + primary: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=200 + -Dartifactory.async.poolMaxQueueSize=100000 + -Dartifactory.http.client.max.total.connections=150 + -Dartifactory.http.client.max.connections.per.route=150 + -Dartifactory.access.client.max.connections=200 + -Dartifactory.metadata.event.operator.threads=5 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=1048576 + -XX:MaxDirectMemorySize=1024m + + tomcat: + connector: + maxThreads: 800 + extraConfig: 'acceptCount="1200" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 200 + +access: + tomcat: + connector: + maxThreads: 200 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + + database: + maxOpenConnections: 200 + +metadata: + database: + maxOpenConnections: 200 + diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge.yaml new file mode 100644 index 0000000000..02cf7f94e4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-2xlarge.yaml @@ -0,0 +1,127 @@ +############################################################## +# The 2xlarge sizing +# This size is intended for very large organizations. It can be increased with adding replicas +############################################################## +splitServicesToContainers: true +artifactory: + primary: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 6 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "4" + memory: 20Gi + limits: + # cpu: "20" + memory: 24Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "16" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +router: + resources: + requests: + cpu: "1" + memory: 1Gi + limits: + # cpu: "6" + memory: 2Gi + +frontend: + resources: + requests: + cpu: "1" + memory: 500Mi + limits: + # cpu: "5" + memory: 1Gi + +metadata: + resources: + requests: + cpu: "1" + memory: 500Mi + limits: + # cpu: "5" + memory: 2Gi + +event: + resources: + requests: + cpu: 200m + memory: 100Mi + limits: + # cpu: "1" + memory: 500Mi + +access: + resources: + requests: + cpu: 1 + memory: 2Gi + limits: + # cpu: 2 + memory: 4Gi + +observability: + resources: + requests: + cpu: 200m + memory: 100Mi + limits: + # cpu: "1" + memory: 500Mi + +jfconnect: + resources: + requests: + cpu: 100m + memory: 100Mi + limits: + # cpu: "1" + memory: 250Mi + +nginx: + replicaCount: 3 + disableProxyBuffering: true + resources: + requests: + cpu: "4" + memory: "6Gi" + limits: + # cpu: "14" + memory: "8Gi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "5000" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 256Gi + cpu: "64" + limits: + memory: 256Gi + # cpu: "128" diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large-extra-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large-extra-config.yaml new file mode 100644 index 0000000000..fac24ad687 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large-extra-config.yaml @@ -0,0 +1,44 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + primary: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=65 + -Dartifactory.async.corePoolSize=80 + -Dartifactory.async.poolMaxQueueSize=20000 + -Dartifactory.http.client.max.total.connections=100 + -Dartifactory.http.client.max.connections.per.route=100 + -Dartifactory.access.client.max.connections=125 + -Dartifactory.metadata.event.operator.threads=4 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=524288 + -XX:MaxDirectMemorySize=512m + + tomcat: + connector: + maxThreads: 500 + extraConfig: 'acceptCount="800" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 100 + +access: + tomcat: + connector: + maxThreads: 125 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + + database: + maxOpenConnections: 100 + +metadata: + database: + maxOpenConnections: 100 + diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large.yaml new file mode 100644 index 0000000000..504edf1ed8 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-large.yaml @@ -0,0 +1,127 @@ +############################################################## +# The large sizing +# This size is intended for large organizations. It can be increased with adding replicas or moving to the xlarge sizing +############################################################## +splitServicesToContainers: true +artifactory: + primary: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 3 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "2" + memory: 10Gi + limits: + # cpu: "14" + memory: 12Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "8" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 1 + memory: 2Gi + limits: + # cpu: 2 + memory: 3Gi + +router: + resources: + requests: + cpu: 200m + memory: 400Mi + limits: + # cpu: "4" + memory: 1Gi + +frontend: + resources: + requests: + cpu: 200m + memory: 300Mi + limits: + # cpu: "3" + memory: 1Gi + +metadata: + resources: + requests: + cpu: 200m + memory: 200Mi + limits: + # cpu: "4" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 2 + disableProxyBuffering: true + resources: + requests: + cpu: "1" + memory: "500Mi" + limits: + # cpu: "4" + memory: "1Gi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "600" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 64Gi + cpu: "16" + limits: + memory: 64Gi + # cpu: "32" diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium-extra-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium-extra-config.yaml new file mode 100644 index 0000000000..b2b20b198b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium-extra-config.yaml @@ -0,0 +1,45 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + primary: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=40 + -Dartifactory.async.poolMaxQueueSize=10000 + -Dartifactory.http.client.max.total.connections=50 + -Dartifactory.http.client.max.connections.per.route=50 + -Dartifactory.access.client.max.connections=75 + -Dartifactory.metadata.event.operator.threads=3 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=262144 + -XX:MaxDirectMemorySize=256m + + tomcat: + connector: + maxThreads: 300 + extraConfig: 'acceptCount="600" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 50 + +access: + tomcat: + connector: + maxThreads: 75 + + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + + database: + maxOpenConnections: 50 + +metadata: + database: + maxOpenConnections: 50 + diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium.yaml new file mode 100644 index 0000000000..93b79788df --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-medium.yaml @@ -0,0 +1,127 @@ +############################################################## +# The medium sizing +# This size is just 2 replicas of the small size. Vertical sizing of all services is not changed +############################################################## +splitServicesToContainers: true +artifactory: + primary: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 2 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "1" + memory: 4Gi + limits: + # cpu: "10" + memory: 5Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "2" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +router: + resources: + requests: + cpu: 100m + memory: 250Mi + limits: + # cpu: "1" + memory: 500Mi + +frontend: + resources: + requests: + cpu: 100m + memory: 150Mi + limits: + # cpu: "2" + memory: 250Mi + +metadata: + resources: + requests: + cpu: 100m + memory: 200Mi + limits: + # cpu: "2" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +access: + resources: + requests: + cpu: 1 + memory: 1.5Gi + limits: + # cpu: 1.5 + memory: 2Gi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 2 + disableProxyBuffering: true + resources: + requests: + cpu: "100m" + memory: "100Mi" + limits: + # cpu: "2" + memory: "500Mi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "200" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 32Gi + cpu: "8" + limits: + memory: 32Gi + # cpu: "16" \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small-extra-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small-extra-config.yaml new file mode 100644 index 0000000000..e8329f1a3e --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small-extra-config.yaml @@ -0,0 +1,43 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + primary: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=40 + -Dartifactory.async.poolMaxQueueSize=10000 + -Dartifactory.http.client.max.total.connections=50 + -Dartifactory.http.client.max.connections.per.route=50 + -Dartifactory.access.client.max.connections=75 + -Dartifactory.metadata.event.operator.threads=3 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=262144 + -XX:MaxDirectMemorySize=256m + + tomcat: + connector: + maxThreads: 300 + extraConfig: 'acceptCount="600" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 50 + +access: + tomcat: + connector: + maxThreads: 75 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 50 + +metadata: + database: + maxOpenConnections: 50 + diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small.yaml new file mode 100644 index 0000000000..b75a22323f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-small.yaml @@ -0,0 +1,127 @@ +############################################################## +# The small sizing +# This is the size recommended for running Artifactory for small teams +############################################################## +splitServicesToContainers: true +artifactory: + primary: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 1 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "1" + memory: 4Gi + limits: + # cpu: "10" + memory: 5Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "2" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +router: + resources: + requests: + cpu: 100m + memory: 250Mi + limits: + # cpu: "1" + memory: 500Mi + +access: + resources: + requests: + cpu: 500m + memory: 1.5Gi + limits: + # cpu: 1 + memory: 2Gi + +frontend: + resources: + requests: + cpu: 100m + memory: 150Mi + limits: + # cpu: "2" + memory: 250Mi + +metadata: + resources: + requests: + cpu: 100m + memory: 200Mi + limits: + # cpu: "2" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 1 + disableProxyBuffering: true + resources: + requests: + cpu: "100m" + memory: "100Mi" + limits: + # cpu: "2" + memory: "500Mi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "100" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 16Gi + cpu: "4" + limits: + memory: 16Gi + # cpu: "10" diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge-extra-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge-extra-config.yaml new file mode 100644 index 0000000000..8d04850ad8 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge-extra-config.yaml @@ -0,0 +1,42 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + primary: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=65 + -Dartifactory.async.corePoolSize=160 + -Dartifactory.async.poolMaxQueueSize=50000 + -Dartifactory.http.client.max.total.connections=150 + -Dartifactory.http.client.max.connections.per.route=150 + -Dartifactory.access.client.max.connections=150 + -Dartifactory.metadata.event.operator.threads=5 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=1048576 + -XX:MaxDirectMemorySize=1024m + tomcat: + connector: + maxThreads: 600 + extraConfig: 'acceptCount="1200" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 150 + +access: + tomcat: + connector: + maxThreads: 150 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 150 + +metadata: + database: + maxOpenConnections: 150 + diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge.yaml new file mode 100644 index 0000000000..550bd051d6 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xlarge.yaml @@ -0,0 +1,127 @@ +############################################################## +# The xlarge sizing +# This size is intended for very large organizations. It can be increased with adding replicas +############################################################## +splitServicesToContainers: true +artifactory: + primary: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 4 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "2" + memory: 14Gi + limits: + # cpu: "14" + memory: 16Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "16" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 1 + memory: 2Gi + limits: + # cpu: 2 + memory: 4Gi + +router: + resources: + requests: + cpu: 200m + memory: 500Mi + limits: + # cpu: "4" + memory: 1Gi + +frontend: + resources: + requests: + cpu: 200m + memory: 300Mi + limits: + # cpu: "3" + memory: 1Gi + +metadata: + resources: + requests: + cpu: 200m + memory: 200Mi + limits: + # cpu: "4" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 2 + disableProxyBuffering: true + resources: + requests: + cpu: "4" + memory: "4Gi" + limits: + # cpu: "12" + memory: "8Gi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "2000" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 128Gi + cpu: "32" + limits: + memory: 128Gi + # cpu: "64" diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall-extra-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall-extra-config.yaml new file mode 100644 index 0000000000..1371e87b8d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall-extra-config.yaml @@ -0,0 +1,43 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + primary: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=10 + -Dartifactory.async.poolMaxQueueSize=2000 + -Dartifactory.http.client.max.total.connections=20 + -Dartifactory.http.client.max.connections.per.route=20 + -Dartifactory.access.client.max.connections=15 + -Dartifactory.metadata.event.operator.threads=2 + -XX:MaxMetaspaceSize=400m + -XX:CompressedClassSpaceSize=96m + -Djdk.nio.maxCachedBufferSize=131072 + -XX:MaxDirectMemorySize=128m + tomcat: + connector: + maxThreads: 50 + extraConfig: 'acceptCount="200" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 15 + +access: + tomcat: + connector: + maxThreads: 15 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 15 + +metadata: + database: + maxOpenConnections: 15 + diff --git a/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall.yaml b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall.yaml new file mode 100644 index 0000000000..3f7b07138b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/sizing/artifactory-xsmall.yaml @@ -0,0 +1,127 @@ +############################################################## +# The xsmall sizing +# This is the minimum size recommended for running Artifactory +############################################################## +splitServicesToContainers: true +artifactory: + primary: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 1 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "1" + memory: 3Gi + limits: + # cpu: "10" + memory: 4Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "2" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 500m + memory: 1.5Gi + limits: + # cpu: 1 + memory: 2Gi + +router: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: "1" + memory: 500Mi + +frontend: + resources: + requests: + cpu: 50m + memory: 150Mi + limits: + # cpu: "2" + memory: 250Mi + +metadata: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: "2" + memory: 1Gi + +event: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 1 + disableProxyBuffering: true + resources: + requests: + cpu: "50m" + memory: "50Mi" + limits: + # cpu: "1" + memory: "250Mi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "50" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 8Gi + cpu: "2" + limits: + memory: 8Gi + # cpu: "8" \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/NOTES.txt b/charts/jfrog/artifactory-ha/107.90.14/templates/NOTES.txt new file mode 100644 index 0000000000..30dfab8b8a --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/NOTES.txt @@ -0,0 +1,149 @@ +Congratulations. You have just deployed JFrog Artifactory HA! + +{{- if .Values.artifactory.masterKey }} +{{- if and (not .Values.artifactory.masterKeySecretName) (eq .Values.artifactory.masterKey "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF") }} + + +***************************************** WARNING ****************************************** +* Your Artifactory master key is still set to the provided example: * +* artifactory.masterKey=FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF * +* * +* You should change this to your own generated key: * +* $ export MASTER_KEY=$(openssl rand -hex 32) * +* $ echo ${MASTER_KEY} * +* * +* Pass the created master key to helm with '--set artifactory.masterKey=${MASTER_KEY}' * +* * +* Alternatively, you can use a pre-existing secret with a key called master-key with * +* '--set artifactory.masterKeySecretName=${SECRET_NAME}' * +******************************************************************************************** +{{- end }} +{{- end }} + +{{- if .Values.artifactory.joinKey }} +{{- if eq .Values.artifactory.joinKey "EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE" }} + + +***************************************** WARNING ****************************************** +* Your Artifactory join key is still set to the provided example: * +* artifactory.joinKey=EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE * +* * +* You should change this to your own generated key: * +* $ export JOIN_KEY=$(openssl rand -hex 32) * +* $ echo ${JOIN_KEY} * +* * +* Pass the created master key to helm with '--set artifactory.joinKey=${JOIN_KEY}' * +* * +******************************************************************************************** +{{- end }} +{{- end }} + + +{{- if .Values.artifactory.setSecurityContext }} +****************************************** WARNING ********************************************** +* From chart version 107.84.x, `setSecurityContext` has been renamed to `podSecurityContext`, * + please change your values.yaml before upgrade , For more Info , refer to 107.84.x changelog * +************************************************************************************************* +{{- end }} + +{{- if and (or (or (or (or (or ( or ( or ( or (or (or ( or (or .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName) .Values.systemYamlOverride.existingSecret) (or .Values.artifactory.customCertificates.enabled .Values.global.customCertificates.enabled)) .Values.aws.licenseConfigSecretName) .Values.artifactory.persistence.customBinarystoreXmlSecret) .Values.access.customCertificatesSecretName) .Values.systemYamlOverride.existingSecret) .Values.artifactory.license.secret) .Values.artifactory.userPluginSecrets) (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey)) (and .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName)) (or .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName)) .Values.artifactory.unifiedSecretInstallation }} +****************************************** WARNING ************************************************************************************************** +* The unifiedSecretInstallation flag is currently enabled, which creates the unified secret. The existing secrets will continue as separate secrets.* +* Update the values.yaml with the existing secrets to add them to the unified secret. * +***************************************************************************************************************************************************** +{{- end }} + +{{- if .Values.postgresql.enabled }} + +DATABASE: +To extract the database password, run the following +export DB_PASSWORD=$(kubectl get --namespace {{ .Release.Namespace }} $(kubectl get secret --namespace {{ .Release.Namespace }} -o name | grep postgresql) -o jsonpath="{.data.postgresql-password}" | base64 --decode) +echo ${DB_PASSWORD} +{{- end }} + +SETUP: +1. Get the Artifactory IP and URL + + {{- if contains "NodePort" .Values.nginx.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "artifactory-ha.nginx.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.nginx.service.type }} + NOTE: It may take a few minutes for the LoadBalancer public IP to be available! + + You can watch the status of the service by running 'kubectl get svc -w {{ template "artifactory-ha.nginx.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "artifactory-ha.nginx.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP/ + + {{- else if contains "ClusterIP" .Values.nginx.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "component={{ .Values.nginx.name }}" -o jsonpath="{.items[0].metadata.name}") + kubectl port-forward --namespace {{ .Release.Namespace }} $POD_NAME 8080:80 + echo http://127.0.0.1:8080 + + {{- end }} + +2. Open Artifactory in your browser + Default credential for Artifactory: + user: admin + password: password + + {{- if .Values.artifactory.license.secret }} + +3. Artifactory license(s) is deployed as a Kubernetes secret. This method is relevant for initial deployment only! + Updating the license should be done via Artifactory UI or REST API. If you want to keep managing the artifactory license using the same method, you can use artifactory.copyOnEveryStartup in values.yaml. + + {{- else }} + +3. Add HA licenses to activate Artifactory HA through the Artifactory UI + NOTE: Each Artifactory node requires a valid license. See https://www.jfrog.com/confluence/display/RTF/HA+Installation+and+Setup for more details. + + {{- end }} + +{{ if or .Values.artifactory.primary.javaOpts.jmx.enabled .Values.artifactory.node.javaOpts.jmx.enabled }} +JMX configuration: +{{- if not (contains "LoadBalancer" .Values.artifactory.service.type) }} +If you want to access JMX from you computer with jconsole, you should set ".Values.artifactory.service.type=LoadBalancer" !!! +{{ end }} + +1. Get the Artifactory service IP: +{{- if .Values.artifactory.primary.javaOpts.jmx.enabled }} +export PRIMARY_SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "artifactory-ha.primary.name" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') +{{- end }} +{{- if .Values.artifactory.node.javaOpts.jmx.enabled }} +export MEMBER_SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "artifactory-ha.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') +{{- end }} + +2. Map the service name to the service IP in /etc/hosts: +{{- if .Values.artifactory.primary.javaOpts.jmx.enabled }} +sudo sh -c "echo \"${PRIMARY_SERVICE_IP} {{ template "artifactory-ha.primary.name" . }}\" >> /etc/hosts" +{{- end }} +{{- if .Values.artifactory.node.javaOpts.jmx.enabled }} +sudo sh -c "echo \"${MEMBER_SERVICE_IP} {{ template "artifactory-ha.fullname" . }}\" >> /etc/hosts" +{{- end }} + +3. Launch jconsole: +{{- if .Values.artifactory.primary.javaOpts.jmx.enabled }} +jconsole {{ template "artifactory-ha.primary.name" . }}:{{ .Values.artifactory.primary.javaOpts.jmx.port }} +{{- end }} +{{- if .Values.artifactory.node.javaOpts.jmx.enabled }} +jconsole {{ template "artifactory-ha.fullname" . }}:{{ .Values.artifactory.node.javaOpts.jmx.port }} +{{- end }} +{{- end }} + + +{{- if ge (.Values.artifactory.node.replicaCount | int) 1 }} +***************************************** WARNING ***************************************************************************** +* Currently member node(s) are enabled, will be deprecated in upcoming releases * +* It is recommended to upgrade from primary-members to primary-only. * +* It can be done by deploying the chart ( >=107.59.x) with the new values. Also, please refer to changelog of 107.59.x chart * +* More Info: https://jfrog.com/help/r/jfrog-installation-setup-documentation/cloud-native-high-availability * +******************************************************************************************************************************* +{{- end }} + +{{- if and .Values.nginx.enabled .Values.ingress.hosts }} +***************************************** WARNING ***************************************************************************** +* when nginx is enabled , .Values.ingress.hosts will be deprecated in upcoming releases * +* It is recommended to use nginx.hosts instead ingress.hosts +******************************************************************************************************************************* +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/_helpers.tpl b/charts/jfrog/artifactory-ha/107.90.14/templates/_helpers.tpl new file mode 100644 index 0000000000..d6fb229fed --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/_helpers.tpl @@ -0,0 +1,563 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "artifactory-ha.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +The primary node name +*/}} +{{- define "artifactory-ha.primary.name" -}} +{{- if .Values.nameOverride -}} +{{- printf "%s-primary" .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := .Release.Name | trunc 29 -}} +{{- printf "%s-%s-primary" $name .Chart.Name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* +The member node name +*/}} +{{- define "artifactory-ha.node.name" -}} +{{- if .Values.nameOverride -}} +{{- printf "%s-member" .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := .Release.Name | trunc 29 -}} +{{- printf "%s-%s-member" $name .Chart.Name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* +Expand the name nginx service. +*/}} +{{- define "artifactory-ha.nginx.name" -}} +{{- default .Values.nginx.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 "artifactory-ha.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 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 "artifactory-ha.nginx.fullname" -}} +{{- if .Values.nginx.fullnameOverride -}} +{{- .Values.nginx.fullnameOverride | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- $name := default .Chart.Name .Values.nginx.name -}} +{{- 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 "artifactory-ha.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} +{{ default (include "artifactory-ha.fullname" .) .Values.serviceAccount.name }} +{{- else -}} +{{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "artifactory-ha.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Generate SSL certificates +*/}} +{{- define "artifactory-ha.gen-certs" -}} +{{- $altNames := list ( printf "%s.%s" (include "artifactory-ha.fullname" .) .Release.Namespace ) ( printf "%s.%s.svc" (include "artifactory-ha.fullname" .) .Release.Namespace ) -}} +{{- $ca := genCA "artifactory-ca" 365 -}} +{{- $cert := genSignedCert ( include "artifactory-ha.fullname" . ) nil $altNames 365 $ca -}} +tls.crt: {{ $cert.Cert | b64enc }} +tls.key: {{ $cert.Key | b64enc }} +{{- end -}} + +{{/* +Scheme (http/https) based on Access or Router TLS enabled/disabled +*/}} +{{- define "artifactory-ha.scheme" -}} +{{- if or .Values.access.accessConfig.security.tls .Values.router.tlsEnabled -}} +{{- printf "%s" "https" -}} +{{- else -}} +{{- printf "%s" "http" -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve joinKey value +*/}} +{{- define "artifactory-ha.joinKey" -}} +{{- if .Values.global.joinKey -}} +{{- .Values.global.joinKey -}} +{{- else if .Values.artifactory.joinKey -}} +{{- .Values.artifactory.joinKey -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve jfConnectToken value +*/}} +{{- define "artifactory-ha.jfConnectToken" -}} +{{- .Values.artifactory.jfConnectToken -}} +{{- end -}} + +{{/* +Resolve masterKey value +*/}} +{{- define "artifactory-ha.masterKey" -}} +{{- if .Values.global.masterKey -}} +{{- .Values.global.masterKey -}} +{{- else if .Values.artifactory.masterKey -}} +{{- .Values.artifactory.masterKey -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve joinKeySecretName value +*/}} +{{- define "artifactory-ha.joinKeySecretName" -}} +{{- if .Values.global.joinKeySecretName -}} +{{- .Values.global.joinKeySecretName -}} +{{- else if .Values.artifactory.joinKeySecretName -}} +{{- .Values.artifactory.joinKeySecretName -}} +{{- else -}} +{{ include "artifactory-ha.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Resolve jfConnectTokenSecretName value +*/}} +{{- define "artifactory-ha.jfConnectTokenSecretName" -}} +{{- if .Values.artifactory.jfConnectTokenSecretName -}} +{{- .Values.artifactory.jfConnectTokenSecretName -}} +{{- else -}} +{{ include "artifactory-ha.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Resolve masterKeySecretName value +*/}} +{{- define "artifactory-ha.masterKeySecretName" -}} +{{- if .Values.global.masterKeySecretName -}} +{{- .Values.global.masterKeySecretName -}} +{{- else if .Values.artifactory.masterKeySecretName -}} +{{- .Values.artifactory.masterKeySecretName -}} +{{- else -}} +{{ include "artifactory-ha.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Resolve imagePullSecrets value +*/}} +{{- define "artifactory-ha.imagePullSecrets" -}} +{{- if .Values.global.imagePullSecrets }} +imagePullSecrets: +{{- range .Values.global.imagePullSecrets }} + - name: {{ . }} +{{- end }} +{{- else if .Values.imagePullSecrets }} +imagePullSecrets: +{{- range .Values.imagePullSecrets }} + - name: {{ . }} +{{- end }} +{{- end -}} +{{- end -}} + +{{/* +Resolve customInitContainersBegin value +*/}} +{{- define "artifactory-ha.customInitContainersBegin" -}} +{{- if .Values.global.customInitContainersBegin -}} +{{- .Values.global.customInitContainersBegin -}} +{{- end -}} +{{- if .Values.artifactory.customInitContainersBegin -}} +{{- .Values.artifactory.customInitContainersBegin -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customInitContainers value +*/}} +{{- define "artifactory-ha.customInitContainers" -}} +{{- if .Values.global.customInitContainers -}} +{{- .Values.global.customInitContainers -}} +{{- end -}} +{{- if .Values.artifactory.customInitContainers -}} +{{- .Values.artifactory.customInitContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumes value +*/}} +{{- define "artifactory-ha.customVolumes" -}} +{{- if .Values.global.customVolumes -}} +{{- .Values.global.customVolumes -}} +{{- end -}} +{{- if .Values.artifactory.customVolumes -}} +{{- .Values.artifactory.customVolumes -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve unifiedCustomSecretVolumeName value +*/}} +{{- define "artifactory-ha.unifiedCustomSecretVolumeName" -}} +{{- printf "%s-%s" (include "artifactory-ha.name" .) ("unified-secret-volume") | trunc 63 -}} +{{- end -}} + +{{/* +Check the Duplication of volume names for secrets. If unifiedSecretInstallation is enabled then the method is checking for volume names, +if the volume exists in customVolume then an extra volume with the same name will not be getting added in unifiedSecretInstallation case.*/}} +{{- define "artifactory-ha.checkDuplicateUnifiedCustomVolume" -}} +{{- if or .Values.global.customVolumes .Values.artifactory.customVolumes -}} +{{- $val := (tpl (include "artifactory-ha.customVolumes" .) .) | toJson -}} +{{- contains (include "artifactory-ha.unifiedCustomSecretVolumeName" .) $val | toString -}} +{{- else -}} +{{- printf "%s" "false" -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumeMounts value +*/}} +{{- define "artifactory-ha.customVolumeMounts" -}} +{{- if .Values.global.customVolumeMounts -}} +{{- .Values.global.customVolumeMounts -}} +{{- end -}} +{{- if .Values.artifactory.customVolumeMounts -}} +{{- .Values.artifactory.customVolumeMounts -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customSidecarContainers value +*/}} +{{- define "artifactory-ha.customSidecarContainers" -}} +{{- if .Values.global.customSidecarContainers -}} +{{- .Values.global.customSidecarContainers -}} +{{- end -}} +{{- if .Values.artifactory.customSidecarContainers -}} +{{- .Values.artifactory.customSidecarContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper artifactory chart image names +*/}} +{{- define "artifactory-ha.getImageInfoByValue" -}} +{{- $dot := index . 0 }} +{{- $indexReference := index . 1 }} +{{- $registryName := index $dot.Values $indexReference "image" "registry" -}} +{{- $repositoryName := index $dot.Values $indexReference "image" "repository" -}} +{{- $tag := "" -}} +{{- if and (eq $indexReference "artifactory") (hasKey $dot.Values "artifactoryService") }} + {{- if default false $dot.Values.artifactoryService.enabled }} + {{- $indexReference = "artifactoryService" -}} + {{- $tag = default $dot.Chart.Annotations.artifactoryServiceVersion (index $dot.Values $indexReference "image" "tag") | toString -}} + {{- $repositoryName = index $dot.Values $indexReference "image" "repository" -}} + {{- else -}} + {{- $tag = default $dot.Chart.AppVersion (index $dot.Values $indexReference "image" "tag") | toString -}} + {{- end -}} +{{- else -}} + {{- $tag = default $dot.Chart.AppVersion (index $dot.Values $indexReference "image" "tag") | toString -}} +{{- end -}} +{{- if $dot.Values.global }} + {{- if and $dot.Values.splitServicesToContainers $dot.Values.global.versions.router (eq $indexReference "router") }} + {{- $tag = $dot.Values.global.versions.router | toString -}} + {{- end -}} + {{- if and $dot.Values.global.versions.initContainers (eq $indexReference "initContainers") }} + {{- $tag = $dot.Values.global.versions.initContainers | toString -}} + {{- end -}} + {{- if $dot.Values.global.versions.artifactory }} + {{- if or (eq $indexReference "artifactory") (eq $indexReference "metadata") (eq $indexReference "nginx") (eq $indexReference "observability") }} + {{- $tag = $dot.Values.global.versions.artifactory | toString -}} + {{- end -}} + {{- end -}} + {{- if $dot.Values.global.imageRegistry }} + {{- printf "%s/%s:%s" $dot.Values.global.imageRegistry $repositoryName $tag -}} + {{- else -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} + {{- end -}} +{{- else -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper artifactory app version +*/}} +{{- define "artifactory-ha.app.version" -}} +{{- $tag := (splitList ":" ((include "artifactory-ha.getImageInfoByValue" (list . "artifactory" )))) | last | toString -}} +{{- printf "%s" $tag -}} +{{- end -}} + +{{/* +Custom certificate copy command +*/}} +{{- define "artifactory-ha.copyCustomCerts" -}} +echo "Copy custom certificates to {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted"; +mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted; +for file in $(ls -1 /tmp/certs/* | grep -v .key | grep -v ":" | grep -v grep); do if [ -f "${file}" ]; then cp -v ${file} {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted; fi done; +if [ -f {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted/tls.crt ]; then mv -v {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted/tls.crt {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted/ca.crt; fi; +{{- end -}} + +{{/* +Circle of trust certificates copy command +*/}} +{{- define "artifactory.copyCircleOfTrustCertsCerts" -}} +echo "Copy circle of trust certificates to {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted"; +mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; +for file in $(ls -1 /tmp/circleoftrustcerts/* | grep -v .key | grep -v ":" | grep -v grep); do if [ -f "${file}" ]; then cp -v ${file} {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; fi done; +{{- end -}} + +{{/* +Resolve requiredServiceTypes value +*/}} +{{- define "artifactory-ha.router.requiredServiceTypes" -}} +{{- $requiredTypes := "jfrt,jfac" -}} +{{- if not .Values.access.enabled -}} + {{- $requiredTypes = "jfrt" -}} +{{- end -}} +{{- if .Values.observability.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfob" -}} +{{- end -}} +{{- if .Values.metadata.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfmd" -}} +{{- end -}} +{{- if .Values.event.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfevt" -}} +{{- end -}} +{{- if .Values.frontend.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jffe" -}} +{{- end -}} +{{- if .Values.jfconnect.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfcon" -}} +{{- end -}} +{{- if .Values.evidence.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfevd" -}} +{{- end -}} +{{- if .Values.mc.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfmc" -}} +{{- end -}} +{{- $requiredTypes -}} +{{- end -}} + +{{/* +nginx scheme (http/https) +*/}} +{{- define "nginx.scheme" -}} +{{- if .Values.nginx.http.enabled -}} +{{- printf "%s" "http" -}} +{{- else -}} +{{- printf "%s" "https" -}} +{{- end -}} +{{- end -}} + + +{{/* +nginx command +*/}} +{{- define "nginx.command" -}} +{{- if .Values.nginx.customCommand }} +{{ toYaml .Values.nginx.customCommand }} +{{- end }} +{{- end -}} + +{{/* +nginx port (8080/8443) based on http/https enabled +*/}} +{{- define "nginx.port" -}} +{{- if .Values.nginx.http.enabled -}} +{{- .Values.nginx.http.internalPort -}} +{{- else -}} +{{- .Values.nginx.https.internalPort -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customInitContainers value +*/}} +{{- define "artifactory.nginx.customInitContainers" -}} +{{- if .Values.nginx.customInitContainers -}} +{{- .Values.nginx.customInitContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumes value +*/}} +{{- define "artifactory.nginx.customVolumes" -}} +{{- if .Values.nginx.customVolumes -}} +{{- .Values.nginx.customVolumes -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumeMounts nginx value +*/}} +{{- define "artifactory.nginx.customVolumeMounts" -}} +{{- if .Values.nginx.customVolumeMounts -}} +{{- .Values.nginx.customVolumeMounts -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customSidecarContainers value +*/}} +{{- define "artifactory.nginx.customSidecarContainers" -}} +{{- if .Values.nginx.customSidecarContainers -}} +{{- .Values.nginx.customSidecarContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve Artifactory pod primary node selector value +*/}} +{{- define "artifactory.nodeSelector" -}} +nodeSelector: +{{- if .Values.global.nodeSelector }} +{{ toYaml .Values.global.nodeSelector | indent 2 }} +{{- else if .Values.artifactory.primary.nodeSelector }} +{{ toYaml .Values.artifactory.primary.nodeSelector | indent 2 }} +{{- end -}} +{{- end -}} + +{{/* +Resolve Artifactory pod node nodeselector value +*/}} +{{- define "artifactory.node.nodeSelector" -}} +nodeSelector: +{{- if .Values.global.nodeSelector }} +{{ toYaml .Values.global.nodeSelector | indent 2 }} +{{- else if .Values.artifactory.node.nodeSelector }} +{{ toYaml .Values.artifactory.node.nodeSelector | indent 2 }} +{{- end -}} +{{- end -}} + +{{/* +Resolve Nginx pods node selector value +*/}} +{{- define "nginx.nodeSelector" -}} +nodeSelector: +{{- if .Values.global.nodeSelector }} +{{ toYaml .Values.global.nodeSelector | indent 2 }} +{{- else if .Values.nginx.nodeSelector }} +{{ toYaml .Values.nginx.nodeSelector | indent 2 }} +{{- end -}} +{{- end -}} + +{{/* +Calculate the systemYaml from structured and unstructured text input +*/}} +{{- define "artifactory.finalSystemYaml" -}} +{{ tpl (mergeOverwrite (include "artifactory.systemYaml" . | fromYaml) .Values.artifactory.extraSystemYaml | toYaml) . }} +{{- end -}} + +{{/* +Calculate the systemYaml from the unstructured text input +*/}} +{{- define "artifactory.systemYaml" -}} +{{ include (print $.Template.BasePath "/_system-yaml-render.tpl") . }} +{{- end -}} + +{{/* +Metrics enabled +*/}} +{{- define "metrics.enabled" -}} +shared: + metrics: + enabled: true +{{- end }} + +{{/* +Resolve artifactory metrics +*/}} +{{- define "artifactory.metrics" -}} +{{- if .Values.artifactory.openMetrics -}} +{{- if .Values.artifactory.openMetrics.enabled -}} +{{ include "metrics.enabled" . }} +{{- if .Values.artifactory.openMetrics.filebeat }} +{{- if .Values.artifactory.openMetrics.filebeat.enabled }} +{{ include "metrics.enabled" . }} + filebeat: +{{ tpl (.Values.artifactory.openMetrics.filebeat | toYaml) . | indent 6 }} +{{- end -}} +{{- end -}} +{{- end -}} +{{- else if .Values.artifactory.metrics -}} +{{- if .Values.artifactory.metrics.enabled -}} +{{ include "metrics.enabled" . }} +{{- if .Values.artifactory.metrics.filebeat }} +{{- if .Values.artifactory.metrics.filebeat.enabled }} +{{ include "metrics.enabled" . }} + filebeat: +{{ tpl (.Values.artifactory.metrics.filebeat | toYaml) . | indent 6 }} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve unified secret prepend release name +*/}} +{{- define "artifactory.unifiedSecretPrependReleaseName" -}} +{{- if .Values.artifactory.unifiedSecretPrependReleaseName }} +{{- printf "%s" (include "artifactory-ha.fullname" .) -}} +{{- else }} +{{- printf "%s" (include "artifactory-ha.name" .) -}} +{{- end }} +{{- end }} + +{{/* +Resolve nginx hosts value +*/}} +{{- define "artifactory.nginx.hosts" -}} +{{- if .Values.ingress.hosts }} +{{- range .Values.ingress.hosts -}} + {{- if contains "." . -}} + {{ "" | indent 0 }} ~(?.+)\.{{ . }} + {{- end -}} +{{- end -}} +{{- else if .Values.nginx.hosts }} +{{- range .Values.nginx.hosts -}} + {{- if contains "." . -}} + {{ "" | indent 0 }} ~(?.+)\.{{ . }} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/_system-yaml-render.tpl b/charts/jfrog/artifactory-ha/107.90.14/templates/_system-yaml-render.tpl new file mode 100644 index 0000000000..deaa773ea9 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/_system-yaml-render.tpl @@ -0,0 +1,5 @@ +{{- if .Values.artifactory.systemYaml -}} +{{- tpl .Values.artifactory.systemYaml . -}} +{{- else -}} +{{ (tpl ( $.Files.Get "files/system.yaml" ) .) }} +{{- end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/additional-resources.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/additional-resources.yaml new file mode 100644 index 0000000000..c4d06f08ad --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/additional-resources.yaml @@ -0,0 +1,3 @@ +{{ if .Values.additionalResources }} +{{ tpl .Values.additionalResources . }} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/admin-bootstrap-creds.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/admin-bootstrap-creds.yaml new file mode 100644 index 0000000000..40d91f75e9 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/admin-bootstrap-creds.yaml @@ -0,0 +1,15 @@ +{{- if not (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) }} +{{- if and .Values.artifactory.admin.password (not .Values.artifactory.unifiedSecretInstallation) }} +kind: Secret +apiVersion: v1 +metadata: + name: {{ template "artifactory-ha.fullname" . }}-bootstrap-creds + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + bootstrap.creds: {{ (printf "%s@%s=%s" .Values.artifactory.admin.username .Values.artifactory.admin.ip .Values.artifactory.admin.password) | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-access-config.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-access-config.yaml new file mode 100644 index 0000000000..0b96a337d4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-access-config.yaml @@ -0,0 +1,15 @@ +{{- if and .Values.access.accessConfig (not .Values.artifactory.unifiedSecretInstallation) }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory-ha.fullname" . }}-access-config + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +type: Opaque +stringData: + access.config.patch.yml: | +{{ tpl (toYaml .Values.access.accessConfig) . | indent 4 }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-binarystore-secret.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-binarystore-secret.yaml new file mode 100644 index 0000000000..6824fe90f8 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-binarystore-secret.yaml @@ -0,0 +1,18 @@ +{{- if and (not .Values.artifactory.persistence.customBinarystoreXmlSecret) (not .Values.artifactory.unifiedSecretInstallation) }} +kind: Secret +apiVersion: v1 +metadata: + name: {{ template "artifactory-ha.fullname" . }}-binarystore + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +stringData: + binarystore.xml: |- +{{- if .Values.artifactory.persistence.binarystoreXml }} +{{ tpl .Values.artifactory.persistence.binarystoreXml . | indent 4 }} +{{- else }} +{{ tpl ( .Files.Get "files/binarystore.xml" ) . | indent 4 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-configmaps.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-configmaps.yaml new file mode 100644 index 0000000000..1385bc5782 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-configmaps.yaml @@ -0,0 +1,13 @@ +{{ if .Values.artifactory.configMaps }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory-ha.fullname" . }}-configmaps + labels: + app: {{ template "artifactory-ha.fullname" . }} + chart: {{ .Chart.Name }}-{{ .Chart.Version }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: +{{ tpl .Values.artifactory.configMaps . | indent 2 }} +{{ end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-custom-secrets.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-custom-secrets.yaml new file mode 100644 index 0000000000..8065fe6865 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-custom-secrets.yaml @@ -0,0 +1,19 @@ +{{- if and .Values.artifactory.customSecrets (not .Values.artifactory.unifiedSecretInstallation) }} +{{- range .Values.artifactory.customSecrets }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory-ha.fullname" $ }}-{{ .name }} + labels: + app: "{{ template "artifactory-ha.name" $ }}" + chart: "{{ template "artifactory-ha.chart" $ }}" + component: "{{ $.Values.artifactory.name }}" + heritage: {{ $.Release.Service | quote }} + release: {{ $.Release.Name | quote }} +type: Opaque +stringData: + {{ .key }}: | +{{ .data | indent 4 -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-database-secrets.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-database-secrets.yaml new file mode 100644 index 0000000000..6daf5db7bc --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-database-secrets.yaml @@ -0,0 +1,24 @@ +{{- if and (not .Values.database.secrets) (not .Values.postgresql.enabled) (not .Values.artifactory.unifiedSecretInstallation) }} +{{- if or .Values.database.url .Values.database.user .Values.database.password }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory-ha.fullname" . }}-database-creds + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +type: Opaque +data: + {{- with .Values.database.url }} + db-url: {{ tpl . $ | b64enc | quote }} + {{- end }} + {{- with .Values.database.user }} + db-user: {{ tpl . $ | b64enc | quote }} + {{- end }} + {{- with .Values.database.password }} + db-password: {{ tpl . $ | b64enc | quote }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-gcp-credentials-secret.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-gcp-credentials-secret.yaml new file mode 100644 index 0000000000..d90769595b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-gcp-credentials-secret.yaml @@ -0,0 +1,16 @@ +{{- if not .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} +{{- if and (.Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled) (not .Values.artifactory.unifiedSecretInstallation) }} +kind: Secret +apiVersion: v1 +metadata: + name: {{ template "artifactory-ha.fullname" . }}-gcpcreds + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +stringData: + gcp.credentials.json: |- +{{ tpl .Values.artifactory.persistence.googleStorage.gcpServiceAccount.config . | indent 4 }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-installer-info.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-installer-info.yaml new file mode 100644 index 0000000000..0dff9dc86f --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-installer-info.yaml @@ -0,0 +1,16 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + name: {{ template "artifactory-ha.fullname" . }}-installer-info + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + installer-info.json: | +{{- if .Values.installerInfo -}} +{{- tpl .Values.installerInfo . | nindent 4 -}} +{{- else -}} +{{ (tpl ( .Files.Get "files/installer-info.json" | nindent 4 ) .) }} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-license-secret.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-license-secret.yaml new file mode 100644 index 0000000000..73f900863b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-license-secret.yaml @@ -0,0 +1,16 @@ +{{ if and (not .Values.artifactory.unifiedSecretInstallation) (not .Values.artifactory.license.secret) (not .Values.artifactory.license.licenseKey) }} +{{- with .Values.artifactory.license.licenseKey }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory-ha.fullname" $ }}-license + labels: + app: {{ template "artifactory-ha.name" $ }} + chart: {{ template "artifactory-ha.chart" $ }} + heritage: {{ $.Release.Service }} + release: {{ $.Release.Name }} +type: Opaque +data: + artifactory.lic: {{ . | b64enc | quote }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-migration-scripts.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-migration-scripts.yaml new file mode 100644 index 0000000000..fe40f980fc --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-migration-scripts.yaml @@ -0,0 +1,18 @@ +{{- if .Values.artifactory.migration.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory-ha.fullname" . }}-migration-scripts + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + migrate.sh: | +{{ .Files.Get "files/migrate.sh" | indent 4 }} + migrationHelmInfo.yaml: | +{{ .Files.Get "files/migrationHelmInfo.yaml" | indent 4 }} + migrationStatus.sh: | +{{ .Files.Get "files/migrationStatus.sh" | indent 4 }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-networkpolicy.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-networkpolicy.yaml new file mode 100644 index 0000000000..9924448f04 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-networkpolicy.yaml @@ -0,0 +1,34 @@ +{{- range .Values.networkpolicy }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "artifactory-ha.fullname" $ }}-{{ .name }}-networkpolicy + labels: + app: {{ template "artifactory-ha.name" $ }} + chart: {{ template "artifactory-ha.chart" $ }} + release: {{ $.Release.Name }} + heritage: {{ $.Release.Service }} +spec: +{{- if .podSelector }} + podSelector: +{{ .podSelector | toYaml | trimSuffix "\n" | indent 4 -}} +{{ else }} + podSelector: {} +{{- end }} + policyTypes: + {{- if .ingress }} + - Ingress + {{- end }} + {{- if .egress }} + - Egress + {{- end }} +{{- if .ingress }} + ingress: +{{ .ingress | toYaml | trimSuffix "\n" | indent 2 -}} +{{- end }} +{{- if .egress }} + egress: +{{ .egress | toYaml | trimSuffix "\n" | indent 2 -}} +{{- end }} +--- +{{- end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-nfs-pvc.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-nfs-pvc.yaml new file mode 100644 index 0000000000..6ed7d82f6b --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-nfs-pvc.yaml @@ -0,0 +1,101 @@ +{{- if eq .Values.artifactory.persistence.type "nfs" }} +### Artifactory HA data +apiVersion: v1 +kind: PersistentVolume +metadata: + name: {{ template "artifactory-ha.fullname" . }}-data-pv + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + id: {{ template "artifactory-ha.name" . }}-data-pv + type: nfs-volume +spec: + {{- if .Values.artifactory.persistence.nfs.mountOptions }} + mountOptions: +{{ toYaml .Values.artifactory.persistence.nfs.mountOptions | indent 4 }} + {{- end }} + capacity: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain + nfs: + server: {{ .Values.artifactory.persistence.nfs.ip }} + path: "{{ .Values.artifactory.persistence.nfs.haDataMount }}" + readOnly: false +--- +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: + name: {{ template "artifactory-ha.fullname" . }}-data-pvc + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + type: nfs-volume +spec: + accessModes: + - ReadWriteOnce + storageClassName: "" + resources: + requests: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + selector: + matchLabels: + id: {{ template "artifactory-ha.name" . }}-data-pv + app: {{ template "artifactory-ha.name" . }} + release: {{ .Release.Name }} +--- +### Artifactory HA backup +apiVersion: v1 +kind: PersistentVolume +metadata: + name: {{ template "artifactory-ha.fullname" . }}-backup-pv + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + id: {{ template "artifactory-ha.name" . }}-backup-pv + type: nfs-volume +spec: + {{- if .Values.artifactory.persistence.nfs.mountOptions }} + mountOptions: +{{ toYaml .Values.artifactory.persistence.nfs.mountOptions | indent 4 }} + {{- end }} + capacity: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain + nfs: + server: {{ .Values.artifactory.persistence.nfs.ip }} + path: "{{ .Values.artifactory.persistence.nfs.haBackupMount }}" + readOnly: false +--- +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: + name: {{ template "artifactory-ha.fullname" . }}-backup-pvc + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + type: nfs-volume +spec: + accessModes: + - ReadWriteOnce + storageClassName: "" + resources: + requests: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + selector: + matchLabels: + id: {{ template "artifactory-ha.name" . }}-backup-pv + app: {{ template "artifactory-ha.name" . }} + release: {{ .Release.Name }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-node-pdb.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-node-pdb.yaml new file mode 100644 index 0000000000..46c6dac21d --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/artifactory-node-pdb.yaml @@ -0,0 +1,26 @@ +{{- if gt (.Values.artifactory.node.replicaCount | int) 0 -}} +{{- if .Values.artifactory.node.minAvailable -}} +{{- if semverCompare " + mkdir -p {{ tpl .Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir . }}; + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + volumeMounts: + - mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + name: volume + {{- end }} + {{- end }} + {{- if .Values.artifactory.deleteDBPropertiesOnStartup }} + - name: "delete-db-properties" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'bash' + - '-c' + - 'rm -fv {{ .Values.artifactory.persistence.mountPath }}/etc/db.properties' + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + volumeMounts: + - mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + name: volume + {{- end }} + {{- end }} + {{- if and .Values.artifactory.node.waitForPrimaryStartup.enabled }} + - name: "wait-for-primary" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - 'bash' + - '-c' + - > + echo "Waiting for primary node to be ready..."; + {{- if and .Values.artifactory.node.waitForPrimaryStartup.enabled .Values.artifactory.node.waitForPrimaryStartup.time }} + echo "Sleeping to allow time for primary node to come up"; + sleep {{ .Values.artifactory.node.waitForPrimaryStartup.time }}; + {{- else }} + ready=false; + while ! $ready; do echo Primary not ready. Waiting...; + timeout 2s bash -c " + if [[ -e "{{ .Values.artifactory.persistence.mountPath }}/etc/filebeat.yaml" ]]; then chmod 644 {{ .Values.artifactory.persistence.mountPath }}/etc/filebeat.yaml; fi; + echo "Copy system.yaml to {{ .Values.artifactory.persistence.mountPath }}/etc"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; + {{- if .Values.systemYamlOverride.existingSecret }} + cp -fv /tmp/etc/{{ .Values.systemYamlOverride.dataKey }} {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml; + {{- else }} + cp -fv /tmp/etc/system.yaml {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml; + {{- end }} + echo "Copy binarystore.xml file"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/artifactory; + cp -fv /tmp/etc/artifactory/binarystore.xml {{ .Values.artifactory.persistence.mountPath }}/etc/artifactory/binarystore.xml; + echo "Removing join.key file"; + rm -fv {{ .Values.artifactory.persistence.mountPath }}/etc/security/join.key; + {{- if .Values.access.resetAccessCAKeys }} + echo "Resetting Access CA Keys - load from database"; + {{- end }} + {{- if .Values.access.customCertificatesSecretName }} + echo "Load custom certificates from database"; + {{- end }} + {{- if or .Values.artifactory.masterKey .Values.global.masterKey .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + echo "Copy masterKey to {{ .Values.artifactory.persistence.mountPath }}/etc/security"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/security; + echo -n ${ARTIFACTORY_MASTER_KEY} > {{ .Values.artifactory.persistence.mountPath }}/etc/security/master.key; + env: + - name: ARTIFACTORY_MASTER_KEY + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) (or .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName) }} + name: {{ include "artifactory-ha.masterKeySecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: master-key + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + + ######################## SystemYaml ######################### + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.systemYamlOverride.existingSecret }} + - name: systemyaml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + {{- if .Values.systemYamlOverride.existingSecret }} + mountPath: "/tmp/etc/{{.Values.systemYamlOverride.dataKey}}" + subPath: {{ .Values.systemYamlOverride.dataKey }} + {{- else }} + mountPath: "/tmp/etc/system.yaml" + subPath: system.yaml + {{- end }} + + ######################## Binarystore ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## CustomCertificates ########################## + {{- if or .Values.artifactory.customCertificates.enabled .Values.global.customCertificates.enabled }} + - name: copy-custom-certificates + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > +{{ include "artifactory-ha.copyCustomCerts" . | indent 10 }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath }} + - name: ca-certs + mountPath: "/tmp/certs" + {{- end }} + + {{- if .Values.artifactory.circleOfTrustCertificatesSecret }} + - name: copy-circle-of-trust-certificates + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > +{{ include "artifactory.copyCircleOfTrustCertsCerts" . | indent 10 }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath }} + - name: circle-of-trust-certs + mountPath: "/tmp/circleoftrustcerts" + {{- end }} + + {{- if .Values.waitForDatabase }} + {{- if or .Values.postgresql.enabled }} + - name: "wait-for-db" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - /bin/bash + - -c + - | + echo "Waiting for postgresql to come up" + ready=false; + while ! $ready; do echo waiting; + timeout 2s bash -c " + {{- if .Values.artifactory.migration.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.artifactory.migration.preStartCommand . }}; + {{- end }} + scriptsPath="/opt/jfrog/artifactory/app/bin"; + mkdir -p $scriptsPath; + echo "Copy migration scripts and Run migration"; + cp -fv /tmp/migrate.sh $scriptsPath/migrate.sh; + cp -fv /tmp/migrationHelmInfo.yaml $scriptsPath/migrationHelmInfo.yaml; + cp -fv /tmp/migrationStatus.sh $scriptsPath/migrationStatus.sh; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/log; + bash $scriptsPath/migrationStatus.sh {{ include "artifactory-ha.app.version" . }} {{ .Values.artifactory.migration.timeoutSeconds }} > >(tee {{ .Values.artifactory.persistence.mountPath }}/log/helm-migration.log) 2>&1; + resources: +{{ toYaml .Values.artifactory.node.resources | indent 10 }} + env: + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} + - name: JF_SHARED_NODE_HAENABLED + value: "true" +{{- with .Values.artifactory.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: migration-scripts + mountPath: "/tmp/migrate.sh" + subPath: migrate.sh + - name: migration-scripts + mountPath: "/tmp/migrationHelmInfo.yaml" + subPath: migrationHelmInfo.yaml + - name: migration-scripts + mountPath: "/tmp/migrationStatus.sh" + subPath: migrationStatus.sh + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) }} + - name: artifactory-ha-data-{{ $sharedClaimNumber }} + mountPath: "{{ tpl $.Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir $ }}/filestore{{ $sharedClaimNumber }}" + {{- end }} + - name: artifactory-ha-backup + mountPath: "{{ $.Values.artifactory.persistence.fileSystem.existingSharedClaim.backupDir }}" + {{- end }} + {{- end }} + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-ha-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-ha-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + + ######################## Artifactory persistence binarystore Xml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + {{- end }} + + ######################## Artifactory persistence google storage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + +{{- end }} + {{- if .Values.hostAliases }} + hostAliases: +{{ toYaml .Values.hostAliases | indent 6 }} + {{- end }} + containers: + {{- if .Values.splitServicesToContainers }} + - name: {{ .Values.router.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "router") }} + imagePullPolicy: {{ .Values.router.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/router/app/bin/entrypoint-router.sh + {{- with .Values.router.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_ROUTER_TOPOLOGY_LOCAL_REQUIREDSERVICETYPES + value: {{ include "artifactory-ha.router.requiredServiceTypes" . }} +{{- with .Values.router.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - name: http + containerPort: {{ .Values.router.internalPort }} + volumeMounts: + - name: volume + mountPath: {{ .Values.router.persistence.mountPath | quote }} +{{- with .Values.router.customVolumeMounts }} +{{ tpl . $ | indent 8 }} +{{- end }} + resources: +{{ toYaml .Values.router.resources | indent 10 }} + {{- if .Values.router.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.router.startupProbe.config . | indent 10 }} + {{- end }} +{{- if .Values.router.readinessProbe.enabled }} + readinessProbe: +{{ tpl .Values.router.readinessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.router.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.router.livenessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.frontend.enabled }} + - name: {{ .Values.frontend.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/third-party/node/bin/node /opt/jfrog/artifactory/app/frontend/bin/server/dist/bundle.js /opt/jfrog/artifactory/app/frontend + {{- with .Values.frontend.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + - name : JF_SHARED_NODE_HAENABLED + value: "true" +{{- with .Values.frontend.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.frontend.resources | indent 10 }} + {{- if .Values.frontend.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.frontend.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.frontend.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.frontend.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.evidence.enabled }} + - name: {{ .Values.evidence.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/evidence/bin/jf-evidence start + {{- with .Values.evidence.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.evidence.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.evidence.internalPort }} + name: http-evidence + - containerPort: {{ .Values.evidence.externalPort }} + name: grpc-evidence + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.evidence.resources | indent 10 }} + {{- if .Values.evidence.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.evidence.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.evidence.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.evidence.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.metadata.enabled }} + - name: {{ .Values.metadata.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "metadata") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/metadata/bin/jf-metadata start + {{- with .Values.metadata.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.metadata.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.metadata.resources | indent 10 }} + {{- if .Values.metadata.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.metadata.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.metadata.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.metadata.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.event.enabled }} + - name: {{ .Values.event.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/event/bin/jf-event start + {{- with .Values.event.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.event.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.event.resources | indent 10 }} + {{- if .Values.event.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.event.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.event.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.event.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.jfconnect.enabled }} + - name: {{ .Values.jfconnect.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/jfconnect/bin/jf-connect start + {{- with .Values.jfconnect.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.jfconnect.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.jfconnect.resources | indent 10 }} + {{- if .Values.jfconnect.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.jfconnect.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.jfconnect.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.jfconnect.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if and .Values.federation.enabled .Values.federation.embedded }} + - name: {{ .Values.federation.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/third-party/java/bin/java {{ .Values.federation.extraJavaOpts }} -jar /opt/jfrog/artifactory/app/rtfs/lib/jf-rtfs + {{- with .Values.federation.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_RTFS_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} +{{- with .Values.federation.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.federation.internalPort }} + name: http-rtfs + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.federation.resources | indent 10 }} + {{- if .Values.federation.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.federation.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.federation.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.federation.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.observability.enabled }} + - name: {{ .Values.observability.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "observability") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/observability/bin/jf-observability start + {{- with .Values.observability.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.observability.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.observability.resources | indent 10 }} + {{- if .Values.observability.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.observability.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.observability.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.observability.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if and .Values.access.enabled (not (.Values.access.runOnArtifactoryTomcat | default false)) }} + - name: {{ .Values.access.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.access.resources }} + resources: +{{ toYaml .Values.access.resources | indent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + set -e; + {{- if .Values.access.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.access.preStartCommand . }}; + {{- end }} + exec /opt/jfrog/artifactory/app/access/bin/entrypoint-access.sh + {{- with .Values.access.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.access.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + {{- if .Values.artifactory.customPersistentVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.artifactory.customPersistentPodVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentPodVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentPodVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: awsmp-product-license + mountPath: "/var/run/secrets/product-license" + {{- end }} + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + + ######################## Artifactory persistence fs ########################## + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) }} + - name: artifactory-ha-data-{{ $sharedClaimNumber }} + mountPath: "{{ tpl $.Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir $ }}/filestore{{ $sharedClaimNumber }}" + {{- end }} + - name: artifactory-ha-backup + mountPath: "{{ $.Values.artifactory.persistence.fileSystem.existingSharedClaim.backupDir }}" + {{- end }} + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-ha-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-ha-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystore Xml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Artifactory persistence google storage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + + ######################## Artifactory ConfigMap ########################## + {{- if .Values.artifactory.configMapName }} + - name: bootstrap-config + mountPath: "/bootstrap/" + {{- end }} + + ######################## Artifactory license ########################## + {{- if or .Values.artifactory.license.secret .Values.artifactory.license.licenseKey }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.license.secret }} + - name: artifactory-license + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/artifactory.cluster.license" + {{- if .Values.artifactory.license.secret }} + subPath: {{ .Values.artifactory.license.dataKey }} + {{- else if .Values.artifactory.license.licenseKey }} + subPath: artifactory.lic + {{- end }} + {{- end }} + {{- end }} + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + {{- if .Values.access.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.access.startupProbe.config . | indent 10 }} + {{- end }} + {{- if semverCompare " + set -e; + {{- range .Values.artifactory.copyOnEveryStartup }} + {{- $targetPath := printf "%s/%s" $.Values.artifactory.persistence.mountPath .target }} + {{- $baseDirectory := regexFind ".*/" $targetPath }} + mkdir -p {{ $baseDirectory }}; + cp -Lrf {{ .source }} {{ $.Values.artifactory.persistence.mountPath }}/{{ .target }}; + {{- end }} + {{- if .Values.artifactory.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.artifactory.preStartCommand . }}; + {{- end }} + {{- with .Values.artifactory.node.preStartCommand }} + echo "Running member node specific custom preStartCommand command"; + {{ tpl . $ }}; + {{- end }} + exec /entrypoint-artifactory.sh + {{- with .Values.artifactory.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + {{- if .Values.aws.license.enabled }} + - name: IS_AWS_LICENSE + value: "true" + - name: AWS_REGION + value: {{ .Values.aws.region | quote }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: AWS_WEB_IDENTITY_REFRESH_TOKEN_FILE + value: "/var/run/secrets/product-license/license_token" + - name: AWS_ROLE_ARN + valueFrom: + secretKeyRef: + name: {{ .Values.aws.licenseConfigSecretName }} + key: iam_role + {{- end }} + {{- end }} + {{- if .Values.splitServicesToContainers }} + - name : JF_ROUTER_ENABLED + value: "true" + - name : JF_ROUTER_SERVICE_ENABLED + value: "false" + - name : JF_EVENT_ENABLED + value: "false" + - name : JF_METADATA_ENABLED + value: "false" + - name : JF_FRONTEND_ENABLED + value: "false" + - name: JF_FEDERATION_ENABLED + value: "false" + - name : JF_OBSERVABILITY_ENABLED + value: "false" + - name : JF_JFCONNECT_SERVICE_ENABLED + value: "false" + - name : JF_EVIDENCE_ENABLED + value: "false" + {{- if not (.Values.access.runOnArtifactoryTomcat | default false) }} + - name : JF_ACCESS_ENABLED + value: "false" + {{- end}} + {{- end }} + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} + - name: JF_SHARED_NODE_HAENABLED + value: "true" +{{- with .Values.artifactory.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.artifactory.internalPort }} + name: http + - containerPort: {{ .Values.artifactory.internalArtifactoryPort }} + name: http-internal + - containerPort: {{ .Values.federation.internalPort }} + name: http-rtfs + {{- if .Values.artifactory.node.javaOpts.jmx.enabled }} + - containerPort: {{ .Values.artifactory.node.javaOpts.jmx.port }} + name: tcp-jmx + {{- end }} + {{- if .Values.artifactory.ssh.enabled }} + - containerPort: {{ .Values.artifactory.ssh.internalPort }} + name: tcp-ssh + {{- end }} + volumeMounts: + {{- if .Values.artifactory.customPersistentVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.artifactory.customPersistentPodVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentPodVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentPodVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: awsmp-product-license + mountPath: "/var/run/secrets/product-license" + {{- end }} + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + + ######################## Artifactory persistence fs ########################## + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) }} + - name: artifactory-ha-data-{{ $sharedClaimNumber }} + mountPath: "{{ tpl $.Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir $ }}/filestore{{ $sharedClaimNumber }}" + {{- end }} + - name: artifactory-ha-backup + mountPath: "{{ $.Values.artifactory.persistence.fileSystem.existingSharedClaim.backupDir }}" + {{- end }} + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-ha-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-ha-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystore Xml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Artifactory persistence google storage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + + ######################## Artifactory ConfigMap ########################## + {{- if .Values.artifactory.configMapName }} + - name: bootstrap-config + mountPath: "/bootstrap/" + {{- end }} + + ######################## Artifactory license ########################## + {{- if or .Values.artifactory.license.secret .Values.artifactory.license.licenseKey }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.license.secret }} + - name: artifactory-license + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/artifactory.cluster.license" + {{- if .Values.artifactory.license.secret }} + subPath: {{ .Values.artifactory.license.dataKey }} + {{- else if .Values.artifactory.license.licenseKey }} + subPath: artifactory.lic + {{- end }} + {{- end }} + {{- end }} + - name: installer-info + mountPath: "/artifactory_bootstrap/info/installer-info.json" + subPath: installer-info.json + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + resources: +{{ toYaml .Values.artifactory.node.resources | indent 10 }} + {{- if .Values.artifactory.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.artifactory.startupProbe.config . | indent 10 }} + {{- end }} + {{- if and (not .Values.splitServicesToContainers) (semverCompare "= 107.79.x), just set databaseUpgradeReady=true \n" .Values.databaseUpgradeReady | quote }} +{{- end }} +{{- if .Values.artifactory.postStartCommand }} + {{- fail ".Values.artifactory.postStartCommand is not supported and should be replaced with .Values.artifactory.lifecycle.postStart.exec.command" }} +{{- end }} +{{- if eq .Values.artifactory.persistence.type "aws-s3" }} + {{- fail "\nPersistence storage type 'aws-s3' is deprecated and is not supported and should be replaced with 'aws-s3-v3'" }} +{{- end }} +{{- if or .Values.artifactory.persistence.googleStorage.identity .Values.artifactory.persistence.googleStorage.credential }} + {{- fail "\nGCP Bucket Authentication with Identity and Credential is deprecated" }} +{{- end }} +{{- if (eq (.Values.artifactory.setSecurityContext | toString) "false" ) }} + {{- fail "\n You need to set security context at the pod level. .Values.artifactory.setSecurityContext is no longer supported. Replace it with .Values.artifactory.podSecurityContext" }} +{{- end }} +{{- if or .Values.artifactory.uid .Values.artifactory.gid }} +{{- if or (not (eq (.Values.artifactory.uid | toString) "1030" )) (not (eq (.Values.artifactory.gid | toString) "1030" )) }} + {{- fail "\n .Values.artifactory.uid and .Values.artifactory.gid are no longer supported. You need to set these values at the pod security context level. Replace them with .Values.artifactory.podSecurityContext.runAsUser, .Values.artifactory.podSecurityContext.runAsGroup and .Values.artifactory.podSecurityContext.fsGroup" }} +{{- end }} +{{- end }} +{{- if or .Values.artifactory.fsGroupChangePolicy .Values.artifactory.seLinuxOptions }} + {{- fail "\n .Values.artifactory.fsGroupChangePolicy and .Values.artifactory.seLinuxOptions are no longer supported. You need to set these values at the pod security context level. Replace them with .Values.artifactory.podSecurityContext.fsGroupChangePolicy and .Values.artifactory.podSecurityContext.seLinuxOptions" }} +{{- end }} +{{- if .Values.initContainerImage }} + {{- fail "\n .Values.initContainerImage is no longer supported. Replace it with .Values.initContainers.image.registry .Values.initContainers.image.repository and .Values.initContainers.image.tag" }} +{{- end }} +{{- with .Values.artifactory.statefulset.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: + serviceName: {{ template "artifactory-ha.primary.name" . }} + replicas: {{ .Values.artifactory.primary.replicaCount }} + updateStrategy: {{- toYaml .Values.artifactory.primary.updateStrategy | nindent 4}} + selector: + matchLabels: + app: {{ template "artifactory-ha.name" . }} + role: {{ template "artifactory-ha.primary.name" . }} + release: {{ .Release.Name }} + template: + metadata: + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + role: {{ template "artifactory-ha.primary.name" . }} + component: {{ .Values.artifactory.name }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + {{- with .Values.artifactory.primary.labels }} +{{ toYaml . | indent 8 }} + {{- end }} + annotations: + {{- if not .Values.artifactory.unifiedSecretInstallation }} + checksum/database-secrets: {{ include (print $.Template.BasePath "/artifactory-database-secrets.yaml") . | sha256sum }} + checksum/binarystore: {{ include (print $.Template.BasePath "/artifactory-binarystore-secret.yaml") . | sha256sum }} + checksum/systemyaml: {{ include (print $.Template.BasePath "/artifactory-system-yaml.yaml") . | sha256sum }} + {{- if .Values.access.accessConfig }} + checksum/access-config: {{ include (print $.Template.BasePath "/artifactory-access-config.yaml") . | sha256sum }} + {{- end }} + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + checksum/gcpcredentials: {{ include (print $.Template.BasePath "/artifactory-gcp-credentials-secret.yaml") . | sha256sum }} + {{- end }} + {{- if not (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) }} + checksum/admin-creds: {{ include (print $.Template.BasePath "/admin-bootstrap-creds.yaml") . | sha256sum }} + {{- end }} + {{- else }} + checksum/artifactory-unified-secret: {{ include (print $.Template.BasePath "/artifactory-unified-secret.yaml") . | sha256sum }} + {{- end }} + {{- with .Values.artifactory.annotations }} +{{ toYaml . | indent 8 }} + {{- end }} + spec: + {{- if .Values.artifactory.schedulerName }} + schedulerName: {{ .Values.artifactory.schedulerName | quote }} + {{- end }} + {{- if .Values.artifactory.priorityClass.existingPriorityClass }} + priorityClassName: {{ .Values.artifactory.priorityClass.existingPriorityClass }} + {{- else -}} + {{- if .Values.artifactory.priorityClass.create }} + priorityClassName: {{ default (include "artifactory-ha.fullname" .) .Values.artifactory.priorityClass.name }} + {{- end }} + {{- end }} + serviceAccountName: {{ template "artifactory-ha.serviceAccountName" . }} + terminationGracePeriodSeconds: {{ add .Values.artifactory.terminationGracePeriodSeconds 10 }} + {{- if or .Values.imagePullSecrets .Values.global.imagePullSecrets }} +{{- include "artifactory-ha.imagePullSecrets" . | indent 6 }} + {{- end }} + {{- if .Values.artifactory.podSecurityContext.enabled }} + securityContext: {{- omit .Values.artifactory.podSecurityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.artifactory.topologySpreadConstraints }} + topologySpreadConstraints: +{{ tpl (toYaml .Values.artifactory.topologySpreadConstraints) . | indent 8 }} + {{- end }} + initContainers: + {{- if or .Values.artifactory.customInitContainersBegin .Values.global.customInitContainersBegin }} +{{ tpl (include "artifactory-ha.customInitContainersBegin" .) . | indent 6 }} + {{- end }} + {{- if .Values.artifactory.persistence.enabled }} + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + - name: "create-artifactory-data-dir" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > + mkdir -p {{ tpl .Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir . }}; + volumeMounts: + - mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + name: volume + {{- end }} + {{- end }} + {{- if .Values.artifactory.deleteDBPropertiesOnStartup }} + - name: "delete-db-properties" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - 'rm -fv {{ .Values.artifactory.persistence.mountPath }}/etc/db.properties' + volumeMounts: + - mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + name: volume + {{- end }} + {{- if or (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) .Values.artifactory.admin.password }} + - name: "access-bootstrap-creds" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > + echo "Preparing {{ .Values.artifactory.persistence.mountPath }}/etc/access/bootstrap.creds"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access; + cp -Lrf /tmp/access/bootstrap.creds {{ .Values.artifactory.persistence.mountPath }}/etc/access/bootstrap.creds; + chmod 600 {{ .Values.artifactory.persistence.mountPath }}/etc/access/bootstrap.creds; + volumeMounts: + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + {{- if or (not .Values.artifactory.unifiedSecretInstallation) (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) }} + - name: access-bootstrap-creds + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/access/bootstrap.creds" + {{- if and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey }} + subPath: {{ .Values.artifactory.admin.dataKey }} + {{- else }} + subPath: bootstrap.creds + {{- end }} + {{- end }} + {{- end }} + - name: 'copy-system-configurations' + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - '/bin/bash' + - '-c' + - > + if [[ -e "{{ .Values.artifactory.persistence.mountPath }}/etc/filebeat.yaml" ]]; then chmod 644 {{ .Values.artifactory.persistence.mountPath }}/etc/filebeat.yaml; fi; + echo "Copy system.yaml to {{ .Values.artifactory.persistence.mountPath }}/etc"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; + {{- if .Values.systemYamlOverride.existingSecret }} + cp -fv /tmp/etc/{{ .Values.systemYamlOverride.dataKey }} {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml; + {{- else }} + cp -fv /tmp/etc/system.yaml {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml; + {{- end }} + echo "Copy binarystore.xml file"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/artifactory; + cp -fv /tmp/etc/artifactory/binarystore.xml {{ .Values.artifactory.persistence.mountPath }}/etc/artifactory/binarystore.xml; + {{- if .Values.access.accessConfig }} + echo "Copy access.config.patch.yml to {{ .Values.artifactory.persistence.mountPath }}/etc/access"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access; + cp -fv /tmp/etc/access.config.patch.yml {{ .Values.artifactory.persistence.mountPath }}/etc/access/access.config.patch.yml; + {{- end }} + {{- if .Values.access.resetAccessCAKeys }} + echo "Resetting Access CA Keys"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys; + touch {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys/reset_ca_keys; + {{- end }} + {{- if .Values.access.customCertificatesSecretName }} + echo "Copying custom certificates to {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys; + cp -fv /tmp/etc/tls.crt {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys/ca.crt; + cp -fv /tmp/etc/tls.key {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys/ca.private.key; + {{- end }} + {{- if or .Values.artifactory.joinKey .Values.global.joinKey .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName }} + echo "Copy joinKey to {{ .Values.artifactory.persistence.mountPath }}/bootstrap/access/etc/security"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/access/etc/security; + echo -n ${ARTIFACTORY_JOIN_KEY} > {{ .Values.artifactory.persistence.mountPath }}/bootstrap/access/etc/security/join.key; + {{- end }} + {{- if or .Values.artifactory.jfConnectToken .Values.artifactory.jfConnectTokenSecretName }} + echo "Copy jfConnectToken to {{ .Values.artifactory.persistence.mountPath }}/bootstrap/jfconnect/registration_token"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/jfconnect/; + echo -n ${ARTIFACTORY_JFCONNECT_TOKEN} > {{ .Values.artifactory.persistence.mountPath }}/bootstrap/jfconnect/registration_token; + {{- end }} + {{- if or .Values.artifactory.masterKey .Values.global.masterKey .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + echo "Copy masterKey to {{ .Values.artifactory.persistence.mountPath }}/etc/security"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/security; + echo -n ${ARTIFACTORY_MASTER_KEY} > {{ .Values.artifactory.persistence.mountPath }}/etc/security/master.key; + {{- end }} + env: + {{- if or .Values.artifactory.joinKey .Values.global.joinKey .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName }} + - name: ARTIFACTORY_JOIN_KEY + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName }} + name: {{ include "artifactory-ha.joinKeySecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: join-key + {{- end }} + {{- if or .Values.artifactory.jfConnectToken .Values.artifactory.jfConnectTokenSecretName }} + - name: ARTIFACTORY_JFCONNECT_TOKEN + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.jfConnectTokenSecretName }} + name: {{ include "artifactory-ha.jfConnectTokenSecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: jfconnect-token + {{- end }} + {{- if or .Values.artifactory.masterKey .Values.global.masterKey .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + - name: ARTIFACTORY_MASTER_KEY + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + name: {{ include "artifactory-ha.masterKeySecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: master-key + {{- end }} + + ######################## Volume Mounts For copy-system-configurations ########################## + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + + ######################## SystemYaml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.systemYamlOverride.existingSecret }} + - name: systemyaml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + {{- if .Values.systemYamlOverride.existingSecret }} + mountPath: "/tmp/etc/{{.Values.systemYamlOverride.dataKey}}" + subPath: {{ .Values.systemYamlOverride.dataKey }} + {{- else }} + mountPath: "/tmp/etc/system.yaml" + subPath: system.yaml + {{- end }} + + ######################## Binarystore ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Access config ########################## + {{- if .Values.access.accessConfig }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + - name: access-config + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/access.config.patch.yml" + subPath: access.config.patch.yml + {{- end }} + + ######################## Access certs external secret ########################## + {{- if .Values.access.customCertificatesSecretName }} + - name: access-certs + mountPath: "/tmp/etc/tls.crt" + subPath: tls.crt + - name: access-certs + mountPath: "/tmp/etc/tls.key" + subPath: tls.key + {{- end }} + + {{- if or .Values.artifactory.customCertificates.enabled .Values.global.customCertificates.enabled }} + - name: copy-custom-certificates + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > +{{ include "artifactory-ha.copyCustomCerts" . | indent 10 }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath }} + - name: ca-certs + mountPath: "/tmp/certs" + {{- end }} + + {{- if .Values.artifactory.circleOfTrustCertificatesSecret }} + - name: copy-circle-of-trust-certificates + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > +{{ include "artifactory.copyCircleOfTrustCertsCerts" . | indent 10 }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath }} + - name: circle-of-trust-certs + mountPath: "/tmp/circleoftrustcerts" + {{- end }} + + {{- if .Values.waitForDatabase }} + {{- if or .Values.postgresql.enabled }} + - name: "wait-for-db" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - /bin/bash + - -c + - | + echo "Waiting for postgresql to come up" + ready=false; + while ! $ready; do echo waiting; + timeout 2s bash -c " + {{- if .Values.artifactory.migration.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.artifactory.migration.preStartCommand . }}; + {{- end }} + scriptsPath="/opt/jfrog/artifactory/app/bin"; + mkdir -p $scriptsPath; + echo "Copy migration scripts and Run migration"; + cp -fv /tmp/migrate.sh $scriptsPath/migrate.sh; + cp -fv /tmp/migrationHelmInfo.yaml $scriptsPath/migrationHelmInfo.yaml; + cp -fv /tmp/migrationStatus.sh $scriptsPath/migrationStatus.sh; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/log; + bash $scriptsPath/migrationStatus.sh {{ include "artifactory-ha.app.version" . }} {{ .Values.artifactory.migration.timeoutSeconds }} > >(tee {{ .Values.artifactory.persistence.mountPath }}/log/helm-migration.log) 2>&1; + env: + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} + - name: JF_SHARED_NODE_HAENABLED + value: "true" +{{- with .Values.artifactory.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: migration-scripts + mountPath: "/tmp/migrate.sh" + subPath: migrate.sh + - name: migration-scripts + mountPath: "/tmp/migrationHelmInfo.yaml" + subPath: migrationHelmInfo.yaml + - name: migration-scripts + mountPath: "/tmp/migrationStatus.sh" + subPath: migrationStatus.sh + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + + ######################## Artifactory persistence fs ########################## + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) }} + - name: artifactory-ha-data-{{ $sharedClaimNumber }} + mountPath: "{{ tpl $.Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir $ }}/filestore{{ $sharedClaimNumber }}" + {{- end }} + - name: artifactory-ha-backup + mountPath: "{{ $.Values.artifactory.persistence.fileSystem.existingSharedClaim.backupDir }}" + {{- end }} + {{- end }} + + ######################## CustomVolumeMounts ########################## + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-ha-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-ha-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystore Xml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Artifactory persistence google storage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + {{- end }} + +{{- end }} + + {{- if .Values.hostAliases }} + hostAliases: +{{ toYaml .Values.hostAliases | indent 6 }} + {{- end }} + containers: + {{- if .Values.splitServicesToContainers }} + - name: {{ .Values.router.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "router") }} + imagePullPolicy: {{ .Values.router.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/router/app/bin/entrypoint-router.sh; + {{- with .Values.router.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_ROUTER_TOPOLOGY_LOCAL_REQUIREDSERVICETYPES + value: {{ include "artifactory-ha.router.requiredServiceTypes" . }} +{{- with .Values.router.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - name: http + containerPort: {{ .Values.router.internalPort }} + volumeMounts: + - name: volume + mountPath: {{ .Values.router.persistence.mountPath | quote }} +{{- with .Values.router.customVolumeMounts }} +{{ tpl . $ | indent 8 }} +{{- end }} + resources: +{{ toYaml .Values.router.resources | indent 10 }} + {{- if .Values.router.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.router.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.router.readinessProbe.enabled }} + readinessProbe: +{{ tpl .Values.router.readinessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.router.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.router.livenessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.frontend.enabled }} + - name: {{ .Values.frontend.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/third-party/node/bin/node /opt/jfrog/artifactory/app/frontend/bin/server/dist/bundle.js /opt/jfrog/artifactory/app/frontend + {{- with .Values.frontend.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + - name : JF_SHARED_NODE_HAENABLED + value: "true" +{{- with .Values.frontend.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.frontend.resources | indent 10 }} + {{- if .Values.frontend.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.frontend.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.frontend.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.frontend.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.evidence.enabled }} + - name: {{ .Values.evidence.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/evidence/bin/jf-evidence start + {{- with .Values.evidence.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.evidence.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.evidence.internalPort }} + name: http-evidence + - containerPort: {{ .Values.evidence.externalPort }} + name: grpc-evidence + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.evidence.resources | indent 10 }} + {{- if .Values.evidence.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.evidence.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.evidence.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.evidence.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.metadata.enabled }} + - name: {{ .Values.metadata.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "metadata") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/metadata/bin/jf-metadata start + {{- with .Values.metadata.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.metadata.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.metadata.resources | indent 10 }} + {{- if .Values.metadata.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.metadata.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.metadata.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.metadata.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.event.enabled }} + - name: {{ .Values.event.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/event/bin/jf-event start + {{- with .Values.event.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.event.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.event.resources | indent 10 }} + {{- if .Values.event.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.event.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.event.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.event.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.jfconnect.enabled }} + - name: {{ .Values.jfconnect.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/jfconnect/bin/jf-connect start + {{- with .Values.jfconnect.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.jfconnect.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.jfconnect.resources | indent 10 }} + {{- if .Values.jfconnect.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.jfconnect.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.jfconnect.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.jfconnect.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if and .Values.federation.enabled .Values.federation.embedded }} + - name: {{ .Values.federation.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/third-party/java/bin/java {{ .Values.federation.extraJavaOpts }} -jar /opt/jfrog/artifactory/app/rtfs/lib/jf-rtfs + {{- with .Values.federation.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + # TODO - Password,Url,Username - should be derived from env variable +{{- with .Values.federation.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.federation.internalPort }} + name: http-rtfs + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.federation.resources | indent 10 }} + {{- if .Values.federation.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.federation.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.federation.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.federation.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.observability.enabled }} + - name: {{ .Values.observability.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "observability") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/observability/bin/jf-observability start + {{- with .Values.observability.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.observability.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.observability.resources | indent 10 }} + {{- if .Values.observability.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.observability.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.observability.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.observability.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if and .Values.access.enabled (not (.Values.access.runOnArtifactoryTomcat | default false)) }} + - name: {{ .Values.access.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.access.resources }} + resources: +{{ toYaml .Values.access.resources | indent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + set -e; + {{- if .Values.access.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.access.preStartCommand . }}; + {{- end }} + exec /opt/jfrog/artifactory/app/access/bin/entrypoint-access.sh + {{- with .Values.access.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.access.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + {{- if .Values.artifactory.customPersistentVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.artifactory.customPersistentPodVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentPodVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentPodVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: awsmp-product-license + mountPath: "/var/run/secrets/product-license" + {{- end }} + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + + ######################## Artifactory persistence fs ########################## + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) }} + - name: artifactory-ha-data-{{ $sharedClaimNumber }} + mountPath: "{{ tpl $.Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir $ }}/filestore{{ $sharedClaimNumber }}" + {{- end }} + - name: artifactory-ha-backup + mountPath: "{{ $.Values.artifactory.persistence.fileSystem.existingSharedClaim.backupDir }}" + {{- end }} + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-ha-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-ha-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystore Xml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Artifactory persistence google storage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + + + ######################## Artifactory license ########################## + {{- if or .Values.artifactory.license.secret .Values.artifactory.license.licenseKey }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.license.secret }} + - name: artifactory-license + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/artifactory.cluster.license" + {{- if .Values.artifactory.license.secret }} + subPath: {{ .Values.artifactory.license.dataKey }} + {{- else if .Values.artifactory.license.licenseKey }} + subPath: artifactory.lic + {{- end }} + {{- end }} + {{- end }} + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + {{- if .Values.access.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.access.startupProbe.config . | indent 10 }} + {{- end }} + {{- if semverCompare " + set -e; + if [ -d /artifactory_extra_conf ] && [ -d /artifactory_bootstrap ]; then + echo "Copying bootstrap config from /artifactory_extra_conf to /artifactory_bootstrap"; + cp -Lrfv /artifactory_extra_conf/ /artifactory_bootstrap/; + fi; + {{- if .Values.artifactory.configMapName }} + echo "Copying bootstrap configs"; + cp -Lrf /bootstrap/* /artifactory_bootstrap/; + {{- end }} + {{- if .Values.artifactory.userPluginSecrets }} + echo "Copying plugins"; + cp -Lrf /tmp/plugin/*/* /artifactory_bootstrap/plugins; + {{- end }} + {{- range .Values.artifactory.copyOnEveryStartup }} + {{- $targetPath := printf "%s/%s" $.Values.artifactory.persistence.mountPath .target }} + {{- $baseDirectory := regexFind ".*/" $targetPath }} + mkdir -p {{ $baseDirectory }}; + cp -Lrf {{ .source }} {{ $.Values.artifactory.persistence.mountPath }}/{{ .target }}; + {{- end }} + {{- with .Values.artifactory.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl . $ }}; + {{- end }} + {{- with .Values.artifactory.primary.preStartCommand }} + echo "Running primary specific custom preStartCommand command"; + {{ tpl . $ }}; + {{- end }} + exec /entrypoint-artifactory.sh + {{- with .Values.artifactory.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + {{- if .Values.aws.license.enabled }} + - name: IS_AWS_LICENSE + value: "true" + - name: AWS_REGION + value: {{ .Values.aws.region | quote }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: AWS_WEB_IDENTITY_REFRESH_TOKEN_FILE + value: "/var/run/secrets/product-license/license_token" + - name: AWS_ROLE_ARN + valueFrom: + secretKeyRef: + name: {{ .Values.aws.licenseConfigSecretName }} + key: iam_role + {{- end }} + {{- end }} + {{- if .Values.splitServicesToContainers }} + - name : JF_ROUTER_ENABLED + value: "true" + - name : JF_ROUTER_SERVICE_ENABLED + value: "false" + - name : JF_EVENT_ENABLED + value: "false" + - name : JF_METADATA_ENABLED + value: "false" + - name : JF_FRONTEND_ENABLED + value: "false" + - name: JF_FEDERATION_ENABLED + value: "false" + - name : JF_OBSERVABILITY_ENABLED + value: "false" + - name : JF_JFCONNECT_SERVICE_ENABLED + value: "false" + - name : JF_EVIDENCE_ENABLED + value: "false" + {{- if not (.Values.access.runOnArtifactoryTomcat | default false) }} + - name : JF_ACCESS_ENABLED + value: "false" + {{- end}} + {{- end }} + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory-ha.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} + - name: JF_SHARED_NODE_HAENABLED + value: "true" +{{- with .Values.artifactory.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.artifactory.internalPort }} + name: http + - containerPort: {{ .Values.artifactory.internalArtifactoryPort }} + name: http-internal + - containerPort: {{ .Values.federation.internalPort }} + name: http-rtfs + {{- if .Values.artifactory.primary.javaOpts.jmx.enabled }} + - containerPort: {{ .Values.artifactory.primary.javaOpts.jmx.port }} + name: tcp-jmx + {{- end }} + {{- if .Values.artifactory.ssh.enabled }} + - containerPort: {{ .Values.artifactory.ssh.internalPort }} + name: tcp-ssh + {{- end }} + + volumeMounts: + {{- if .Values.artifactory.customPersistentVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.artifactory.customPersistentPodVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentPodVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentPodVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: awsmp-product-license + mountPath: "/var/run/secrets/product-license" + {{- end }} + {{- if .Values.artifactory.userPluginSecrets }} + - name: bootstrap-plugins + mountPath: "/artifactory_bootstrap/plugins/" + {{- range .Values.artifactory.userPluginSecrets }} + - name: {{ tpl . $ }} + mountPath: "/tmp/plugin/{{ tpl . $ }}" + {{- end }} + {{- end }} + - name: volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + + ######################## Artifactory persistence fs ########################## + {{- if eq .Values.artifactory.persistence.type "file-system" }} + {{- if .Values.artifactory.persistence.fileSystem.existingSharedClaim.enabled }} + {{- range $sharedClaimNumber, $e := until (.Values.artifactory.persistence.fileSystem.existingSharedClaim.numberOfExistingClaims|int) }} + - name: artifactory-ha-data-{{ $sharedClaimNumber }} + mountPath: "{{ tpl $.Values.artifactory.persistence.fileSystem.existingSharedClaim.dataDir $ }}/filestore{{ $sharedClaimNumber }}" + {{- end }} + - name: artifactory-ha-backup + mountPath: "{{ $.Values.artifactory.persistence.fileSystem.existingSharedClaim.backupDir }}" + {{- end }} + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-ha-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-ha-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystoreXml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Artifactory persistence googleStorage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + {{- end }} + + ######################## Artifactory configMapName ########################## + {{- if .Values.artifactory.configMapName }} + - name: bootstrap-config + mountPath: "/bootstrap/" + {{- end }} + + ######################## Artifactory license ########################## + {{- if or .Values.artifactory.license.secret .Values.artifactory.license.licenseKey }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.license.secret }} + - name: artifactory-license + {{- else }} + - name: {{ include "artifactory-ha.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/artifactory.cluster.license" + {{- if .Values.artifactory.license.secret }} + subPath: {{ .Values.artifactory.license.dataKey }} + {{- else if .Values.artifactory.license.licenseKey }} + subPath: artifactory.lic + {{- end }} + {{- end }} + + - name: installer-info + mountPath: "/artifactory_bootstrap/info/installer-info.json" + subPath: installer-info.json + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory-ha.customVolumeMounts" .) . | indent 8 }} + {{- end }} + resources: +{{ toYaml .Values.artifactory.primary.resources | indent 10 }} + {{- if .Values.artifactory.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.artifactory.startupProbe.config . | indent 10 }} + {{- end }} + {{- if and (not .Values.splitServicesToContainers) (semverCompare "=1.18.0-0" .Capabilities.KubeVersion.GitVersion) }} + ingressClassName: {{ .Values.ingress.className }} + {{- end }} + {{- if .Values.ingress.defaultBackend.enabled }} + {{- if .Capabilities.APIVersions.Has "networking.k8s.io/v1" }} + defaultBackend: + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + backend: + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end }} + rules: +{{- if .Values.ingress.hosts }} + {{- if .Capabilities.APIVersions.Has "networking.k8s.io/v1" }} + {{- range $host := .Values.ingress.hosts }} + - host: {{ $host | quote }} + http: + paths: + - path: {{ $.Values.ingress.routerPath }} + pathType: ImplementationSpecific + backend: + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- if not $.Values.ingress.disableRouterBypass }} + - path: {{ $.Values.ingress.artifactoryPath }} + pathType: ImplementationSpecific + backend: + service: + name: {{ $serviceName }} + port: + number: {{ $artifactoryServicePort }} + {{- end }} + {{- if and $.Values.federation.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" $.Values.artifactory.image.repository)) }} + - path: {{ $.Values.ingress.rtfsPath }} + pathType: ImplementationSpecific + backend: + service: + name: {{ $serviceName }} + port: + number: {{ $.Values.federation.internalPort }} + {{- end }} + {{- end }} + {{- else }} + {{- range $host := .Values.ingress.hosts }} + - host: {{ $host | quote }} + http: + paths: + - path: {{ $.Values.ingress.routerPath }} + backend: + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + - path: {{ $.Values.ingress.artifactoryPath }} + backend: + serviceName: {{ $serviceName }} + servicePort: {{ $artifactoryServicePort }} + {{- end }} + {{- end }} +{{- end -}} + {{- with .Values.ingress.additionalRules }} +{{ tpl . $ | indent 2 }} + {{- end }} + {{- if .Values.ingress.tls }} + tls: +{{ toYaml .Values.ingress.tls | indent 4 }} + {{- end -}} + +{{- if .Values.customIngress }} +--- +{{ .Values.customIngress | toYaml | trimSuffix "\n" }} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/logger-configmap.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/logger-configmap.yaml new file mode 100644 index 0000000000..d3597905d9 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/logger-configmap.yaml @@ -0,0 +1,63 @@ +{{- if or .Values.artifactory.loggers .Values.artifactory.catalinaLoggers }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory-ha.fullname" . }}-logger + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + tail-log.sh: | + #!/bin/sh + + LOG_DIR=$1 + LOG_NAME=$2 + PID= + + # Wait for log dir to appear + while [ ! -d ${LOG_DIR} ]; do + sleep 1 + done + + cd ${LOG_DIR} + + LOG_PREFIX=$(echo ${LOG_NAME} | sed 's/.log$//g') + + # Find the log to tail + LOG_FILE=$(ls -1t ./${LOG_PREFIX}.log 2>/dev/null) + + # Wait for the log file + while [ -z "${LOG_FILE}" ]; do + sleep 1 + LOG_FILE=$(ls -1t ./${LOG_PREFIX}.log 2>/dev/null) + done + + echo "Log file ${LOG_FILE} is ready!" + + # Get inode number + INODE_ID=$(ls -i ${LOG_FILE}) + + # echo "Tailing ${LOG_FILE}" + tail -F ${LOG_FILE} & + PID=$! + + # Loop forever to see if a new log was created + while true; do + # Check inode number + NEW_INODE_ID=$(ls -i ${LOG_FILE}) + + # If inode number changed, this means log was rotated and need to start a new tail + if [ "${INODE_ID}" != "${NEW_INODE_ID}" ]; then + kill -9 ${PID} 2>/dev/null + INODE_ID="${NEW_INODE_ID}" + + # Start a new tail + tail -F ${LOG_FILE} & + PID=$! + fi + sleep 1 + done + +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-artifactory-conf.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-artifactory-conf.yaml new file mode 100644 index 0000000000..97ae5f27be --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-artifactory-conf.yaml @@ -0,0 +1,18 @@ +{{- if and (not .Values.nginx.customArtifactoryConfigMap) .Values.nginx.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory-ha.fullname" . }}-nginx-artifactory-conf + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + artifactory.conf: | +{{- if .Values.nginx.artifactoryConf }} +{{ tpl .Values.nginx.artifactoryConf . | indent 4 }} +{{- else }} +{{ tpl ( .Files.Get "files/nginx-artifactory-conf.yaml" ) . | indent 4 }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-certificate-secret.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-certificate-secret.yaml new file mode 100644 index 0000000000..29c77ad5a4 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-certificate-secret.yaml @@ -0,0 +1,14 @@ +{{- if and (not .Values.nginx.tlsSecretName) .Values.nginx.enabled .Values.nginx.https.enabled }} +apiVersion: v1 +kind: Secret +type: kubernetes.io/tls +metadata: + name: {{ template "artifactory-ha.fullname" . }}-nginx-certificate + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: +{{ ( include "artifactory-ha.gen-certs" . ) | indent 2 }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-conf.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-conf.yaml new file mode 100644 index 0000000000..4f0d65f25c --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-conf.yaml @@ -0,0 +1,18 @@ +{{- if and (not .Values.nginx.customConfigMap) .Values.nginx.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory-ha.fullname" . }}-nginx-conf + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + nginx.conf: | +{{- if .Values.nginx.mainConf }} +{{ tpl .Values.nginx.mainConf . | indent 4 }} +{{- else }} +{{ tpl ( .Files.Get "files/nginx-main-conf.yaml" ) . | indent 4 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-deployment.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-deployment.yaml new file mode 100644 index 0000000000..d43689b8cd --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-deployment.yaml @@ -0,0 +1,221 @@ +{{- if .Values.nginx.enabled -}} +{{- $serviceName := include "artifactory-ha.fullname" . -}} +{{- $servicePort := .Values.artifactory.externalPort -}} +apiVersion: apps/v1 +kind: {{ .Values.nginx.kind }} +metadata: + name: {{ template "artifactory-ha.nginx.fullname" . }} + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + component: {{ .Values.nginx.name }} +{{- if .Values.nginx.labels }} +{{ toYaml .Values.nginx.labels | indent 4 }} +{{- end }} +{{- with .Values.nginx.deployment.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: +{{- if ne .Values.nginx.kind "DaemonSet" }} + replicas: {{ .Values.nginx.replicaCount }} +{{- end }} + selector: + matchLabels: + app: {{ template "artifactory-ha.name" . }} + release: {{ .Release.Name }} + component: {{ .Values.nginx.name }} + template: + metadata: + annotations: + checksum/nginx-conf: {{ include (print $.Template.BasePath "/nginx-conf.yaml") . | sha256sum }} + checksum/nginx-artifactory-conf: {{ include (print $.Template.BasePath "/nginx-artifactory-conf.yaml") . | sha256sum }} + {{- range $key, $value := .Values.nginx.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + labels: + app: {{ template "artifactory-ha.name" . }} + chart: {{ template "artifactory-ha.chart" . }} + component: {{ .Values.nginx.name }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +{{- if .Values.nginx.labels }} +{{ toYaml .Values.nginx.labels | indent 8 }} +{{- end }} + spec: + {{- if .Values.nginx.podSecurityContext.enabled }} + securityContext: {{- omit .Values.nginx.podSecurityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + serviceAccountName: {{ template "artifactory-ha.serviceAccountName" . }} + terminationGracePeriodSeconds: {{ .Values.nginx.terminationGracePeriodSeconds }} + {{- if or .Values.imagePullSecrets .Values.global.imagePullSecrets }} +{{- include "artifactory-ha.imagePullSecrets" . | indent 6 }} + {{- end }} + {{- if .Values.nginx.priorityClassName }} + priorityClassName: {{ .Values.nginx.priorityClassName | quote }} + {{- end }} + {{- if .Values.nginx.topologySpreadConstraints }} + topologySpreadConstraints: +{{ tpl (toYaml .Values.nginx.topologySpreadConstraints) . | indent 8 }} + {{- end }} + initContainers: + {{- if .Values.nginx.customInitContainers }} +{{ tpl (include "artifactory.nginx.customInitContainers" .) . | indent 6 }} + {{- end }} + - name: "setup" + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.imagePullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/sh' + - '-c' + - > + rm -rfv {{ .Values.nginx.persistence.mountPath }}/lost+found; + mkdir -p {{ .Values.nginx.persistence.mountPath }}/logs; + resources: + {{- toYaml .Values.initContainers.resources | nindent 10 }} + volumeMounts: + - mountPath: {{ .Values.nginx.persistence.mountPath | quote }} + name: nginx-volume + containers: + - name: {{ .Values.nginx.name }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list . "nginx") }} + imagePullPolicy: {{ .Values.nginx.image.pullPolicy }} + {{- if .Values.nginx.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.nginx.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.nginx.customCommand }} + command: +{{- tpl (include "nginx.command" .) . | indent 10 }} + {{- end }} + ports: +{{ if .Values.nginx.customPorts }} +{{ toYaml .Values.nginx.customPorts | indent 8 }} +{{ end }} + # DEPRECATION NOTE: The following is to maintain support for values pre 1.3.1 and + # will be cleaned up in a later version + {{- if .Values.nginx.http }} + {{- if .Values.nginx.http.enabled }} + - containerPort: {{ .Values.nginx.http.internalPort }} + name: http + {{- end }} + {{- else }} # DEPRECATED + - containerPort: {{ .Values.nginx.internalPortHttp }} + name: http-internal + {{- end }} + {{- if .Values.nginx.https }} + {{- if .Values.nginx.https.enabled }} + - containerPort: {{ .Values.nginx.https.internalPort }} + name: https + {{- end }} + {{- else }} # DEPRECATED + - containerPort: {{ .Values.nginx.internalPortHttps }} + name: https-internal + {{- end }} + {{- if .Values.artifactory.ssh.enabled }} + - containerPort: {{ .Values.nginx.ssh.internalPort }} + name: tcp-ssh + {{- end }} + {{- with .Values.nginx.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + volumeMounts: + - name: nginx-conf + mountPath: /etc/nginx/nginx.conf + subPath: nginx.conf + - name: nginx-artifactory-conf + mountPath: "{{ .Values.nginx.persistence.mountPath }}/conf.d/" + - name: nginx-volume + mountPath: {{ .Values.nginx.persistence.mountPath | quote }} + {{- if .Values.nginx.https.enabled }} + - name: ssl-certificates + mountPath: "{{ .Values.nginx.persistence.mountPath }}/ssl" + {{- end }} + {{- if .Values.nginx.customVolumeMounts }} +{{ tpl (include "artifactory.nginx.customVolumeMounts" .) . | indent 8 }} + {{- end }} + resources: +{{ toYaml .Values.nginx.resources | indent 10 }} + {{- if .Values.nginx.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.nginx.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.nginx.readinessProbe.enabled }} + readinessProbe: +{{ tpl .Values.nginx.readinessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.nginx.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.nginx.livenessProbe.config . | indent 10 }} + {{- end }} + {{- $mountPath := .Values.nginx.persistence.mountPath }} + {{- range .Values.nginx.loggers }} + - name: {{ . | replace "_" "-" | replace "." "-" }} + image: {{ include "artifactory-ha.getImageInfoByValue" (list $ "initContainers") }} + imagePullPolicy: {{ $.Values.initContainers.image.pullPolicy }} + command: + - tail + args: + - '-F' + - '{{ $mountPath }}/logs/{{ . }}' + volumeMounts: + - name: nginx-volume + mountPath: {{ $mountPath }} + resources: +{{ toYaml $.Values.nginx.loggersResources | indent 10 }} + {{- end }} + {{- if .Values.nginx.customSidecarContainers }} +{{ tpl (include "artifactory.nginx.customSidecarContainers" .) . | indent 6 }} + {{- end }} + {{- if or .Values.nginx.nodeSelector .Values.global.nodeSelector }} +{{ tpl (include "nginx.nodeSelector" .) . | indent 6 }} + {{- end }} + {{- with .Values.nginx.affinity }} + affinity: +{{ toYaml . | indent 8 }} + {{- end }} + {{- with .Values.nginx.tolerations }} + tolerations: +{{ toYaml . | indent 8 }} + {{- end }} + volumes: + {{- if .Values.nginx.customVolumes }} +{{ tpl (include "artifactory.nginx.customVolumes" .) . | indent 6 }} + {{- end }} + - name: nginx-conf + configMap: + {{- if .Values.nginx.customConfigMap }} + name: {{ .Values.nginx.customConfigMap }} + {{- else }} + name: {{ template "artifactory-ha.fullname" . }}-nginx-conf + {{- end }} + - name: nginx-artifactory-conf + configMap: + {{- if .Values.nginx.customArtifactoryConfigMap }} + name: {{ .Values.nginx.customArtifactoryConfigMap }} + {{- else }} + name: {{ template "artifactory-ha.fullname" . }}-nginx-artifactory-conf + {{- end }} + + - name: nginx-volume + {{- if .Values.nginx.persistence.enabled }} + persistentVolumeClaim: + claimName: {{ .Values.nginx.persistence.existingClaim | default (include "artifactory-ha.nginx.fullname" .) }} + {{- else }} + emptyDir: {} + {{- end }} + {{- if .Values.nginx.https.enabled }} + - name: ssl-certificates + secret: + {{- if .Values.nginx.tlsSecretName }} + secretName: {{ .Values.nginx.tlsSecretName }} + {{- else }} + secretName: {{ template "artifactory-ha.fullname" . }}-nginx-certificate + {{- end }} + {{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-pdb.yaml b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-pdb.yaml new file mode 100644 index 0000000000..0aed993682 --- /dev/null +++ b/charts/jfrog/artifactory-ha/107.90.14/templates/nginx-pdb.yaml @@ -0,0 +1,23 @@ +{{- if .Values.nginx.enabled -}} +{{- if semverCompare " --from-literal=license_token=${TOKEN} --from-literal=iam_role=${ROLE_ARN}` +aws: + license: + enabled: false + licenseConfigSecretName: + region: us-east-1 +## Container Security Context +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container +## @param containerSecurityContext.enabled Enabled containers' Security Context +## @param containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot +## @param containerSecurityContext.privileged Set container's Security Context privileged +## @param containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation +## @param containerSecurityContext.capabilities.drop List of capabilities to be dropped +## @param containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile +## +containerSecurityContext: + enabled: true + runAsNonRoot: true + privileged: false + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + capabilities: + drop: + - ALL +## The following router settings are to configure only when splitServicesToContainers set to true +router: + name: router + image: + registry: releases-docker.jfrog.io + repository: jfrog/router + tag: 7.118.3 + pullPolicy: IfNotPresent + serviceRegistry: + ## Service registry (Access) TLS verification skipped if enabled + insecure: false + internalPort: 8082 + externalPort: 8082 + tlsEnabled: false + ## Extra environment variables that can be used to tune router to your needs. + ## Uncomment and set value as needed + extraEnvironmentVariables: + # - name: MY_ENV_VAR + # value: "" + resources: {} + # requests: + # memory: "100Mi" + # cpu: "100m" + # limits: + # memory: "1Gi" + # cpu: "1" + + ## Add lifecycle hooks for router container + lifecycle: + ## From Artifactory versions 7.52.x, Wait for Artifactory to complete any open uploads or downloads before terminating + preStop: + exec: + command: ["sh", "-c", "while [[ $(curl --fail --silent --connect-timeout 2 http://localhost:8081/artifactory/api/v1/system/liveness) =~ OK ]]; do echo Artifactory is still alive; sleep 2; done"] + # postStart: + # exec: + # command: ["/bin/sh", "-c", "echo Hello from the postStart handler"] + ## Add custom volumesMounts + customVolumeMounts: | + # - name: custom-script + # mountPath: /scripts/script.sh + # subPath: script.sh + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl -s -k --fail --max-time {{ .Values.probes.timeoutSeconds }} {{ include "artifactory-ha.scheme" . }}://localhost:{{ .Values.router.internalPort }}/router/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " prepended. + unifiedSecretPrependReleaseName: true + image: + registry: releases-docker.jfrog.io + repository: jfrog/artifactory-pro + # tag: + pullPolicy: IfNotPresent + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + schedulerName: + ## Create a priority class for the Artifactory pods or use an existing one + ## NOTE - Maximum allowed value of a user defined priority is 1000000000 + priorityClass: + create: false + value: 1000000000 + ## Override default name + # name: + ## Use an existing priority class + # existingPriorityClass: + ## Delete the db.properties file in ARTIFACTORY_HOME/etc/db.properties + deleteDBPropertiesOnStartup: true + database: + maxOpenConnections: 80 + tomcat: + maintenanceConnector: + port: 8091 + connector: + maxThreads: 200 + sendReasonPhrase: false + extraConfig: 'acceptCount="400"' + ## certificates added to this secret will be copied to $JFROG_HOME/artifactory/var/etc/security/keys/trusted directory + customCertificates: + enabled: false + # certificateSecretName: + ## Support for metrics is only available for Artifactory 7.7.x (appVersions) and above. + ## To enable set `.Values.artifactory.metrics.enabled` to `true` + ## Note: Depricated `openMetrics` as part of 7.87.x and renamed to `metrics` + ## Refer - https://www.jfrog.com/confluence/display/JFROG/Open+Metrics + metrics: + enabled: false + ## Settings for pushing metrics to Insight - enable filebeat to true + filebeat: + enabled: false + log: + enabled: false + ## Log level for filebeat. Possible values: debug, info, warning, or error. + level: "info" + ## Elasticsearch details for filebeat to connect + elasticsearch: + url: "Elasticsearch url where JFrog Insight is installed For example, http://:8082" + username: "" + password: "" + ## Support for Cold Artifact Storage + ## set 'coldStorage.enabled' to 'true' only for Artifactory instance that you are designating as the Cold instance + ## Refer - https://jfrog.com/help/r/jfrog-platform-administration-documentation/setting-up-cold-artifact-storage + coldStorage: + enabled: false + ## This directory is intended for use with NFS eventual configuration for HA + ## When enabling this section, The system.yaml will include haDataDir section. + ## The location of Artifactory Data directory and Artifactory Filestore will be modified accordingly and will be shared among all nodes. + ## It's recommended to leave haDataDir disabled, and the default BinarystoreXml will set the Filestore location as configured in artifactory.persistence.nfs.dataDir. + haDataDir: + enabled: false + path: + haBackupDir: + enabled: false + path: + ## Files to copy to ARTIFACTORY_HOME/ on each Artifactory startup + ## Note : From 107.46.x chart versions, copyOnEveryStartup is not needed for binarystore.xml, it is always copied via initContainers + copyOnEveryStartup: + ## Absolute path + # - source: /artifactory_bootstrap/artifactory.cluster.license + ## Relative to ARTIFACTORY_HOME/ + # target: etc/artifactory/ + + ## Sidecar containers for tailing Artifactory logs + loggers: [] + # - access-audit.log + # - access-request.log + # - access-security-audit.log + # - access-service.log + # - artifactory-access.log + # - artifactory-event.log + # - artifactory-import-export.log + # - artifactory-request.log + # - artifactory-service.log + # - frontend-request.log + # - frontend-service.log + # - metadata-request.log + # - metadata-service.log + # - router-request.log + # - router-service.log + # - router-traefik.log + # - derby.log + + ## Loggers containers resources + loggersResources: {} + # requests: + # memory: "10Mi" + # cpu: "10m" + # limits: + # memory: "100Mi" + # cpu: "50m" + + ## Sidecar containers for tailing Tomcat (catalina) logs + catalinaLoggers: [] + # - tomcat-catalina.log + # - tomcat-localhost.log + + ## Tomcat (catalina) loggers resources + catalinaLoggersResources: {} + # requests: + # memory: "10Mi" + # cpu: "10m" + # limits: + # memory: "100Mi" + # cpu: "50m" + + ## Migration support from 6.x to 7.x. + migration: + enabled: false + timeoutSeconds: 3600 + ## Extra pre-start command in migration Init Container to install JDBC driver for MySql/MariaDb/Oracle + # preStartCommand: "mkdir -p /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib; cd /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib && curl -o /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib/mysql-connector-java-5.1.41.jar https://repo1.maven.org/maven2/mysql/mysql-connector-java/5.1.41/mysql-connector-java-5.1.41.jar" + ## Add custom init containers execution before predefined init containers + customInitContainersBegin: | + # - name: "custom-setup" + # image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + # imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + # securityContext: + # runAsNonRoot: true + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - NET_RAW + # command: + # - 'sh' + # - '-c' + # - 'touch {{ .Values.artifactory.persistence.mountPath }}/example-custom-setup' + # volumeMounts: + # - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + # name: volume + ## Add custom init containers + ## Add custom init containers execution after predefined init containers + customInitContainers: | + # - name: "custom-systemyaml-setup" + # image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + # imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + # securityContext: + # runAsNonRoot: true + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - NET_RAW + # command: + # - 'sh' + # - '-c' + # - 'curl -o {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml https:///systemyaml' + # volumeMounts: + # - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + # name: volume + ## Add custom sidecar containers + ## - The provided example uses a custom volume (customVolumes) + ## - The provided example shows running container as root (id 0) + customSidecarContainers: | + # - name: "sidecar-list-etc" + # image: {{ include "artifactory-ha.getImageInfoByValue" (list . "initContainers") }} + # imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + # securityContext: + # runAsNonRoot: true + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - NET_RAW + # command: + # - 'sh' + # - '-c' + # - 'sh /scripts/script.sh' + # volumeMounts: + # - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + # name: volume + # - mountPath: "/scripts/script.sh" + # name: custom-script + # subPath: script.sh + # resources: + # requests: + # memory: "32Mi" + # cpu: "50m" + # limits: + # memory: "128Mi" + # cpu: "100m" + ## Add custom volumes + ## If .Values.artifactory.unifiedSecretInstallation is true then secret name should be '{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret'. + customVolumes: | + # - name: custom-script + # configMap: + # name: custom-script + ## Add custom volumesMounts + customVolumeMounts: | + # - name: custom-script + # mountPath: "/scripts/script.sh" + # subPath: script.sh + # - name: posthook-start + # mountPath: "/scripts/posthoook-start.sh" + # subPath: posthoook-start.sh + # - name: prehook-start + # mountPath: "/scripts/prehook-start.sh" + # subPath: prehook-start.sh + ## Add custom persistent volume mounts - Available to the entire namespace + customPersistentVolumeClaim: {} + # name: + # mountPath: + # accessModes: + # - "-" + # size: + # storageClassName: + + ## Artifactory HA requires a unique master key. Each Artifactory node must have the same master key! + ## You can generate one with the command: "openssl rand -hex 32" + ## Pass it to helm with '--set artifactory.masterKey=${MASTER_KEY}' + ## Alternatively, you can use a pre-existing secret with a key called master-key by specifying masterKeySecretName + ## IMPORTANT: You should NOT use the example masterKey for a production deployment! + ## IMPORTANT: This is a mandatory for fresh Install of 7.x (App version) + # masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + # masterKeySecretName: + + ## Join Key to connect to other services to Artifactory. + ## IMPORTANT: Setting this value overrides the existing joinKey + ## IMPORTANT: You should NOT use the example joinKey for a production deployment! + # joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + ## Alternatively, you can use a pre-existing secret with a key called join-key by specifying joinKeySecretName + # joinKeySecretName: + + ## Registration Token for JFConnect + # jfConnectToken: + ## Alternatively, you can use a pre-existing secret with a key called jfconnect-token by specifying jfConnectTokenSecretName + # jfConnectTokenSecretName: + + ## Add custom secrets - secret per file + ## If .Values.artifactory.unifiedSecretInstallation is true then secret name should be '{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret' common to all secrets + customSecrets: + # - name: custom-secret + # key: custom-secret.yaml + # data: > + # custom_secret_config: + # parameter1: value1 + # parameter2: value2 + # - name: custom-secret2 + # key: custom-secret2.config + # data: | + # here the custom secret 2 config + + ## If false, all service console logs will not redirect to a common console.log + consoleLog: false + ## admin allows to set the password for the default admin user. + ## See: https://www.jfrog.com/confluence/display/JFROG/Users+and+Groups#UsersandGroups-RecreatingtheDefaultAdminUserrecreate + admin: + ip: "127.0.0.1" + username: "admin" + password: + secret: + dataKey: + ## Artifactory license. + license: + ## licenseKey is the license key in plain text. Use either this or the license.secret setting + licenseKey: + ## If artifactory.license.secret is passed, it will be mounted as + ## ARTIFACTORY_HOME/etc/artifactory.cluster.license and loaded at run time. + secret: + ## The dataKey should be the name of the secret data key created. + dataKey: + ## Create configMap with artifactory.config.import.xml and security.import.xml and pass name of configMap in following parameter + configMapName: + ## Add any list of configmaps to Artifactory + configMaps: | + # posthook-start.sh: |- + # echo "This is a post start script" + # posthook-end.sh: |- + # echo "This is a post end script" + ## List of secrets for Artifactory user plugins. + ## One Secret per plugin's files. + userPluginSecrets: + # - archive-old-artifacts + # - build-cleanup + # - webhook + # - '{{ template "my-chart.fullname" . }}' + + ## Extra pre-start command to install JDBC driver for MySql/MariaDb/Oracle + # preStartCommand: "mkdir -p /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib; cd /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib && curl -o /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib/mysql-connector-java-5.1.41.jar https://repo1.maven.org/maven2/mysql/mysql-connector-java/5.1.41/mysql-connector-java-5.1.41.jar" + + ## Add lifecycle hooks for artifactory container + lifecycle: {} + # postStart: + # exec: + # command: ["/bin/sh", "-c", "echo Hello from the postStart handler"] + # preStop: + # exec: + # command: ["/bin/sh","-c","echo Hello from the preStop handler"] + + ## Extra environment variables that can be used to tune Artifactory to your needs. + ## Uncomment and set value as needed + extraEnvironmentVariables: + # - name: SERVER_XML_ARTIFACTORY_PORT + # value: "8081" + # - name: SERVER_XML_ARTIFACTORY_MAX_THREADS + # value: "200" + # - name: SERVER_XML_ACCESS_MAX_THREADS + # value: "50" + # - name: SERVER_XML_ARTIFACTORY_EXTRA_CONFIG + # value: "" + # - name: SERVER_XML_ACCESS_EXTRA_CONFIG + # value: "" + # - name: SERVER_XML_EXTRA_CONNECTOR + # value: "" + # - name: DB_POOL_MAX_ACTIVE + # value: "100" + # - name: DB_POOL_MAX_IDLE + # value: "10" + # - name: MY_SECRET_ENV_VAR + # valueFrom: + # secretKeyRef: + # name: my-secret-name + # key: my-secret-key + + ## System YAML entries now reside under files/system.yaml. + ## You can provide the specific values that you want to add or override under 'artifactory.extraSystemYaml'. + ## For example: + ## extraSystemYaml: + ## shared: + ## node: + ## id: my-instance + ## The entries provided under 'artifactory.extraSystemYaml' are merged with files/system.yaml to create the final system.yaml. + ## If you have already provided system.yaml under, 'artifactory.systemYaml', the values in that entry take precedence over files/system.yaml + ## You can modify specific entries with your own value under `artifactory.extraSystemYaml`, The values under extraSystemYaml overrides the values under 'artifactory.systemYaml' and files/system.yaml + extraSystemYaml: {} + ## systemYaml is intentionally commented and the previous content has been moved under files/system.yaml. + ## You have to add the all entries of the system.yaml file here, and it overrides the values in files/system.yaml. + # systemYaml: + + ## IMPORTANT: If overriding artifactory.internalPort: + ## DO NOT use port lower than 1024 as Artifactory runs as non-root and cannot bind to ports lower than 1024! + externalPort: 8082 + internalPort: 8082 + externalArtifactoryPort: 8081 + internalArtifactoryPort: 8081 + terminationGracePeriodSeconds: 30 + ## Pod Security Context + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ + ## @param artifactory.podSecurityContext.enabled Enable security context + ## @param artifactory.podSecurityContext.runAsNonRoot Set pod's Security Context runAsNonRoot + ## @param artifactory.podSecurityContext.runAsUser User ID for the pod + ## @param artifactory.podSecurityContext.runASGroup Group ID for the pod + ## @param artifactory.podSecurityContext.fsGroup Group ID for the pod + ## + podSecurityContext: + enabled: true + runAsNonRoot: true + runAsUser: 1030 + runAsGroup: 1030 + fsGroup: 1030 + # fsGroupChangePolicy: "Always" + # seLinuxOptions: {} + ## The following settings are to configure the frequency of the liveness and startup probes. + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl -s -k --fail --max-time {{ .Values.probes.timeoutSeconds }} http://localhost:{{ .Values.artifactory.tomcat.maintenanceConnector.port }}/artifactory/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " + ## 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) + ## + # storageClassName: "-" + + ## Set the persistence storage type. This will apply the matching binarystore.xml to Artifactory config + ## Supported types are: + ## file-system (default) + ## nfs + ## google-storage + ## google-storage-v2 + ## google-storage-v2-direct (Recommended for GCS - Google Cloud Storage) + ## aws-s3-v3 + ## s3-storage-v3-direct (Recommended for AWS S3) + ## s3-storage-v3-archive + ## azure-blob + ## azure-blob-storage-direct + ## azure-blob-storage-v2-direct (Recommended for Azure Blob Storage) + type: file-system + ## Use binarystoreXml to provide a custom binarystore.xml + ## This is intentionally commented and below previous content of binarystoreXml is moved under files/binarystore.xml + ## binarystoreXml: + + ## For artifactory.persistence.type file-system + fileSystem: + ## Need to have the following set + existingSharedClaim: + enabled: false + numberOfExistingClaims: 1 + ## Should be a child directory of {{ .Values.artifactory.persistence.mountPath }} + dataDir: "{{ .Values.artifactory.persistence.mountPath }}/artifactory-data" + backupDir: "/var/opt/jfrog/artifactory-backup" + ## You may also use existing shared claims for the data and backup storage. This allows storage (NAS for example) to be used for Data and Backup dirs which are safe to share across multiple artifactory nodes. + ## You may specify numberOfExistingClaims to indicate how many of these existing shared claims to mount. (Default = 1) + ## Create PVCs with ReadWriteMany that match the naming convetions: + ## {{ template "artifactory-ha.fullname" . }}-data-pvc- + ## {{ template "artifactory-ha.fullname" . }}-backup-pvc + ## Example (using numberOfExistingClaims: 2) + ## myexample-data-pvc-0 + ## myexample-data-pvc-1 + ## myexample-backup-pvc + ## Note: While you need two PVC fronting two PVs, multiple PVs can be attached to the same storage in many cases allowing you to share an underlying drive. + ## For artifactory.persistence.type nfs + ## If using NFS as the shared storage, you must have a running NFS server that is accessible by your Kubernetes + ## cluster nodes. + ## Need to have the following set + nfs: + ## Must pass actual IP of NFS server with '--set For artifactory.persistence.nfs.ip=${NFS_IP}' + ip: + haDataMount: "/data" + haBackupMount: "/backup" + dataDir: "/var/opt/jfrog/artifactory-ha" + backupDir: "/var/opt/jfrog/artifactory-backup" + capacity: 200Gi + mountOptions: [] + ## For artifactory.persistence.type google-storage, google-storage-v2, google-storage-v2-direct + googleStorage: + ## When using GCP buckets as your binary store (Available with enterprise license only) + gcpServiceAccount: + enabled: false + ## Use either an existing secret prepared in advance or put the config (replace the content) in the values + ## ref: https://github.com/jfrog/charts/blob/master/stable/artifactory-ha/README.md#google-storage + # customSecretName: + # config: | + # { + # "type": "service_account", + # "project_id": "", + # "private_key_id": "?????", + # "private_key": "-----BEGIN PRIVATE KEY-----\n????????==\n-----END PRIVATE KEY-----\n", + # "client_email": "???@j.iam.gserviceaccount.com", + # "client_id": "???????", + # "auth_uri": "https://accounts.google.com/o/oauth2/auth", + # "token_uri": "https://oauth2.googleapis.com/token", + # "auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs", + # "client_x509_cert_url": "https://www.googleapis.com/robot/v1....." + # } + endpoint: commondatastorage.googleapis.com + httpsOnly: false + ## Set a unique bucket name + bucketName: "artifactory-ha-gcp" + ## GCP Bucket Authentication with Identity and Credential is deprecated. + ## identity: + ## credential: + path: "artifactory-ha/filestore" + bucketExists: false + useInstanceCredentials: false + enableSignedUrlRedirect: false + ## For artifactory.persistence.type aws-s3-v3, s3-storage-v3-direct, s3-storage-v3-archive + awsS3V3: + testConnection: false + identity: + credential: + region: + bucketName: artifactory-aws + path: artifactory/filestore + endpoint: + port: + useHttp: + maxConnections: 50 + connectionTimeout: + socketTimeout: + kmsServerSideEncryptionKeyId: + kmsKeyRegion: + kmsCryptoMode: + useInstanceCredentials: true + usePresigning: false + signatureExpirySeconds: 300 + signedUrlExpirySeconds: 30 + cloudFrontDomainName: + cloudFrontKeyPairId: + cloudFrontPrivateKey: + enableSignedUrlRedirect: false + enablePathStyleAccess: false + multiPartLimit: + multipartElementSize: + ## For artifactory.persistence.type azure-blob, azure-blob-storage-direct, azure-blob-storage-v2-direct + azureBlob: + accountName: + accountKey: + endpoint: + containerName: + multiPartLimit: 100000000 + multipartElementSize: 50000000 + testConnection: false + service: + name: artifactory + type: ClusterIP + ## @param service.ipFamilyPolicy Controller Service ipFamilyPolicy (optional, cloud specific) + ## This can be either SingleStack, PreferDualStack or RequireDualStack + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilyPolicy: "" + ## @param service.ipFamilies Controller Service ipFamilies (optional, cloud specific) + ## This can be either ["IPv4"], ["IPv6"], ["IPv4", "IPv6"] or ["IPv6", "IPv4"] + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilies: [] + ## For supporting whitelist on the Artifactory service (useful if setting service.type=LoadBalancer) + ## Set this to a list of IP CIDR ranges + ## Example: loadBalancerSourceRanges: ['10.10.10.5/32', '10.11.10.5/32'] + ## or pass from helm command line + ## Example: helm install ... --set nginx.service.loadBalancerSourceRanges='{10.10.10.5/32,10.11.10.5/32}' + loadBalancerSourceRanges: [] + annotations: {} + ## Which nodes in the cluster should be in the external load balancer pool (have external traffic routed to them) + ## Supported pool values + ## members + ## all + pool: members + ## If the type is NodePort you can set a fixed port + # nodePort: 32082 + statefulset: + annotations: {} + ssh: + enabled: false + internalPort: 1339 + externalPort: 1339 + annotations: {} + ## Spread Artifactory pods evenly across your nodes or some other topology + ## Note this applies to both the primary and replicas + topologySpreadConstraints: [] + # - maxSkew: 1 + # topologyKey: kubernetes.io/hostname + # whenUnsatisfiable: DoNotSchedule + # labelSelector: + # matchLabels: + # app: '{{ template "artifactory-ha.name" . }}' + # role: '{{ template "artifactory-ha.name" . }}' + # release: "{{ .Release.Name }}" + + ## Type specific configurations. + ## There is a difference between the primary and the member nodes. + ## Customising their resources and java parameters is done here. + primary: + name: artifactory-ha-primary + ## preStartCommand specific to the primary node, to be run after artifactory.preStartCommand + # preStartCommand: + labels: {} + persistence: + ## Set existingClaim to true or false + ## If true, you must prepare a PVC with the name e.g `volume-myrelease-artifactory-ha-primary-0` + existingClaim: false + replicaCount: 3 + # minAvailable: 1 + + updateStrategy: + type: RollingUpdate + ## Resources for the primary node + resources: {} + # requests: + # memory: "1Gi" + # cpu: "500m" + # limits: + # memory: "2Gi" + # cpu: "1" + ## The following Java options are passed to the java process running Artifactory primary node. + ## You should set them according to the resources set above + javaOpts: + # xms: "1g" + # xmx: "2g" + # corePoolSize: 24 + jmx: + enabled: false + port: 9010 + host: + ssl: false + # When authenticate is true, accessFile and passwordFile are required + authenticate: false + accessFile: + passwordFile: + # other: "" + nodeSelector: {} + tolerations: [] + affinity: {} + ## Only used if "affinity" is empty + podAntiAffinity: + ## Valid values are "soft" or "hard"; any other value indicates no anti-affinity + type: "soft" + topologyKey: "kubernetes.io/hostname" + node: + name: artifactory-ha-member + ## preStartCommand specific to the member node, to be run after artifactory.preStartCommand + # preStartCommand: + labels: {} + persistence: + ## Set existingClaim to true or false + ## If true, you must prepare a PVC with the name e.g `volume-myrelease-artifactory-ha-member-0` + existingClaim: false + replicaCount: 0 + updateStrategy: + type: RollingUpdate + minAvailable: 1 + ## Resources for the member nodes + resources: {} + # requests: + # memory: "1Gi" + # cpu: "500m" + # limits: + # memory: "2Gi" + # cpu: "1" + ## The following Java options are passed to the java process running Artifactory member nodes. + ## You should set them according to the resources set above + javaOpts: + # xms: "1g" + # xmx: "2g" + # corePoolSize: 24 + jmx: + enabled: false + port: 9010 + host: + ssl: false + # When authenticate is true, accessFile and passwordFile are required + authenticate: false + accessFile: + passwordFile: + # other: "" + nodeSelector: {} + ## Wait for Artifactory primary + waitForPrimaryStartup: + enabled: true + ## Setting time will override the built in test and will just wait the set time + time: + tolerations: [] + ## Complete specification of the "affinity" of the member nodes; if this is non-empty, + ## "podAntiAffinity" values are not used. + affinity: {} + ## Only used if "affinity" is empty + podAntiAffinity: + ## Valid values are "soft" or "hard"; any other value indicates no anti-affinity + type: "soft" + topologyKey: "kubernetes.io/hostname" +frontend: + name: frontend + enabled: true + internalPort: 8070 + ## Extra environment variables that can be used to tune frontend to your needs. + ## Uncomment and set value as needed + extraEnvironmentVariables: + # - name: MY_ENV_VAR + # value: "" + resources: {} + # requests: + # memory: "100Mi" + # cpu: "100m" + # limits: + # memory: "1Gi" + # cpu: "1" + ## Session settings + session: + ## Time in minutes after which the frontend token will need to be refreshed + timeoutMinutes: '30' + ## Add lifecycle hooks for frontend container + lifecycle: {} + # postStart: + # exec: + # command: ["/bin/sh", "-c", "echo Hello from the postStart handler"] + # preStop: + # exec: + # command: ["/bin/sh","-c","echo Hello from the preStop handler"] + + ## The following settings are to configure the frequency of the liveness and startup probes when splitServicesToContainers set to true + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl --fail --max-time {{ .Values.probes.timeoutSeconds }} http://localhost:{{ .Values.frontend.internalPort }}/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " --cert=ca.crt --key=ca.private.key` + # customCertificatesSecretName: + + ## When resetAccessCAKeys is true, Access will regenerate the CA certificate and matching private key + # resetAccessCAKeys: false + database: + maxOpenConnections: 80 + tomcat: + connector: + maxThreads: 50 + sendReasonPhrase: false + extraConfig: 'acceptCount="100"' + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl --fail --max-time {{ .Values.probes.timeoutSeconds }} http://localhost:8040/access/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " /var/opt/jfrog/nginx/message"] + # preStop: + # exec: + # command: ["/bin/sh","-c","nginx -s quit; while killall -0 nginx; do sleep 1; done"] + + ## Sidecar containers for tailing Nginx logs + loggers: [] + # - access.log + # - error.log + + ## Loggers containers resources + loggersResources: {} + # requests: + # memory: "64Mi" + # cpu: "25m" + # limits: + # memory: "128Mi" + # cpu: "50m" + + ## Logs options + logs: + stderr: false + stdout: false + level: warn + ## A list of custom ports to expose on the NGINX pod. Follows the conventional Kubernetes yaml syntax for container ports. + customPorts: [] + # - containerPort: 8066 + # name: docker + + ## The nginx main conf was moved to files/nginx-main-conf.yaml. This key is commented out to keep support for the old configuration + # mainConf: | + + ## The nginx artifactory conf was moved to files/nginx-artifactory-conf.yaml. This key is commented out to keep support for the old configuration + # artifactoryConf: | + customInitContainers: "" + customSidecarContainers: "" + customVolumes: "" + customVolumeMounts: "" + customCommand: + ## allows overwriting the command for the nginx container. + ## defaults to [ 'nginx', '-g', 'daemon off;' ] + + service: + ## For minikube, set this to NodePort, elsewhere use LoadBalancer + type: LoadBalancer + ssloffload: false + ## @param service.ipFamilyPolicy Controller Service ipFamilyPolicy (optional, cloud specific) + ## This can be either SingleStack, PreferDualStack or RequireDualStack + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilyPolicy: "" + ## @param service.ipFamilies Controller Service ipFamilies (optional, cloud specific) + ## This can be either ["IPv4"], ["IPv6"], ["IPv4", "IPv6"] or ["IPv6", "IPv4"] + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilies: [] + ## For supporting whitelist on the Nginx LoadBalancer service + ## Set this to a list of IP CIDR ranges + ## Example: loadBalancerSourceRanges: ['10.10.10.5/32', '10.11.10.5/32'] + ## or pass from helm command line + ## Example: helm install ... --set nginx.service.loadBalancerSourceRanges='{10.10.10.5/32,10.11.10.5/32}' + loadBalancerSourceRanges: [] + ## Provide static ip address + loadBalancerIP: + ## There are two available options: "Cluster" (default) and "Local". + externalTrafficPolicy: Cluster + labels: {} + # label-key: label-value + ## If the type is NodePort you can set a fixed port + # nodePort: 32082 + ## A list of custom ports to be exposed on nginx service. Follows the conventional Kubernetes yaml syntax for service ports. + customPorts: [] + # - port: 8066 + # targetPort: 8066 + # protocol: TCP + # name: docker + + annotations: {} + ## Renamed nginx internalPort 80,443 to 8080,8443 to support openshift + http: + enabled: true + externalPort: 80 + internalPort: 8080 + https: + enabled: true + externalPort: 443 + internalPort: 8443 + ## DEPRECATED: The following will be replaced by L1065-L1076 in a future release + # externalPortHttp: 80 + # internalPortHttp: 8080 + # externalPortHttps: 443 + # internalPortHttps: 8443 + + ssh: + internalPort: 1339 + externalPort: 1339 + ## The following settings are to configure the frequency of the liveness and readiness probes. + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl -s -k --fail --max-time {{ .Values.probes.timeoutSeconds }} {{ include "nginx.scheme" . }}://localhost:{{ include "nginx.port" . }}/ + initialDelaySeconds: {{ if semverCompare " + ## 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) + ## + # storageClassName: "-" + resources: {} + # requests: + # memory: "250Mi" + # cpu: "100m" + # limits: + # memory: "250Mi" + # cpu: "500m" + + nodeSelector: {} + tolerations: [] + affinity: {} +## Filebeat Sidecar container +## The provided filebeat configuration is for Artifactory logs. It assumes you have a logstash installed and configured properly. +filebeat: + enabled: false + name: artifactory-filebeat + image: + repository: "docker.elastic.co/beats/filebeat" + version: 7.16.2 + logstashUrl: "logstash:5044" + terminationGracePeriod: 10 + livenessProbe: + exec: + command: + - sh + - -c + - | + #!/usr/bin/env bash -e + curl --fail 127.0.0.1:5066 + failureThreshold: 3 + initialDelaySeconds: 10 + periodSeconds: 10 + timeoutSeconds: 5 + readinessProbe: + exec: + command: + - sh + - -c + - | + #!/usr/bin/env bash -e + filebeat test output + failureThreshold: 3 + initialDelaySeconds: 10 + periodSeconds: 10 + timeoutSeconds: 5 + resources: {} + # requests: + # memory: "100Mi" + # cpu: "100m" + # limits: + # memory: "100Mi" + # cpu: "100m" + + filebeatYml: | + logging.level: info + path.data: {{ .Values.artifactory.persistence.mountPath }}/log/filebeat + name: artifactory-filebeat + queue.spool: + file: + permissions: 0760 + filebeat.inputs: + - type: log + enabled: true + close_eof: ${CLOSE:false} + paths: + - {{ .Values.artifactory.persistence.mountPath }}/log/*.log + fields: + service: "jfrt" + log_type: "artifactory" + output: + logstash: + hosts: ["{{ .Values.filebeat.logstashUrl }}"] +## Allows to add additional kubernetes resources +## Use --- as a separator between multiple resources +## For an example, refer - https://github.com/jfrog/log-analytics-prometheus/blob/master/helm/artifactory-ha-values.yaml +additionalResources: "" +## Adding entries to a Pod's /etc/hosts file +## For an example, refer - https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases +hostAliases: [] +# - ip: "127.0.0.1" +# hostnames: +# - "foo.local" +# - "bar.local" +# - ip: "10.1.2.3" +# hostnames: +# - "foo.remote" +# - "bar.remote" + +## Toggling this feature is seamless and requires helm upgrade +## will enable all microservices to run in different containers in a single pod (by default it is true) +splitServicesToContainers: true +## Specify common probes parameters +probes: + timeoutSeconds: 5 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/CHANGELOG.md b/charts/jfrog/artifactory-jcr/107.90.14/CHANGELOG.md new file mode 100644 index 0000000000..3a984d30f2 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/CHANGELOG.md @@ -0,0 +1,206 @@ +# JFrog Container Registry Chart Changelog +All changes to this chart will be documented in this file. + +## [107.90.14] - Feb 20, 2024 +* Updated `artifactory.installerInfo` content + +## [107.80.0] - Feb 1, 2024 +* Updated README.md to create a namespace using `--create-namespace` as part of helm install + +## [107.74.0] - Nov 23, 2023 +* **IMPORTANT** +* Added min kubeVersion ">= 1.19.0-0" in chart.yaml + +## [107.66.0] - Jul 20, 2023 +* Disabled federation services when splitServicesToContainers=true + +## [107.45.0] - Aug 25, 2022 +* Included event service as mandatory and remove the flag from values.yaml + +## [107.41.0] - Jul 22, 2022 +* Bumping chart version to align with app version +* Disabled jfconnect and event services when splitServicesToContainers=true + +## [107.19.4] - May 27, 2021 +* Bumping chart version to align with app version +* Update dependency Artifactory chart version to 107.19.4 + +## [4.0.0] - Apr 22, 2021 +* **Breaking change:** +* Increased default postgresql persistence size to `200Gi` +* Update postgresql tag version to `13.2.0-debian-10-r55` +* Update postgresql chart version to `10.3.18` in chart.yaml - [10.x Upgrade Notes](https://github.com/bitnami/charts/tree/master/bitnami/postgresql#to-1000) +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x/12.x's postgresql.image.tag, previous postgresql.persistence.size and databaseUpgradeReady=true +* **IMPORTANT** +* This chart is only helm v3 compatible. +* Update dependency Artifactory chart version to 12.0.0 (Artifactory 7.18.3) + +## [3.8.0] - Apr 5, 2021 +* **IMPORTANT** +* Added `charts.jfrog.io` as default JFrog Helm repository +* Update dependency Artifactory chart version to 11.13.0 (Artifactory 7.17.5) + +## [3.7.0] - Mar 31, 2021 +* Update dependency Artifactory chart version to 11.12.2 (Artifactory 7.17.4) + +## [3.6.0] - Mar 15, 2021 +* Update dependency Artifactory chart version to 11.10.0 (Artifactory 7.16.3) + +## [3.5.1] - Mar 03, 2021 +* Update dependency Artifactory chart version to 11.9.3 (Artifactory 7.15.4) + +## [3.5.0] - Feb 18, 2021 +* Update dependency Artifactory chart version to 11.9.0 (Artifactory 7.15.3) + +## [3.4.1] - Feb 08, 2021 +* Update dependency Artifactory chart version to 11.8.0 (Artifactory 7.12.8) + +## [3.4.0] - Jan 4, 2020 +* Update dependency Artifactory chart version to 11.7.4 (Artifactory 7.12.5) + +## [3.3.1] - Dec 1, 2020 +* Update dependency Artifactory chart version to 11.5.4 (Artifactory 7.11.5) + +## [3.3.0] - Nov 23, 2020 +* Update dependency Artifactory chart version to 11.5.2 (Artifactory 7.11.2) + +## [3.2.2] - Nov 9, 2020 +* Update dependency Artifactory chart version to 11.4.5 (Artifactory 7.10.6) + +## [3.2.1] - Nov 2, 2020 +* Update dependency Artifactory chart version to 11.4.4 (Artifactory 7.10.5) + +## [3.2.0] - Oct 19, 2020 +* Update dependency Artifactory chart version to 11.4.0 (Artifactory 7.10.2) + +## [3.1.0] - Sep 30, 2020 +* Update dependency Artifactory chart version to 11.1.0 (Artifactory 7.9.0) + +## [3.0.2] - Sep 23, 2020 +* Updates readme + +## [3.0.1] - Sep 15, 2020 +* Update dependency Artifactory chart version to 11.0.1 (Artifactory 7.7.8) + +## [3.0.0] - Sep 14, 2020 +* **Breaking change:** Added `image.registry` and changed `image.version` to `image.tag` for docker images +* Update dependency Artifactory chart version to 11.0.0 (Artifactory 7.7.3) + +## [2.5.1] - Jul 29, 2020 +* Update dependency Artifactory chart version to 10.0.12 (Artifactory 7.6.3) + +## [2.5.0] - Jul 10, 2020 +* Update dependency Artifactory chart version to 10.0.3 (Artifactory 7.6.2) +* **IMPORTANT** +* Added ChartCenter Helm repository in README + +## [2.4.0] - Jun 30, 2020 +* Update dependency Artifactory chart version to 9.6.0 (Artifactory 7.6.1) + +## [2.3.1] - Jun 12, 2020 +* Update dependency Artifactory chart version to 9.5.2 (Artifactory 7.5.7) + +## [2.3.0] - Jun 1, 2020 +* Update dependency Artifactory chart version to 9.5.0 (Artifactory 7.5.5) + +## [2.2.5] - May 27, 2020 +* Update dependency Artifactory chart version to 9.4.9 (Artifactory 7.4.3) + +## [2.2.4] - May 20, 2020 +* Update dependency Artifactory chart version to 9.4.6 (Artifactory 7.4.3) + +## [2.2.3] - May 07, 2020 +* Update dependency Artifactory chart version to 9.4.5 (Artifactory 7.4.3) +* Add `installerInfo` string format + +## [2.2.2] - Apr 28, 2020 +* Update dependency Artifactory chart version to 9.4.4 (Artifactory 7.4.3) + +## [2.2.1] - Apr 27, 2020 +* Update dependency Artifactory chart version to 9.4.3 (Artifactory 7.4.1) + +## [2.2.0] - Apr 14, 2020 +* Update dependency Artifactory chart version to 9.4.0 (Artifactory 7.4.1) + +## [2.2.0] - Apr 14, 2020 +* Update dependency Artifactory chart version to 9.4.0 (Artifactory 7.4.1) + +## [2.1.6] - Apr 13, 2020 +* Update dependency Artifactory chart version to 9.3.1 (Artifactory 7.3.2) + +## [2.1.5] - Apr 8, 2020 +* Update dependency Artifactory chart version to 9.2.8 (Artifactory 7.3.2) + +## [2.1.4] - Mar 30, 2020 +* Update dependency Artifactory chart version to 9.2.3 (Artifactory 7.3.2) + +## [2.1.3] - Mar 30, 2020 +* Update dependency Artifactory chart version to 9.2.1 (Artifactory 7.3.2) + +## [2.1.2] - Mar 26, 2020 +* Update dependency Artifactory chart version to 9.1.5 (Artifactory 7.3.2) + +## [2.1.1] - Mar 25, 2020 +* Update dependency Artifactory chart version to 9.1.4 (Artifactory 7.3.2) + +## [2.1.0] - Mar 23, 2020 +* Update dependency Artifactory chart version to 9.1.3 (Artifactory 7.3.2) + +## [2.0.13] - Mar 19, 2020 +* Update dependency Artifactory chart version to 9.0.28 (Artifactory 7.2.1) + +## [2.0.12] - Mar 17, 2020 +* Update dependency Artifactory chart version to 9.0.26 (Artifactory 7.2.1) + +## [2.0.11] - Mar 11, 2020 +* Unified charts public release + +## [2.0.10] - Mar 8, 2020 +* Update dependency Artifactory chart version to 9.0.20 (Artifactory 7.2.1) + +## [2.0.9] - Feb 26, 2020 +* Update dependency Artifactory chart version to 9.0.15 (Artifactory 7.2.1) + +## [2.0.0] - Feb 12, 2020 +* Update dependency Artifactory chart version to 9.0.0 (Artifactory 7.0.0) + +## [1.1.0] - Jan 19, 2020 +* Update dependency Artifactory chart version to 8.4.1 (Artifactory 6.17.0) + +## [1.1.1] - Feb 3, 2020 +* Update dependency Artifactory chart version to 8.4.4 + +## [1.1.0] - Jan 19, 2020 +* Update dependency Artifactory chart version to 8.4.1 (Artifactory 6.17.0) + +## [1.0.1] - Dec 31, 2019 +* Update dependency Artifactory chart version to 8.3.5 + +## [1.0.0] - Dec 23, 2019 +* Update dependency Artifactory chart version to 8.3.3 + +## [0.2.1] - Dec 12, 2019 +* Update dependency Artifactory chart version to 8.3.1 + +## [0.2.0] - Dec 1, 2019 +* Updated Artifactory version to 6.16.0 + +## [0.1.5] - Nov 28, 2019 +* Update dependency Artifactory chart version to 8.2.6 + +## [0.1.4] - Nov 20, 2019 +* Update Readme + +## [0.1.3] - Nov 20, 2019 +* Fix JCR logo url +* Update dependency to Artifactory 8.2.2 chart + +## [0.1.2] - Nov 20, 2019 +* Update JCR logo + +## [0.1.1] - Nov 20, 2019 +* Add `appVersion` to Chart.yaml + +## [0.1.0] - Nov 20, 2019 +* Initial release of the JFrog Container Registry helm chart diff --git a/charts/jfrog/artifactory-jcr/107.90.14/Chart.yaml b/charts/jfrog/artifactory-jcr/107.90.14/Chart.yaml new file mode 100644 index 0000000000..d0da8ca665 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/Chart.yaml @@ -0,0 +1,30 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: JFrog Container Registry + catalog.cattle.io/kube-version: '>= 1.19.0-0' + catalog.cattle.io/release-name: artifactory-jcr +apiVersion: v2 +appVersion: 7.90.14 +dependencies: +- name: artifactory + repository: file://charts/artifactory + version: 107.90.14 +description: JFrog Container Registry +home: https://jfrog.com/container-registry/ +icon: file://assets/icons/artifactory-jcr.png +keywords: +- artifactory +- jfrog +- container +- registry +- devops +- jfrog-container-registry +kubeVersion: '>= 1.19.0-0' +maintainers: +- email: helm@jfrog.com + name: Chart Maintainers at JFrog +name: artifactory-jcr +sources: +- https://github.com/jfrog/charts +type: application +version: 107.90.14 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/LICENSE b/charts/jfrog/artifactory-jcr/107.90.14/LICENSE new file mode 100644 index 0000000000..8dada3edaf --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/charts/jfrog/artifactory-jcr/107.90.14/README.md b/charts/jfrog/artifactory-jcr/107.90.14/README.md new file mode 100644 index 0000000000..c0051e61d1 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/README.md @@ -0,0 +1,125 @@ +# JFrog Container Registry Helm Chart + +JFrog Container Registry is a free Artifactory edition with Docker and Helm repositories support. + +**Heads up: Our Helm Chart docs are moving to our main documentation site. For Artifactory installers, see [Installing Artifactory](https://www.jfrog.com/confluence/display/JFROG/Installing+Artifactory).** + +## Prerequisites Details + +* Kubernetes 1.19+ + +## Chart Details +This chart will do the following: + +* Deploy JFrog Container Registry +* Deploy an optional Nginx server +* Deploy an optional PostgreSQL Database +* Optionally expose Artifactory with Ingress [Ingress documentation](https://kubernetes.io/docs/concepts/services-networking/ingress/) + +## Installing the Chart + +### Add JFrog Helm repository + +Before installing JFrog helm charts, you need to add the [JFrog helm repository](https://charts.jfrog.io) to your helm client. + +```bash +helm repo add jfrog https://charts.jfrog.io +helm repo update +``` + +### Install Chart +To install the chart with the release name `jfrog-container-registry`: +```bash +helm upgrade --install jfrog-container-registry --set artifactory.postgresql.postgresqlPassword= jfrog/artifactory-jcr --namespace artifactory-jcr --create-namespace +``` + +### Accessing JFrog Container Registry +**NOTE:** If using artifactory or nginx service type `LoadBalancer`, it might take a few minutes for JFrog Container Registry's public IP to become available. + +### Updating JFrog Container Registry +Once you have a new chart version, you can upgrade your deployment with +```bash +helm upgrade jfrog-container-registry jfrog/artifactory-jcr --namespace artifactory-jcr --create-namespace +``` + +### Special Upgrade Notes +#### Artifactory upgrade from 6.x to 7.x (App Version) +Arifactory 6.x to 7.x upgrade requires a one time migration process. This is done automatically on pod startup if needed. +It's possible to configure the migration timeout with the following configuration in extreme cases. The provided default should be more than enough for completion of the migration. +```yaml +artifactory: + artifactory: + # Migration support from 6.x to 7.x + migration: + enabled: true + timeoutSeconds: 3600 +``` +* Note: If you are upgrading from 1.x to 3.x and above chart versions, please delete the existing statefulset of postgresql before upgrading the chart due to breaking changes in postgresql subchart. +```bash +kubectl delete statefulsets -postgresql +``` +* For more details about artifactory chart upgrades refer [here](https://github.com/jfrog/charts/blob/master/stable/artifactory/UPGRADE_NOTES.md) + +### Deleting JFrog Container Registry + +```bash +helm delete jfrog-container-registry --namespace artifactory-jcr +``` + +This will delete your JFrog Container Registry deployment.
+**NOTE:** You might have left behind persistent volumes. You should explicitly delete them with +```bash +kubectl delete pvc ... +kubectl delete pv ... +``` + +## Database +The JFrog Container Registry chart comes with PostgreSQL deployed by default.
+For details on the PostgreSQL configuration or customising the database, Look at the options described in the [Artifactory helm chart](https://github.com/jfrog/charts/tree/master/stable/artifactory). + +### Ingress and TLS +To get Helm to create an ingress object with a hostname, add these two lines to your Helm command: +```bash +helm upgrade --install jfrog-container-registry \ + --set artifactory.nginx.enabled=false \ + --set artifactory.ingress.enabled=true \ + --set artifactory.ingress.hosts[0]="artifactory.company.com" \ + --set artifactory.artifactory.service.type=NodePort \ + jfrog/artifactory-jcr --namespace artifactory-jcr --create-namespace +``` + +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: + +```bash +kubectl create secret tls artifactory-tls --cert=path/to/tls.cert --key=path/to/tls.key +``` + +Include the secret's name, along with the desired hostnames, in the Artifactory Ingress TLS section of your custom `values.yaml` file: + +```yaml +artifactory: + artifactory: + ingress: + ## If true, Artifactory Ingress will be created + ## + enabled: true + + ## Artifactory Ingress hostnames + ## Must be provided if Ingress is enabled + ## + hosts: + - jfrog-container-registry.domain.com + annotations: + kubernetes.io/tls-acme: "true" + ## Artifactory Ingress TLS configuration + ## Secrets must be manually created in the namespace + ## + tls: + - secretName: artifactory-tls + hosts: + - jfrog-container-registry.domain.com +``` + +## Useful links +https://www.jfrog.com +https://www.jfrog.com/confluence/ diff --git a/charts/jfrog/artifactory-jcr/107.90.14/app-readme.md b/charts/jfrog/artifactory-jcr/107.90.14/app-readme.md new file mode 100644 index 0000000000..9d9b7d85fc --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/app-readme.md @@ -0,0 +1,18 @@ +# JFrog Container Registry Helm Chart + +Universal Repository Manager supporting all major packaging formats, build tools and CI servers. + +## Chart Details +This chart will do the following: + +* Deploy JFrog Container Registry +* Deploy an optional Nginx server +* Optionally expose Artifactory with Ingress [Ingress documentation](https://kubernetes.io/docs/concepts/services-networking/ingress/) + + +## Useful links +Blog: [Herd Trust Into Your Rancher Labs Multi-Cloud Strategy with Artifactory](https://jfrog.com/blog/herd-trust-into-your-rancher-labs-multi-cloud-strategy-with-artifactory/) + +## Activate Your Artifactory Instance +Don't have a license? Please send an email to [rancher-jfrog-licenses@jfrog.com](mailto:rancher-jfrog-licenses@jfrog.com) to get it. + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/.helmignore b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/.helmignore new file mode 100644 index 0000000000..b6e97f07fb --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/.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 + +tests/ \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/CHANGELOG.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/CHANGELOG.md new file mode 100644 index 0000000000..61f0d3d9a2 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/CHANGELOG.md @@ -0,0 +1,1365 @@ +# JFrog Artifactory Chart Changelog +All changes to this chart will be documented in this file. + +## [107.90.14] - July 18, 2024 +* Fixed #adding colon in image registry which breaks deployment [GH-1892](https://github.com/jfrog/charts/pull/1892) +* Added new `nginx.hosts` to use Nginx server_name directive instead of `ingress.hosts` +* Added a deprecation notice of ingress.hosts when `ngnix.enabled` is true +* Added new evidence service +* Corrected database connection values based on sizing +* **IMPORTANT** +* Separate access from artifactory tomcat to run on its own dedicated tomcat + * With this change access will be running in its own dedicated container + * This will give the ability to control resources and java options specific to access + Can be done by passing the following, + `access.javaOpts.other` + `access.resources` + `access.extraEnvironmentVariables` +* Updating the example link for downloading the DB driver +* Added Binary Provider recommendations + +## [107.89.0] - June 7, 2024 +* Fix the indentation of the commented-out sections in the values.yaml file +* Fixed sizing values by removing `JF_SHARED_NODE_HAENABLED` in xsmall/small configurations + +## [107.88.0] - May 29, 2024 +* **IMPORTANT** +* Refactored `nginx.artifactoryConf` and `nginx.mainConf` configuration (moved to files/nginx-artifactory-conf.yaml and files/nginx-main-conf.yaml instead of keys in values.yaml) + +## [107.87.0] - May 29, 2024 +* Renamed `.Values.artifactory.openMetrics` to `.Values.artifactory.metrics` + +## [107.85.0] - May 29, 2024 +* Changed `migration.enabled` to false by default. For 6.x to 7.x migration, this flag needs to be set to `true` + +## [107.84.0] - May 29, 2024 +* Added image section for `initContainers` instead of `initContainerImage` +* Renamed `router.image.imagePullPolicy` to `router.image.pullPolicy` +* Removed image section for `loggers` +* Added support for `global.verisons.initContainers` to override `initContainers.image.tag` +* Fixed an issue with extraSystemYaml merge +* **IMPORTANT** +* Renamed `artifactory.setSecurityContext` to `artifactory.podSecurityContext` +* Renamed `artifactory.uid` to `artifactory.podSecurityContext.runAsUser` +* Renamed `artifactory.gid` to `artifactory.podSecurityContext.runAsGroup` and `artifactory.podSecurityContext.fsGroup` +* Renamed `artifactory.fsGroupChangePolicy` to `artifactory.podSecurityContext.fsGroupChangePolicy` +* Renamed `artifactory.seLinuxOptions` to `artifactory.podSecurityContext.seLinuxOptions` +* Added flag `allowNonPostgresql` defaults to false +* Update postgresql tag version to `15.6.0-debian-12-r5` +* Added a check if `initContainerImage` exists +* Fixed an issue to generate unified secret to support artifactory fullname [GH-1882](https://github.com/jfrog/charts/issues/1882) +* Fixed an issue template render on loggers [GH-1883](https://github.com/jfrog/charts/issues/1883) +* Fixed resource constraints for "setup" initContainer of nginx deployment [GH-962] (https://github.com/jfrog/charts/issues/962) +* Added .Values.artifactory.unifiedSecretPrependReleaseName` for unified secret to prepend release name +* Fixed maxCacheSize and cacheProviderDir mix up under azure-blob-storage-v2-direct template in binarystore.xml + +## [107.82.0] - Mar 04, 2024 +* Added `disableRouterBypass` flag as experimental feature, to disable the artifactoryPath /artifactory/ and route all traffic through the Router. +* Removed Replicator service + +## [107.81.0] - Feb 20, 2024 +* **IMPORTANT** +* Refactored systemYaml configuration (moved to files/system.yaml instead of key in values.yaml) +* Added ability to provide `extraSystemYaml` configuration in values.yaml which will merge with the existing system yaml when `systemYamlOverride` is not given [GH-1848](https://github.com/jfrog/charts/pull/1848) +* Added option to modify the new cache configs, maxFileSizeLimit and skipDuringUpload +* Added IPV4/IPV6 Dualstack flag support for Artifactory and nginx service +* Added `singleStackIPv6Cluster` flag, which manages the Nginx configuration to enable listening on IPv6 and proxying. +* Fixing broken link for creating additional kubernetes resources. Refer [here](https://github.com/jfrog/log-analytics-prometheus/blob/master/helm/artifactory-values.yaml) +* Refactored installerInfo configuration (moved to files/installer-info.json instead of key in values.yaml) + +## [107.80.0] - Feb 20, 2024 +* Updated README.md to create a namespace using `--create-namespace` as part of helm install + +## [107.79.0] - Feb 20, 2024 +* **IMPORTANT** +* Added `unifiedSecretInstallation` flag which enables single unified secret holding all internal (chart) secrets to `true` by default +* Added support for azure-blob-storage-v2-direct config +* Added option to set Nginx to write access_log to container STDOUT +* **Important change:** +* Update postgresql tag version to `15.2.0-debian-11-r23` +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default bundles PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x/12.x/13.x's postgresql.image.tag, previous postgresql.persistence.size and databaseUpgradeReady=true + +## [107.77.0] - April 22, 2024 +* Removed integration service +* Added recommended postgresql sizing configurations under sizing directory +* Updated artifactory-federation (probes, port, embedded mode) +* Fixed - Removed duplicate keys of the sizing yaml file +* Fixing broken nginx port [GH-1860](https://github.com/jfrog/charts/issues/1860) +* Added nginx.customCommand to use custom commands for the nginx container + +## [107.76.0] - Dec 13, 2023 +* Added connectionTimeout and socketTimeout paramaters under AWSS3 binarystore section +* Reduced nginx startupProbe initialDelaySeconds + +## [107.74.0] - Nov 30, 2023 +* Added recommended sizing configurations under sizing directory, please refer [here](README.md/#apply-sizing-configurations-to-the-chart) +* **IMPORTANT** +* Added min kubeVersion ">= 1.19.0-0" in chart.yaml + +## [107.70.0] - Nov 30, 2023 +* Fixed - StatefulSet pod annotations changed from range to toYaml [GH-1828](https://github.com/jfrog/charts/issues/1828) +* Fixed - Invalid format for awsS3V3 `multiPartLimit,multipartElementSize` in binarystore.xml. +* Fixed - SecurityContext with runAsGroup in artifactory [GH-1838](https://github.com/jfrog/charts/issues/1838) +* Added support for custom labels in the Nginx pods [GH-1836](https://github.com/jfrog/charts/pull/1836) +* Added podSecurityContext and containerSecurityContext for nginx +* Added support for nginx on openshift, set `podSecurityContext` and `containerSecurityContext` to false +* Renamed nginx internalPort 80,443 to 8080,8443 to support openshift + +## [107.69.0] - Sep 18, 2023 +* Adjust rtfs context +* Fixed - Metadata service does not respect customVolumeMounts for DB CAs [GH-1815](https://github.com/jfrog/charts/issues/1815) + +## [107.68.8] - Sep 18, 2023 +* Reverted - Enabled `unifiedSecretInstallation` by default [GH-1819](https://github.com/jfrog/charts/issues/1819) +* Removed openshift condition check from NOTES.txt + +## [107.68.7] - Aug 28, 2023 +* Enabled `unifiedSecretInstallation` by default + +## [107.67.0] - Aug 28, 2023 +* Add 'extraJavaOpts' and 'port' values to federation service + +## [107.66.0] - Aug 28, 2023 +* Added federation service container in artifactory +* Add rtfs service to ingress in artifactory + +## [107.64.0] - Aug 28, 2023 +* Added support to configure event.webhooks within generated system.yaml +* Fixed an issue to generate ssl certificate should support artifactory fullname +* Added binarystore.xml template to persistence storage type `nfs`. The default Filestore location configured according to artifactory.persistence.nfs.dataDir. +* Added 'multiPartLimit' and 'multipartElementSize' parameters to awsS3V3 binary providers. +* Increased default Artifactory Tomcat acceptCount config to 400 +* Fixed Illegal Strict-Transport-Security header in nginx config + +## [107.63.0] - Aug 28, 2023 +* Added support for Openshift by adding the securityContext in container level. +* **IMPORTANT** +* Disable securityContext in container and pod level to deploy postgres on openshift. +* Fixed support for fsGroup in non openshift environemnt and runAsGroup in openshift environment. +* Fixed - Helm Template Error when using artifactory.loggers [GH-1791](https://github.com/jfrog/charts/issues/1791) +* Removed the nginx disable condition for openshift +* Fixed jfconnect disabling as micro-service on splitcontainers [GH-1806](https://github.com/jfrog/charts/issues/1806) + +## [107.62.0] - Jun 5, 2023 +* Upgraded to autoscaling/v2 +* Added support for 'port' and 'useHttp' parameters for s3-storage-v3 binary provider [GH-1767](https://github.com/jfrog/charts/issues/1767) + +## [107.61.0] - May 31, 2023 +* Added new binary provider `google-storage-v2-direct` +* Added missing parameter 'enableSignedUrlRedirect' to 'googleStorage' + +## [107.60.0] - May 31, 2023 +* Enabled `splitServicesToContainers` to true by default +* Updated the recommended values for small, medium and large installations to support the 'splitServicesToContainers' + +## [107.59.0] - May 31, 2023 +* Fixed reference of `terminationGracePeriodSeconds` +* Added Support for Cold Artifact Storage as part of the systemYaml configuration (disabled by default) +* Added new binary provider `s3-storage-v3-archive` +* Fixed jfconnect disabling as micro-service on non-splitcontainers +* Fixed wrong cache-fs provider ID of cluster-s3-storage-v3 in the binarystore.xml [GH-1772](https://github.com/jfrog/charts/issues/1772) + +## [107.58.0] - Mar 23, 2023 +* Updated postgresql multi-arch tag version to `13.10.0-debian-11-r14` +* Removed obselete remove-lost-found initContainer` +* Added env JF_SHARED_NODE_HAENABLED under frontend when running in the container split mode + +## [107.57.0] - Mar 02, 2023 +* Updated initContainerImage and logger image to `ubi9/ubi-minimal:9.1.0.1793` + +## [107.55.0] - Jan 31, 2023 +* Updated initContainerImage and logger image to `ubi9/ubi-minimal:9.1.0.1760` +* Adding a custom preStop to Artifactory router for allowing graceful termination to complete + +## [107.53.0] - Jan 20, 2023 +* Updated initContainerImage and logger image to `ubi8/ubi-minimal:8.7.1049` + +## [107.50.0] - Jan 20, 2023 +* Updated postgresql tag version to `13.9.0-debian-11-11` +* Fixed an issue for capabilities check of ingress +* Updated jfrogUrl text path in migrate.sh file +* Added a note that from 107.46.x chart versions, `copyOnEveryStartup` is not needed for binarystore.xml, it is always copied via initContainers. For more Info, Refer [GH-1723](https://github.com/jfrog/charts/issues/1723) + +## [107.49.0] - Jan 16, 2023 +* Added support for setting `seLinuxOptions` in `securityContext` [GH-1699](https://github.com/jfrog/charts/pull/1699) +* Added option to enable/disable proxy_request_buffering and proxy_buffering_off [GH-1686](https://github.com/jfrog/charts/pull/1686) +* Updated initContainerImage and logger image to `ubi8/ubi-minimal:8.7.1049` + +## [107.48.0] - Oct 27, 2022 +* Updated router version to `7.51.0` + +## [107.47.0] - Sep 29, 2022 +* Updated initContainerImage to `ubi8/ubi-minimal:8.6-941` +* Added support for annotations for artifactory statefulset and nginx deployment [GH-1665](https://github.com/jfrog/charts/pull/1665) +* Updated router version to `7.49.0` + +## [107.46.0] - Sep 14, 2022 +* **IMPORTANT** +* Added support for lifecycle hooks for all containers, changed `artifactory.postStartCommand` to `.Values.artifactory.lifecycle.postStart.exec.command` +* Updated initContainerImage to `ubi8/ubi-minimal:8.6-902` +* Update nginx configuration to allow websocket requests when using pipelines +* Fixed an issue to allow artifactory to make direct API calls to store instead via jfconnect service when `splitServicesToContainers=true` +* Refactor binarystore.xml configuration (moved to `files/binarystore.xml` instead of key in values.yaml) +* Added new binary providers `cluster-s3-storage-v3`, `s3-storage-v3-direct`, `azure-blob-storage-direct`, `google-storage-v2` +* Deprecated (removed) `aws-s3` binary provider [JetS3t library](https://www.jfrog.com/confluence/display/JFROG/Configuring+the+Filestore#ConfiguringtheFilestore-BinaryProvider) +* Deprecated (removed) `google-storage` binary provider and force persistence storage type `google-storage` to work with `google-storage-v2` only +* Copy binarystore.xml in init Container to fix existing persistence on file system in clear text +* Removed obselete `.Values.artifactory.binarystore.enabled` key +* Removed `newProbes.enabled`, default to new probes +* Added nginx.customCommand using inotifyd to reload nginx's config upon ssl secret or configmap changes [GH-1640](https://github.com/jfrog/charts/pull/1640) + +## [107.43.0] - Aug 25, 2022 +* Added flag `artifactory.replicator.ingress.enabled` to enable/disable ingress for replicator +* Updated initContainerImage to `ubi8/ubi-minimal:8.6-854` +* Updated router version to `7.45.0` +* Added flag `artifactory.schedulerName` to set for the pods the value of schedulerName field [GH-1606](https://github.com/jfrog/charts/issues/1606) +* Enabled TLS based on access or router in values.yaml + +## [107.42.0] - Aug 25, 2022 +* Enabled database creds secret to use from unified secret +* Updated router version to `7.42.0` +* Fix duplicate volumes for userPluginSecrets [GH-1650] (https://github.com/jfrog/charts/issues/1650) +* Added support to truncate (> 63 chars) for unifiedCustomSecretVolumeName + +## [107.41.0] - June 27, 2022 +* Added support for nginx.terminationGracePeriodSeconds [GH-1645](https://github.com/jfrog/charts/issues/1645) +* Use an alternate command for `find` to copy custom certificates +* Added support for circle of trust using `circleOfTrustCertificatesSecret` secret name [GH-1623](https://github.com/jfrog/charts/pull/1623) + +## [107.40.0] - June 16, 2022 +* Added support for PodDisruptionBudget [GH-1618](https://github.com/jfrog/charts/issues/1618) +* From artifactory 7.38.x, joinKey can be retrived from Admin > User Management > Settings in UI +* Allow templating for pod annotations [GH-1634](https://github.com/jfrog/charts/pull/1634) +* Fixed `customPersistentPodVolumeClaim` name to `customPersistentVolumeClaim` +* Added flags to control enable/disable infra services in splitServicesToContainers + +## [107.39.0] - May 31, 2022 +* Fix default `artifactory.async.corePoolSize` [GH-1612](https://github.com/jfrog/charts/issues/1612) +* Added support of nginx annotations +* Reduce startupProbe `initialDelaySeconds` +* Align all liveness and readiness probes failureThreshold to `5` seconds +* Added new flag `unifiedSecretInstallation` to enables single unified secret holding all the artifactory secrets +* Updated router version to `7.38.0` +* Add support for NFS config with directories `haBackupDir` and `haDataDir` +* Fixed - disable jfconnect on oss/jcr/cpp flavours [GH-1630](https://github.com/jfrog/charts/issues/1630) + +## [107.38.0] - May 04, 2022 +* Added support for `global.nodeSelector` to artifactory and nginx pods +* Updated router version to `7.36.1` +* Added support for custom global probes timeout +* Updated frontend container command +* Added topologySpreadConstraints to artifactory and nginx, and add lifecycle hooks to nginx [GH-1596](https://github.com/jfrog/charts/pull/1596) +* Added support of extraEnvironmentVariables for all infra services containers +* Enabled the consumption (jfconnect) flag by default +* Fix jfconnect disabling on non-splitcontainers + +## [107.37.0] - Mar 08, 2022 +* Added support for customPorts in nginx deployment +* Bugfix - Wrong proxy_pass configurations for /artifactory/ in the default artifactory.conf +* Added signedUrlExpirySeconds option to artifactory.persistence.type aws-S3-V3 +* Updated router version to `7.35.0` +* Added useInstanceCredentials,enableSignedUrlRedirect option to google-storage-v2 +* Changed dependency charts repo to `charts.jfrog.io` + +## [107.36.0] - Mar 03, 2022 +* Remove pdn tracker which starts replicator service +* Added silent option for curl probes +* Added readiness health check for the artifactory container for k8s version < 1.20 +* Fix property file migration issue to system.yaml 6.x to 7.x + +## [107.35.0] - Feb 08, 2022 +* Updated router version to `7.32.1` + +## [107.33.0] - Jan 11, 2022 +* Add more user friendly support for anti-affinity +* Pod anti-affinity is now enabled by default (soft rule) +* Readme fixes +* Added support for setting `fsGroupChangePolicy` +* Added nginx customInitContainers, customVolumes, customSidecarContainers [GH-1565](https://github.com/jfrog/charts/pull/1565) +* Updated router version to `7.30.0` + +## [107.32.0] - Dec 22, 2021 +* Updated logger image to `jfrog/ubi-minimal:8.5-204` +* Added default `8091` as `artifactory.tomcat.maintenanceConnector.port` for probes check +* Refactored probes to replace httpGet probes with basic exec + curl +* Refactored `database-creds` secret to create only when database values are passed +* Added new endpoints for probes `/artifactory/api/v1/system/liveness` and `/artifactory/api/v1/system/readiness` +* Enabled `newProbes:true` by default to use these endpoints +* Fix filebeat sidecar spool file permissions +* Updated filebeat sidecar container to `7.16.2` + +## [107.31.0] - Dec 17, 2021 +* Added support for HorizontalPodAutoscaler apiVersion `autoscaling/v2beta2` +* Remove integration service feature flag to make it mandatory service +* Update postgresql tag version to `13.4.0-debian-10-r39` +* Fixed `artifactory.resources` indentation in `migration-artifactory` init container [GH-1562](https://github.com/jfrog/charts/issues/1562) +* Refactored `router.requiredServiceTypes` to support platform chart + +## [107.30.0] - Nov 30, 2021 +* Fixed incorrect permission for filebeat.yaml +* Updated healthcheck (liveness/readiness) api for integration service +* Disable readiness health check for the artifactory container when running in the container split mode +* Ability to start replicator on enabling pdn tracker + +## [107.29.0] - Nov 26, 2021 +* Added integration service container in artifactory +* Add support for Ingress Class Name in Ingress Spec [GH-1516](https://github.com/jfrog/charts/pull/1516) +* Fixed chart values to use curl instead of wget [GH-1529](https://github.com/jfrog/charts/issues/1529) +* Updated nginx config to allow websockets when pipelines is enabled +* Moved router.topology.local.requireqservicetypes from system.yaml to router as environment variable +* Added jfconnect in system.yaml +* Updated artifactory container’s health probes to use artifactory api on rt-split +* Updated initContainerImage to `jfrog/ubi-minimal:8.5-204` +* Updated router version to `7.28.2` +* Set Jfconnect enabled to `false` in the artifactory container when running in the container split mode + +## [107.28.0] - Nov 11, 2021 +* Added default values cpu and memeory in initContainers +* Updated router version to `7.26.0` +* Updated (`rbac.create` and `serviceAccount.create` to false by default) for least privileges +* Fixed incorrect data type for `Values.router.serviceRegistry.insecure` in default values.yaml [GH-1514](https://github.com/jfrog/charts/pull/1514/files) +* **IMPORTANT** +* Changed init-container images from `alpine` to `ubi8/ubi-minimal` +* Added support for AWS License Manager using `.Values.aws.licenseConfigSecretName` + +## [107.27.0] - Oct 6, 2021 +* **Breaking change** +* Aligned probe structure (moved probes variables under config block) +* Added support for new probes(set to false by default) +* Bugfix - Invalid format for `multiPartLimit,multipartElementSize,maxCacheSize` in binarystore.xml [GH-1466](https://github.com/jfrog/charts/issues/1466) +* Added missioncontrol container in artifactory +* Dropped NET_RAW capability for the containers +* Added resources to migration-artifactory init container +* Added resources to all rt split containers +* Updated router version to `7.25.1` +* Added support for Ingress networking.k8s.io/v1/Ingress for k8s >=1.22 [GH-1487](https://github.com/jfrog/charts/pull/1487) +* Added min kubeVersion ">= 1.14.0-0" in chart.yaml +* Update alpine tag version to `3.14.2` +* Update busybox tag version to `1.33.1` +* Artifactory chart support for cluster license + +## [107.26.0] - Aug 23, 2021 +* Added Observability container (only when `splitServicesToContainers` is enabled) +* Support for high availability (when replicaCount > 1) +* Added min kubeVersion ">= 1.12.0-0" in chart.yaml + +## [107.25.0] - Aug 13, 2021 +* Updated readme of chart to point to wiki. Refer [Installing Artifactory](https://www.jfrog.com/confluence/display/JFROG/Installing+Artifactory) +* Added startupProbe and livenessProbe for RT-split containers +* Updated router version to 7.24.1 +* Added security hardening fixes +* Enabled startup probes for k8s >= 1.20.x +* Changed network policy to allow all ingress and egress traffic +* Added Observability changes +* Added support for global.versions.router (only when `splitServicesToContainers` is enabled) + +## [107.24.0] - July 27, 2021 +* Support global and product specific tags at the same time +* Added support for artifactory containers split + +## [107.23.0] - July 8, 2021 +* Bug fix - logger sideCar picks up Wrong File in helm +* Allow filebeat metrics configuration in values.yaml + +## [107.22.0] - July 6, 2021 +* Update alpine tag version to `3.14.0` +* Added `nodePort` support to artifactory-service and nginx-service templates +* Removed redundant `terminationGracePeriodSeconds` in statefulset +* Increased `startupProbe.failureThreshold` time + +## [107.21.3] - July 2, 2021 +* Added ability to change sendreasonphrase value in server.xml via system yaml + +## [107.19.3] - May 20, 2021 +* Fix broken support for startupProbe for k8s < 1.18.x +* Added support for `nameOverride` and `fullnameOverride` in values.yaml + +## [107.18.6] - April 29, 2021 +* Bumping chart version to align with app version +* Add `securityContext` option on nginx container + +## [12.0.0] - April 22, 2021 +* **Breaking change:** +* Increased default postgresql persistence size to `200Gi` +* Update postgresql tag version to `13.2.0-debian-10-r55` +* Update postgresql chart version to `10.3.18` in chart.yaml - [10.x Upgrade Notes](https://github.com/bitnami/charts/tree/master/bitnami/postgresql#to-1000) +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x/12.x's postgresql.image.tag, previous postgresql.persistence.size and databaseUpgradeReady=true +* **IMPORTANT** +* This chart is only helm v3 compatible. +* Fixed filebeat-configmap naming +* Explicitly set ServiceAccount `automountServiceAccountToken` to 'true' +* Update alpine tag version to `3.13.5` + +## [11.13.2] - April 15, 2021 +* Updated Artifactory version to 7.17.9 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.9) + +## [11.13.1] - April 6, 2021 +* Updated Artifactory version to 7.17.6 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.6) +* Update alpine tag version to `3.13.4` + +## [11.13.0] - April 5, 2021 +* **IMPORTANT** +* Added `charts.jfrog.io` as default JFrog Helm repository +* Updated Artifactory version to 7.17.5 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.5) + +## [11.12.2] - Mar 31, 2021 +* Updated Artifactory version to 7.17.4 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.17.4) + +## [11.12.1] - Mar 30, 2021 +* Updated Artifactory version to 7.17.3 +* Add `timeoutSeconds` to all exec probes - Please refer [here](https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes) + +## [11.12.0] - Mar 24, 2021 +* Updated Artifactory version to 7.17.2 +* Optimized startupProbe time + +## [11.11.0] - Mar 18, 2021 +* Add support to startupProbe + +## [11.10.0] - Mar 15, 2021 +* Updated Artifactory version to 7.16.3 + +## [11.9.5] - Mar 09, 2021 +* Added HSTS header to nginx conf + +## [11.9.4] - Mar 9, 2021 +* Removed bintray URL references in the chart + +## [11.9.3] - Mar 04, 2021 +* Updated Artifactory version to 7.15.4 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.15.4) + +## [11.9.2] - Mar 04, 2021 +* Fixed creation of nginx-certificate-secret when Nginx is disabled + +## [11.9.1] - Feb 19, 2021 +* Update busybox tag version to `1.32.1` + +## [11.9.0] - Feb 18, 2021 +* Updated Artifactory version to 7.15.3 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.15.3) +* Add option to specify update strategy for Artifactory statefulset + +## [11.8.1] - Feb 11, 2021 +* Exposed "multiPartLimit" and "multipartElementSize" for the Azure Blob Storage Binary Provider + +## [11.8.0] - Feb 08, 2021 +* Updated Artifactory version to 7.12.8 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.12.8) +* Support for custom certificates using secrets +* **Important:** Switched docker images download from `docker.bintray.io` to `releases-docker.jfrog.io` +* Update alpine tag version to `3.13.1` + +## [11.7.8] - Jan 25, 2021 +* Add support for hostAliases + +## [11.7.7] - Jan 11, 2021 +* Fix failures when using creds file for configurating google storage + +## [11.7.6] - Jan 11, 2021 +* Updated Artifactory version to 7.12.6 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.12.6) + +## [11.7.5] - Jan 07, 2021 +* Added support for optional tracker dedicated ingress `.Values.artifactory.replicator.trackerIngress.enabled` (defaults to false) + +## [11.7.4] - Jan 04, 2021 +* Fixed gid support for statefulset + +## [11.7.3] - Dec 31, 2020 +* Added gid support for statefulset +* Add setSecurityContext flag to allow securityContext block to be removed from artifactory statefulset + +## [11.7.2] - Dec 29, 2020 +* **Important:** Removed `.Values.metrics` and `.Values.fluentd` (Fluentd and Prometheus integrations) +* Add support for creating additional kubernetes resources - [refer here](https://github.com/jfrog/log-analytics-prometheus/blob/master/artifactory-values.yaml) +* Updated Artifactory version to 7.12.5 + +## [11.7.1] - Dec 21, 2020 +* Updated Artifactory version to 7.12.3 + +## [11.7.0] - Dec 18, 2020 +* Updated Artifactory version to 7.12.2 +* Added `.Values.artifactory.openMetrics.enabled` + +## [11.6.1] - Dec 11, 2020 +* Added configurable `.Values.global.versions.artifactory` in values.yaml + +## [11.6.0] - Dec 10, 2020 +* Update postgresql tag version to `12.5.0-debian-10-r25` +* Fixed `artifactory.persistence.googleStorage.endpoint` from `storage.googleapis.com` to `commondatastorage.googleapis.com` +* Updated chart maintainers email + +## [11.5.5] - Dec 4, 2020 +* **Important:** Renamed `.Values.systemYaml` to `.Values.systemYamlOverride` + +## [11.5.4] - Dec 1, 2020 +* Improve error message returned when attempting helm upgrade command + +## [11.5.3] - Nov 30, 2020 +* Updated Artifactory version to 7.11.5 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.11) + +## [11.5.2] - Nov 23, 2020 +* Updated Artifactory version to 7.11.2 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.11) +* Updated port namings on services and pods to allow for istio protocol discovery +* Change semverCompare checks to support hosted Kubernetes +* Add flag to disable creation of ServiceMonitor when enabling prometheus metrics +* Prevent the PostHook command to be executed if the user did not specify a command in the values file +* Fix issue with tls file generation when nginx.https.enabled is false + +## [11.5.1] - Nov 19, 2020 +* Updated Artifactory version to 7.11.2 +* Bugfix - access.config.import.xml override Access Federation configurations + +## [11.5.0] - Nov 17, 2020 +* Updated Artifactory version to 7.11.1 +* Update alpine tag version to `3.12.1` + +## [11.4.6] - Nov 10, 2020 +* Pass system.yaml via external secret for advanced usecases +* Added support for custom ingress +* Bugfix - stateful set not picking up changes to database secrets + +## [11.4.5] - Nov 9, 2020 +* Updated Artifactory version to 7.10.6 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.10.6) + +## [11.4.4] - Nov 2, 2020 +* Add enablePathStyleAccess property for aws-s3-v3 binary provider template + +## [11.4.3] - Nov 2, 2020 +* Updated Artifactory version to 7.10.5 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.10.5) + +## [11.4.2] - Oct 22, 2020 +* Chown bug fix where Linux capability cannot chown all files causing log line warnings +* Fix Frontend timeout linting issue + +## [11.4.1] - Oct 20, 2020 +* Add flag to disable prepare-custom-persistent-volume init container + +## [11.4.0] - Oct 19, 2020 +* Updated Artifactory version to 7.10.2 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.10.2) + +## [11.3.2] - Oct 15, 2020 +* Add support to specify priorityClassName for nginx deployment + +## [11.3.1] - Oct 9, 2020 +* Add support for customInitContainersBegin + +## [11.3.0] - Oct 7, 2020 +* Updated Artifactory version to 7.9.1 +* **Breaking change:** Fix `storageClass` to correct `storageClassName` in values.yaml + +## [11.2.0] - Oct 5, 2020 +* Expose Prometheus metrics via a ServiceMonitor +* Parse log files for metric data with Fluentd + +## [11.1.0] - Sep 30, 2020 +* Updated Artifactory version to 7.9.0 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.9) +* Added support for resources in init container + +## [11.0.11] - Sep 25, 2020 +* Update to use linux capability CAP_CHOWN instead of root base init container to avoid any use of root containers to pass Redhat security requirements + +## [11.0.10] - Sep 28, 2020 +* Setting chart coordinates in migitation yaml + +## [11.0.9] - Sep 25, 2020 +* Update filebeat version to `7.9.2` + +## [11.0.8] - Sep 24, 2020 +* Fixed broken issue - when setting `waitForDatabase: false` container startup still waits for DB + +## [11.0.7] - Sep 22, 2020 +* Readme updates + +## [11.0.6] - Sep 22, 2020 +* Fix lint issue in migitation yaml + +## [11.0.5] - Sep 22, 2020 +* Fix broken migitation yaml + +## [11.0.4] - Sep 21, 2020 +* Added mitigation yaml for Artifactory - [More info](https://github.com/jfrog/chartcenter/blob/master/docs/securitymitigationspec.md) + +## [11.0.3] - Sep 17, 2020 +* Added configurable session(UI) timeout in frontend microservice + +## [11.0.2] - Sep 17, 2020 +* Added proper required text to be shown while postgres upgrades + +## [11.0.1] - Sep 14, 2020 +* Updated Artifactory version to 7.7.8 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.7.8) + +## [11.0.0] - Sep 2, 2020 +* **Breaking change:** Changed `imagePullSecrets`values from string to list. +* **Breaking change:** Added `image.registry` and changed `image.version` to `image.tag` for docker images +* Added support for global values +* Updated maintainers in chart.yaml +* Update postgresql tag version to `12.3.0-debian-10-r71` +* Update postgresql chart version to `9.3.4` in requirements.yaml - [9.x Upgrade Notes](https://github.com/bitnami/charts/tree/master/bitnami/postgresql#900) +* **IMPORTANT** +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass previous 9.x/10.x's postgresql.image.tag and databaseUpgradeReady=true + +## [10.1.0] - Aug 13, 2020 +* Updated Artifactory version to 7.7.3 - [Release Notes](https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.7) + +## [10.0.15] - Aug 10, 2020 +* Added enableSignedUrlRedirect for persistent storage type aws-s3-v3. + +## [10.0.14] - Jul 31, 2020 +* Update the README section on Nginx SSL termination to reflect the actual YAML structure. + +## [10.0.13] - Jul 30, 2020 +* Added condition to disable the migration scripts. + +## [10.0.12] - Jul 28, 2020 +* Document Artifactory node affinity. + +## [10.0.11] - Jul 28, 2020 +* Added maxConnections for persistent storage type aws-s3-v3. + +## [10.0.10] - Jul 28, 2020 +* Bugfix / support for userPluginSecrets with Artifactory 7 + +## [10.0.9] - Jul 27, 2020 +* Add tpl to external database secrets +* Modified `scheme` to `artifactory.scheme` + +## [10.0.8] - Jul 23, 2020 +* Added condition to disable the migration init container. + +## [10.0.7] - Jul 21, 2020 +* Updated Artifactory Chart to add node and primary labels to pods and service objects. + +## [10.0.6] - Jul 20, 2020 +* Support custom CA and certificates + +## [10.0.5] - Jul 13, 2020 +* Updated Artifactory version to 7.6.3 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.6.3 +* Fixed Mysql database jar path in `preStartCommand` in README + +## [10.0.4] - Jul 10, 2020 +* Move some postgresql values to where they should be according to the subchart + +## [10.0.3] - Jul 8, 2020 +* Set Artifactory access client connections to the same value as the access threads + +## [10.0.2] - Jul 6, 2020 +* Updated Artifactory version to 7.6.2 +* **IMPORTANT** +* Added ChartCenter Helm repository in README + +## [10.0.1] - Jul 01, 2020 +* Add dedicated ingress object for Replicator service when enabled + +## [10.0.0] - Jun 30, 2020 +* Update postgresql tag version to `10.13.0-debian-10-r38` +* Update alpine tag version to `3.12` +* Update busybox tag version to `1.31.1` +* **IMPORTANT** +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), you need to pass postgresql.image.tag=9.6.18-debian-10-r7 and databaseUpgradeReady=true + +## [9.6.0] - Jun 29, 2020 +* Updated Artifactory version to 7.6.1 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.6.1 +* Add tpl for external database secrets + +## [9.5.5] - Jun 25, 2020 +* Stop loading the Nginx stream module because it is now a core module + +## [9.5.4] - Jun 25, 2020 +* Notes.txt update - add --namespace parameter + +## [9.5.3] - Jun 11, 2020 +* Support list of custom secrets + +## [9.5.2] - Jun 12, 2020 +* Updated Artifactory version to 7.5.7 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.5.7 + +## [9.5.1] - Jun 8, 2020 +* Readme update - configuring Artifactory with oracledb + +## [9.5.0] - Jun 1, 2020 +* Updated Artifactory version to 7.5.5 - https://www.jfrog.com/confluence/display/JFROG/Artifactory+Release+Notes#ArtifactoryReleaseNotes-Artifactory7.5 +* Fixes bootstrap configMap permission issue +* Update postgresql tag version to `9.6.18-debian-10-r7` + +## [9.4.9] - May 27, 2020 +* Added Tomcat maxThreads & acceptCount + +## [9.4.8] - May 25, 2020 +* Fixed postgresql README `image` Parameters + +## [9.4.7] - May 24, 2020 +* Fixed typo in README regarding migration timeout + +## [9.4.6] - May 19, 2020 +* Added metadata maxOpenConnections + +## [9.4.5] - May 07, 2020 +* Fix `installerInfo` string format + +## [9.4.4] - Apr 27, 2020 +* Updated Artifactory version to 7.4.3 + +## [9.4.3] - Apr 26, 2020 +* Change order of the customInitContainers to run before the "migration-artifactory" initContainer. + +## [9.4.2] - Apr 24, 2020 +* Fix `artifactory.persistence.awsS3V3.useInstanceCredentials` incorrect conditional logic +* Bump postgresql tag version to `9.6.17-debian-10-r72` in values.yaml + +## [9.4.1] - Apr 16, 2020 +* Custom volumes in migration init container. + +## [9.4.0] - Apr 14, 2020 +* Updated Artifactory version to 7.4.1 + +## [9.3.1] - April 13, 2020 +* Update README with helm v3 commands + +## [9.3.0] - April 10, 2020 +* Use dependency charts from `https://charts.bitnami.com/bitnami` +* Bump postgresql chart version to `8.7.3` in requirements.yaml +* Bump postgresql tag version to `9.6.17-debian-10-r21` in values.yaml + +## [9.2.9] - Apr 8, 2020 +* Added recommended ingress annotation to avoid 413 errors + +## [9.2.8] - Apr 8, 2020 +* Moved migration scripts under `files` directory +* Support preStartCommand in migration Init container as `artifactory.migration.preStartCommand` + +## [9.2.7] - Apr 6, 2020 +* Fix cache size (should be 5gb instead of 50gb since volume claim is only 20gb). + +## [9.2.6] - Apr 1, 2020 +* Support masterKey and joinKey as secrets + +## [9.2.5] - Apr 1, 2020 +* Fix readme use to `-hex 32` instead of `-hex 16` + +## [9.2.4] - Mar 31, 2020 +* Change the way the artifactory `command:` is set so it will properly pass a SIGTERM to java + +## [9.2.3] - Mar 29, 2020 +* Add Nginx log options: stderr as logfile and log level + +## [9.2.2] - Mar 30, 2020 +* Use the same defaulting mechanism used for the artifactory version used elsewhere in the chart + +## [9.2.1] - Mar 29, 2020 +* Fix loggers sidecars configurations to support new file system layout and new log names + +## [9.2.0] - Mar 29, 2020 +* Fix broken admin user bootstrap configuration +* **Breaking change:** renamed `artifactory.accessAdmin` to `artifactory.admin` + +## [9.1.5] - Mar 26, 2020 +* Fix volumeClaimTemplate issue + +## [9.1.4] - Mar 25, 2020 +* Fix volume name used by filebeat container + +## [9.1.3] - Mar 24, 2020 +* Use `postgresqlExtendedConf` for setting custom PostgreSQL configuration (instead of `postgresqlConfiguration`) + +## [9.1.2] - Mar 22, 2020 +* Support for SSL offload in Nginx service(LoadBalancer) layer. Introduced `nginx.service.ssloffload` field with boolean type. + +## [9.1.1] - Mar 23, 2020 +* Moved installer info to values.yaml so it is fully customizable + +## [9.1.0] - Mar 23, 2020 +* Updated Artifactory version to 7.3.2 + +## [9.0.29] - Mar 20, 2020 +* Add support for masterKey trim during 6.x to 7.x migration if 6.x masterKey is 32 hex (64 characters) + +## [9.0.28] - Mar 18, 2020 +* Increased Nginx proxy_buffers size + +## [9.0.27] - Mar 17, 2020 +* Changed all single quotes to double quotes in values files +* useInstanceCredentials variable was declared in S3 settings but not used in chart. Now it is being used. + +## [9.0.26] - Mar 17, 2020 +* Fix rendering of Service Account annotations + +## [9.0.25] - Mar 16, 2020 +* Update Artifactory readme with extra ingress annotations needed for Artifactory to be set as SSO provider + +## [9.0.24] - Mar 16, 2020 +* Add Unsupported message from 6.18 to 7.2.x (migration) + +## [9.0.23] - Mar 12, 2020 +* Fix README.md rendering issue + +## [9.0.22] - Mar 11, 2020 +* Upgrade Docs update + +## [9.0.21] - Mar 11, 2020 +* Unified charts public release + +## [9.0.20] - Mar 6, 2020 +* Fix path to `/artifactory_bootstrap` +* Add support for controlling the name of the ingress and allow to set more than one cname + +## [9.0.19] - Mar 4, 2020 +* Add support for disabling `consoleLog` in `system.yaml` file + +## [9.0.18] - Feb 28, 2020 +* Add support to process `valueFrom` for extraEnvironmentVariables + +## [9.0.17] - Feb 26, 2020 +* Fix join key secret naming + +## [9.0.16] - Feb 26, 2020 +* Store join key to secret + +## [9.0.15] - Feb 26, 2020 +* Updated Artifactory version to 7.2.1 + +## [9.0.10] - Feb 07, 2020 +* Remove protection flag `databaseUpgradeReady` which was added to check internal postgres upgrade + +## [9.0.0] - Feb 07, 2020 +* Updated Artifactory version to 7.0.0 + +## [8.4.8] - Feb 13, 2020 +* Add support for SSH authentication to Artifactory + +## [8.4.7] - Feb 11, 2020 +* Change Artifactory service port name to be hard-coded to `http` instead of using `{{ .Release.Name }}` + +## [8.4.6] - Feb 9, 2020 +* Add support for `tpl` in the `postStartCommand` + +## [8.4.5] - Feb 4, 2020 +* Support customisable Nginx kind + +## [8.4.4] - Feb 2, 2020 +* Add a comment stating that it is recommended to use an external PostgreSQL with a static password for production installations + +## [8.4.3] - Jan 30, 2020 +* Add the option to configure resources for the logger containers + +## [8.4.2] - Jan 26, 2020 +* Improve `database.user` and `database.password` logic in order to support more use cases and make the configuration less repetitive + +## [8.4.1] - Jan 19, 2020 +* Fix replicator port config in nginx replicator configmap + +## [8.4.0] - Jan 19, 2020 +* Updated Artifactory version to 6.17.0 + +## [8.3.6] - Jan 16, 2020 +* Added example for external nginx-ingress + +## [8.3.5] - Dec 30, 2019 +* Fix for nginx probes failing when launched with http disabled + +## [8.3.4] - Dec 24, 2019 +* Better support for custom `artifactory.internalPort` + +## [8.3.3] - Dec 23, 2019 +* Mark empty map values with `{}` + +## [8.3.2] - Dec 16, 2019 +* Fix for toggling nginx service ports + +## [8.3.1] - Dec 12, 2019 +* Add support for toggling nginx service ports + +## [8.3.0] - Dec 1, 2019 +* Updated Artifactory version to 6.16.0 + +## [8.2.6] - Nov 28, 2019 +* Add support for using existing PriorityClass + +## [8.2.5] - Nov 27, 2019 +* Add support for PriorityClass + +## [8.2.4] - Nov 21, 2019 +* Add an option to use a file system cache-fs with the file-system binarystore template + +## [8.2.3] - Nov 20, 2019 +* Update Artifactory Readme + +## [8.2.2] - Nov 20, 2019 +* Update Artfactory logo + +## [8.2.1] - Nov 18, 2019 +* Add the option to provide service account annotations (in order to support stuff like https://docs.aws.amazon.com/eks/latest/userguide/specify-service-account-role.html) + +## [8.2.0] - Nov 18, 2019 +* Updated Artifactory version to 6.15.0 + +## [8.1.11] - Nov 17, 2019 +* Do not provide a default master key. Allow it to be auto generated by Artifactory on first startup + +## [8.1.10] - Nov 17, 2019 +* Fix creation of double slash in nginx artifactory configuration + +## [8.1.9] - Nov 14, 2019 +* Set explicit `postgresql.postgresqlPassword=""` to avoid helm v3 error + +## [8.1.8] - Nov 12, 2019 +* Updated Artifactory version to 6.14.1 + +## [8.1.7] - Nov 9, 2019 +* Additional documentation for masterKey + +## [8.1.6] - Nov 10, 2019 +* Update PostgreSQL chart version to 7.0.1 +* Use formal PostgreSQL configuration format + +## [8.1.5] - Nov 8, 2019 +* Add support `artifactory.service.loadBalancerSourceRanges` for whitelisting when setting `artifactory.service.type=LoadBalancer` + +## [8.1.4] - Nov 6, 2019 +* Add support for any type of environment variable by using `extraEnvironmentVariables` as-is + +## [8.1.3] - Nov 6, 2019 +* Add nodeselector support for Postgresql + +## [8.1.2] - Nov 5, 2019 +* Add support for the aws-s3-v3 filestore, which adds support for pod IAM roles + +## [8.1.1] - Nov 4, 2019 +* When using `copyOnEveryStartup`, make sure that the target base directories are created before copying the files + +## [8.1.0] - Nov 3, 2019 +* Updated Artifactory version to 6.14.0 + +## [8.0.1] - Nov 3, 2019 +* Make sure the artifactory pod exits when one of the pre-start stages fail + +## [8.0.0] - Oct 27, 2019 +**IMPORTANT - BREAKING CHANGES!**
+**DOWNTIME MIGHT BE REQUIRED FOR AN UPGRADE!** +* If this is a new deployment or you already use an external database (`postgresql.enabled=false`), these changes **do not affect you**! +* If this is an upgrade and you are using the default PostgreSQL (`postgresql.enabled=true`), must use the upgrade instructions in [UPGRADE_NOTES.md](UPGRADE_NOTES.md)! +* PostgreSQL sub chart was upgraded to version `6.5.x`. This version is **not backward compatible** with the old version (`0.9.5`)! +* Note the following **PostgreSQL** Helm chart changes + * The chart configuration has changed! See [values.yaml](values.yaml) for the new keys used + * **PostgreSQL** is deployed as a StatefulSet + * See [PostgreSQL helm chart](https://hub.helm.sh/charts/stable/postgresql) for all available configurations + +## [7.18.3] - Oct 24, 2019 +* Change the preStartCommand to support templating + +## [7.18.2] - Oct 21, 2019 +* Add support for setting `artifactory.labels` +* Add support for setting `nginx.labels` + +## [7.18.1] - Oct 10, 2019 +* Updated Artifactory version to 6.13.1 + +## [7.18.0] - Oct 7, 2019 +* Updated Artifactory version to 6.13.0 + +## [7.17.5] - Sep 24, 2019 +* Option to skip wait-for-db init container with '--set waitForDatabase=false' + +## [7.17.4] - Sep 11, 2019 +* Updated Artifactory version to 6.12.2 + +## [7.17.3] - Sep 9, 2019 +* Updated Artifactory version to 6.12.1 + +## [7.17.2] - Aug 22, 2019 +* Fix the nginx server_name directive used with ingress.hosts + +## [7.17.1] - Aug 21, 2019 +* Enable the Artifactory container's liveness and readiness probes + +## [7.17.0] - Aug 21, 2019 +* Updated Artifactory version to 6.12.0 + +## [7.16.11] - Aug 14, 2019 +* Updated Artifactory version to 6.11.6 + +## [7.16.10] - Aug 11, 2019 +* Fix Ingress routing and add an example + +## [7.16.9] - Aug 5, 2019 +* Do not mount `access/etc/bootstrap.creds` unless user specifies a custom password or secret (Access already generates a random password if not provided one) +* If custom `bootstrap.creds` is provided (using keys or custom secret), prepare it with an init container so the temp file does not persist + +## [7.16.8] - Aug 4, 2019 +* Improve binarystore config + 1. Convert to a secret + 2. Move config to values.yaml + 3. Support an external secret + +## [7.16.7] - Jul 29, 2019 +* Don't create the nginx configmaps when nginx.enabled is false + +## [7.16.6] - Jul 24, 2019 +* Simplify nginx setup and shorten initial wait for probes + +## [7.16.5] - Jul 22, 2019 +* Change Ingress API to be compatible with recent kubernetes versions + +## [7.16.4] - Jul 22, 2019 +* Updated Artifactory version to 6.11.3 + +## [7.16.3] - Jul 11, 2019 +* Add ingress.hosts to the Nginx server_name directive when ingress is enabled to help with Docker repository sub domain configuration + +## [7.16.2] - Jul 3, 2019 +* Fix values key in reverse proxy example + +## [7.16.1] - Jul 1, 2019 +* Updated Artifactory version to 6.11.1 + +## [7.16.0] - Jun 27, 2019 +* Update Artifactory version to 6.11 and add restart to Artifactory when bootstrap.creds file has been modified + +## [7.15.8] - Jun 27, 2019 +* Add the option for changing nginx config using values.yaml and remove outdated reverse proxy documentation + +## [7.15.6] - Jun 24, 2019 +* Update chart maintainers + +## [7.15.5] - Jun 24, 2019 +* Change Nginx to point to the artifactory externalPort + +## [7.15.4] - Jun 23, 2019 +* Add the option to provide an IP for the access-admin endpoints + +## [7.15.3] - Jun 23, 2019 +* Add values files for small, medium and large installations + +## [7.15.2] - Jun 20, 2019 +* Add missing terminationGracePeriodSeconds to values.yaml + +## [7.15.1] - Jun 19, 2019 +* Updated Artifactory version to 6.10.4 + +## [7.15.0] - Jun 17, 2019 +* Use configmaps for nginx configuration and remove nginx postStart command + +## [7.14.8] - Jun 18, 2019 +* Add the option to provide additional ingress rules + +## [7.14.7] - Jun 14, 2019 +* Updated readme with improved external database setup example + +## [7.14.6] - Jun 11, 2019 +* Updated Artifactory version to 6.10.3 +* Updated installer-info template + +## [7.14.5] - Jun 6, 2019 +* Updated Google Cloud Storage API URL and https settings + +## [7.14.4] - Jun 5, 2019 +* Delete the db.properties file on Artifactory startup + +## [7.14.3] - Jun 3, 2019 +* Updated Artifactory version to 6.10.2 + +## [7.14.2] - May 21, 2019 +* Updated Artifactory version to 6.10.1 + +## [7.14.1] - May 19, 2019 +* Fix missing logger image tag + +## [7.14.0] - May 7, 2019 +* Updated Artifactory version to 6.10.0 + +## [7.13.21] - May 5, 2019 +* Add support for setting `artifactory.async.corePoolSize` + +## [7.13.20] - May 2, 2019 +* Remove unused property `artifactory.releasebundle.feature.enabled` + +## [7.13.19] - May 1, 2019 +* Fix indentation issue with the replicator system property + +## [7.13.18] - Apr 30, 2019 +* Add support for JMX monitoring + +## [7.13.17] - Apr 25, 2019 +* Added support for `cacheProviderDir` + +## [7.13.16] - Apr 18, 2019 +* Changing API StatefulSet version to `v1` and permission fix for custom `artifactory.conf` for Nginx + +## [7.13.15] - Apr 16, 2019 +* Updated documentation for Reverse Proxy Configuration + +## [7.13.14] - Apr 15, 2019 +* Added support for `customVolumeMounts` + +## [7.13.13] - Aprl 12, 2019 +* Added support for `bucketExists` flag for googleStorage + +## [7.13.12] - Apr 11, 2019 +* Replace `curl` examples with `wget` due to the new base image + +## [7.13.11] - Aprl 07, 2019 +* Add support for providing the Artifactory license as a parameter + +## [7.13.10] - Apr 10, 2019 +* Updated Artifactory version to 6.9.1 + +## [7.13.9] - Aprl 04, 2019 +* Add support for templated extraEnvironmentVariables + +## [7.13.8] - Aprl 07, 2019 +* Change network policy API group + +## [7.13.7] - Aprl 04, 2019 +* Bugfix for userPluginSecrets + +## [7.13.6] - Apr 4, 2019 +* Add information about upgrading Artifactory with auto-generated postgres password + +## [7.13.5] - Aprl 03, 2019 +* Added installer info + +## [7.13.4] - Aprl 03, 2019 +* Allow secret names for user plugins to contain template language + +## [7.13.3] - Apr 02, 2019 +* Allow NetworkPolicy configurations (defaults to allow all) + +## [7.13.2] - Aprl 01, 2019 +* Add support for user plugin secret + +## [7.13.1] - Mar 27, 2019 +* Add the option to copy a list of files to ARTIFACTORY_HOME on startup + +## [7.13.0] - Mar 26, 2019 +* Updated Artifactory version to 6.9.0 + +## [7.12.18] - Mar 25, 2019 +* Add CI tests for persistence, ingress support and nginx + +## [7.12.17] - Mar 22, 2019 +* Add the option to change the default access-admin password + +## [7.12.16] - Mar 22, 2019 +* Added support for `.Probe.path` to customise the paths used for health probes + +## [7.12.15] - Mar 21, 2019 +* Added support for `artifactory.customSidecarContainers` to create custom sidecar containers +* Added support for `artifactory.customVolumes` to create custom volumes + +## [7.12.14] - Mar 21, 2019 +* Make ingress path configurable + +## [7.12.13] - Mar 19, 2019 +* Move the copy of bootstrap config from postStart to preStart + +## [7.12.12] - Mar 19, 2019 +* Fix existingClaim example + +## [7.12.11] - Mar 18, 2019 +* Add information about nginx persistence + +## [7.12.10] - Mar 15, 2019 +* Wait for nginx configuration file before using it + +## [7.12.9] - Mar 15, 2019 +* Revert securityContext changes since they were causing issues + +## [7.12.8] - Mar 15, 2019 +* Fix issue #247 (init container failing to run) + +## [7.12.7] - Mar 14, 2019 +* Updated Artifactory version to 6.8.7 +* Add support for Artifactory-CE for C++ + +## [7.12.6] - Mar 13, 2019 +* Move securityContext to container level + +## [7.12.5] - Mar 11, 2019 +* Updated Artifactory version to 6.8.6 + +## [7.12.4] - Mar 8, 2019 +* Fix existingClaim option + +## [7.12.3] - Mar 5, 2019 +* Updated Artifactory version to 6.8.4 + +## [7.12.2] - Mar 4, 2019 +* Add support for catalina logs sidecars + +## [7.12.1] - Feb 27, 2019 +* Updated Artifactory version to 6.8.3 + +## [7.12.0] - Feb 25, 2019 +* Add nginx support for tail sidecars + +## [7.11.1] - Feb 20, 2019 +* Added support for enterprise storage + +## [7.10.2] - Feb 19, 2019 +* Updated Artifactory version to 6.8.2 + +## [7.10.1] - Feb 17, 2019 +* Updated Artifactory version to 6.8.1 +* Add example of `SERVER_XML_EXTRA_CONNECTOR` usage + +## [7.10.0] - Feb 15, 2019 +* Updated Artifactory version to 6.8.0 + +## [7.9.6] - Feb 13, 2019 +* Updated Artifactory version to 6.7.3 + +## [7.9.5] - Feb 12, 2019 +* Add support for tail sidecars to view logs from k8s api + +## [7.9.4] - Feb 6, 2019 +* Fix support for customizing statefulset `terminationGracePeriodSeconds` + +## [7.9.3] - Feb 5, 2019 +* Add instructions on how to deploy Artifactory with embedded Derby database + +## [7.9.2] - Feb 5, 2019 +* Add support for customizing statefulset `terminationGracePeriodSeconds` + +## [7.9.1] - Feb 3, 2019 +* Updated Artifactory version to 6.7.2 + +## [7.9.0] - Jan 23, 2019 +* Updated Artifactory version to 6.7.0 + +## [7.8.9] - Jan 22, 2019 +* Added support for `artifactory.customInitContainers` to create custom init containers + +## [7.8.8] - Jan 17, 2019 +* Added support of values ingress.labels + +## [7.8.7] - Jan 16, 2019 +* Mount replicator.yaml (config) directly to /replicator_extra_conf + +## [7.8.6] - Jan 13, 2019 +* Fix documentation about nginx group id + +## [7.8.5] - Jan 13, 2019 +* Updated Artifactory version to 6.6.5 + +## [7.8.4] - Jan 8, 2019 +* Make artifactory.replicator.publicUrl required when the replicator is enabled + +## [7.8.3] - Jan 1, 2019 +* Updated Artifactory version to 6.6.3 +* Add support for `artifactory.extraEnvironmentVariables` to pass more environment variables to Artifactory + +## [7.8.2] - Dec 28, 2018 +* Fix location `replicator.yaml` is copied to + +## [7.8.1] - Dec 27, 2018 +* Updated Artifactory version to 6.6.1 + +## [7.8.0] - Dec 20, 2018 +* Updated Artifactory version to 6.6.0 + +## [7.7.13] - Dec 17, 2018 +* Updated Artifactory version to 6.5.13 + +## [7.7.12] - Dec 12, 2018 +* Fix documentation about Artifactory license setup using secret + +## [7.7.11] - Dec 10, 2018 +* Fix issue when using existing claim + +## [7.7.10] - Dec 5, 2018 +* Remove Distribution certificates creation. + +## [7.7.9] - Nov 30, 2018 +* Updated Artifactory version to 6.5.9 + +## [7.7.8] - Nov 29, 2018 +* Updated postgresql version to 9.6.11 + +## [7.7.7] - Nov 27, 2018 +* Updated Artifactory version to 6.5.8 + +## [7.7.6] - Nov 19, 2018 +* Added support for configMap to use custom Reverse Proxy Configuration with Nginx + +## [7.7.5] - Nov 14, 2018 +* Fix location of `nodeSelector`, `affinity` and `tolerations` + +## [7.7.4] - Nov 14, 2018 +* Updated Artifactory version to 6.5.3 + +## [7.7.3] - Nov 12, 2018 +* Support artifactory.preStartCommand for running command before entrypoint starts + +## [7.7.2] - Nov 7, 2018 +* Support database.url parameter (DB_URL) + +## [7.7.1] - Oct 29, 2018 +* Change probes port to 8040 (so they will not be blocked when all tomcat threads on 8081 are exhausted) + +## [7.7.0] - Oct 28, 2018 +* Update postgresql chart to version 0.9.5 to be able and use `postgresConfig` options + +## [7.6.8] - Oct 23, 2018 +* Fix providing external secret for database credentials + +## [7.6.7] - Oct 23, 2018 +* Allow user to configure externalTrafficPolicy for Loadbalancer + +## [7.6.6] - Oct 22, 2018 +* Updated ingress annotation support (with examples) to support docker registry v2 + +## [7.6.5] - Oct 21, 2018 +* Updated Artifactory version to 6.5.2 + +## [7.6.4] - Oct 19, 2018 +* Allow providing pre-existing secret containing master key +* Allow arbitrary annotations on primary and member node pods +* Enforce size limits when using local storage with `emptyDir` +* Allow providing pre-existing secrets containing external database credentials + +## [7.6.3] - Oct 18, 2018 +* Updated Artifactory version to 6.5.1 + +## [7.6.2] - Oct 17, 2018 +* Add Apache 2.0 license + +## [7.6.1] - Oct 11, 2018 +* Supports master-key in the secrets and stateful-set +* Allows ingress default `backend` to be enabled or disabled (defaults to enabled) + +## [7.6.0] - Oct 11, 2018 +* Updated Artifactory version to 6.5.0 + +## [7.5.4] - Oct 9, 2018 +* Quote ingress hosts to support wildcard names + +## [7.5.3] - Oct 4, 2018 +* Add PostgreSQL resources template + +## [7.5.2] - Oct 2, 2018 +* Add `helm repo add jfrog https://charts.jfrog.io` to README + +## [7.5.1] - Oct 2, 2018 +* Set Artifactory to 6.4.1 + +## [7.5.0] - Sep 27, 2018 +* Set Artifactory to 6.4.0 + +## [7.4.3] - Sep 26, 2018 +* Add ci/test-values.yaml + +## [7.4.2] - Sep 2, 2018 +* Updated Artifactory version to 6.3.2 +* Removed unused PVC + +## [7.4.0] - Aug 22, 2018 +* Added support to run as non root +* Updated Artifactory version to 6.2.0 + +## [7.3.0] - Aug 22, 2018 +* Enabled RBAC Support +* Added support for PostStartCommand (To download Database JDBC connector) +* Increased postgresql max_connections +* Added support for `nginx.conf` ConfigMap +* Updated Artifactory version to 6.1.0 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.lock b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.lock new file mode 100644 index 0000000000..8064c323b5 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: postgresql + repository: https://charts.jfrog.io/ + version: 10.3.18 +digest: sha256:404ce007353baaf92a6c5f24b249d5b336c232e5fd2c29f8a0e4d0095a09fd53 +generated: "2022-03-08T08:53:16.293311+05:30" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.yaml new file mode 100644 index 0000000000..58bbbab3fe --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/Chart.yaml @@ -0,0 +1,24 @@ +apiVersion: v2 +appVersion: 7.90.14 +dependencies: +- condition: postgresql.enabled + name: postgresql + repository: https://charts.jfrog.io/ + version: 10.3.18 +description: Universal Repository Manager supporting all major packaging formats, + build tools and CI servers. +home: https://www.jfrog.com/artifactory/ +icon: https://raw.githubusercontent.com/jfrog/charts/master/stable/artifactory/logo/artifactory-logo.png +keywords: +- artifactory +- jfrog +- devops +kubeVersion: '>= 1.19.0-0' +maintainers: +- email: installers@jfrog.com + name: Chart Maintainers at JFrog +name: artifactory +sources: +- https://github.com/jfrog/charts +type: application +version: 107.90.14 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/LICENSE b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/LICENSE new file mode 100644 index 0000000000..8dada3edaf --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/README.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/README.md new file mode 100644 index 0000000000..da3304ee58 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/README.md @@ -0,0 +1,59 @@ +# JFrog Artifactory Helm Chart + +**IMPORTANT!** Our Helm Chart docs have moved to our main documentation site. Below you will find the basic instructions for installing, uninstalling, and deleting Artifactory. For all other information, refer to [Installing Artifactory](https://www.jfrog.com/confluence/display/JFROG/Installing+Artifactory#InstallingArtifactory-HelmInstallation). + +## Prerequisites +* Kubernetes 1.19+ +* Artifactory Pro trial license [get one from here](https://www.jfrog.com/artifactory/free-trial/) + +## Chart Details +This chart will do the following: + +* Deploy Artifactory-Pro/Artifactory-Edge (or OSS/CE if custom image is set) +* Deploy a PostgreSQL database using the stable/postgresql chart (can be changed) **NOTE:** For production grade installations it is recommended to use an external PostgreSQL. +* Deploy an optional Nginx server +* Optionally expose Artifactory with Ingress [Ingress documentation](https://kubernetes.io/docs/concepts/services-networking/ingress/) + +## Installing the Chart + +### Add JFrog Helm repository + +Before installing JFrog helm charts, you need to add the [JFrog helm repository](https://charts.jfrog.io) to your helm client + +```bash +helm repo add jfrog https://charts.jfrog.io +helm repo update +``` + +### Install Chart +To install the chart with the release name `artifactory`: +```bash +helm upgrade --install artifactory jfrog/artifactory --namespace artifactory --create-namespace +``` + +### Apply Sizing configurations to the Chart +To apply the chart with recommended sizing configurations : +For small configurations : +```bash +helm upgrade --install artifactory jfrog/artifactory -f sizing/artifactory-small-extra-config.yaml -f sizing/artifactory-small.yaml --namespace artifactory --create-namespace +``` + +## Uninstalling Artifactory + +Uninstall is supported only on Helm v3 and on. + +Uninstall Artifactory using the following command. + +```bash +helm uninstall artifactory && sleep 90 && kubectl delete pvc -l app=artifactory +``` + +## Deleting Artifactory + +**IMPORTANT:** Deleting Artifactory will also delete your data volumes and you will lose all of your data. You must back up all this information before deletion. You do not need to uninstall Artifactory before deleting it. + +To delete Artifactory use the following command. + +```bash +helm delete artifactory --namespace artifactory +``` diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/.helmignore b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/.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/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.lock b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.lock new file mode 100644 index 0000000000..3687f52df5 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common + repository: https://charts.bitnami.com/bitnami + version: 1.4.2 +digest: sha256:dce0349883107e3ff103f4f17d3af4ad1ea3c7993551b1c28865867d3e53d37c +generated: "2021-03-30T09:13:28.360322819Z" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.yaml new file mode 100644 index 0000000000..4b197b2071 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/Chart.yaml @@ -0,0 +1,29 @@ +annotations: + category: Database +apiVersion: v2 +appVersion: 11.11.0 +dependencies: +- name: common + repository: https://charts.bitnami.com/bitnami + version: 1.x.x +description: Chart for PostgreSQL, an object-relational database management system + (ORDBMS) with an emphasis on extensibility and on standards-compliance. +home: https://github.com/bitnami/charts/tree/master/bitnami/postgresql +icon: https://bitnami.com/assets/stacks/postgresql/img/postgresql-stack-220x234.png +keywords: +- postgresql +- postgres +- database +- sql +- replication +- cluster +maintainers: +- email: containers@bitnami.com + name: Bitnami +- email: cedric@desaintmartin.fr + name: desaintmartin +name: postgresql +sources: +- https://github.com/bitnami/bitnami-docker-postgresql +- https://www.postgresql.org/ +version: 10.3.18 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/README.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/README.md new file mode 100644 index 0000000000..63d3605bb8 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/README.md @@ -0,0 +1,770 @@ +# PostgreSQL + +[PostgreSQL](https://www.postgresql.org/) is an object-relational database management system (ORDBMS) with an emphasis on extensibility and on standards-compliance. + +For HA, please see [this repo](https://github.com/bitnami/charts/tree/master/bitnami/postgresql-ha) + +## TL;DR + +```console +$ helm repo add bitnami https://charts.bitnami.com/bitnami +$ helm install my-release bitnami/postgresql +``` + +## Introduction + +This chart bootstraps a [PostgreSQL](https://github.com/bitnami/bitnami-docker-postgresql) deployment on a [Kubernetes](http://kubernetes.io) cluster using the [Helm](https://helm.sh) package manager. + +Bitnami charts can be used with [Kubeapps](https://kubeapps.com/) for deployment and management of Helm Charts in clusters. This chart has been tested to work with NGINX Ingress, cert-manager, fluentd and Prometheus on top of the [BKPR](https://kubeprod.io/). + +## Prerequisites + +- Kubernetes 1.12+ +- Helm 3.1.0 +- PV provisioner support in the underlying infrastructure + +## Installing the Chart +To install the chart with the release name `my-release`: + +```console +$ helm install my-release bitnami/postgresql +``` + +The command deploys PostgreSQL on the Kubernetes cluster in the default configuration. The [Parameters](#parameters) section lists the parameters that can be configured during installation. + +> **Tip**: List all releases using `helm list` + +## Uninstalling the Chart + +To uninstall/delete the `my-release` deployment: + +```console +$ helm delete my-release +``` + +The command removes all the Kubernetes components but PVC's associated with the chart and deletes the release. + +To delete the PVC's associated with `my-release`: + +```console +$ kubectl delete pvc -l release=my-release +``` + +> **Note**: Deleting the PVC's will delete postgresql data as well. Please be cautious before doing it. + +## Parameters + +The following tables lists the configurable parameters of the PostgreSQL chart and their default values. + +| Parameter | Description | Default | +|-----------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------| +| `global.imageRegistry` | Global Docker Image registry | `nil` | +| `global.postgresql.postgresqlDatabase` | PostgreSQL database (overrides `postgresqlDatabase`) | `nil` | +| `global.postgresql.postgresqlUsername` | PostgreSQL username (overrides `postgresqlUsername`) | `nil` | +| `global.postgresql.existingSecret` | Name of existing secret to use for PostgreSQL passwords (overrides `existingSecret`) | `nil` | +| `global.postgresql.postgresqlPassword` | PostgreSQL admin password (overrides `postgresqlPassword`) | `nil` | +| `global.postgresql.servicePort` | PostgreSQL port (overrides `service.port`) | `nil` | +| `global.postgresql.replicationPassword` | Replication user password (overrides `replication.password`) | `nil` | +| `global.imagePullSecrets` | Global Docker registry secret names as an array | `[]` (does not add image pull secrets to deployed pods) | +| `global.storageClass` | Global storage class for dynamic provisioning | `nil` | +| `image.registry` | PostgreSQL Image registry | `docker.io` | +| `image.repository` | PostgreSQL Image name | `bitnami/postgresql` | +| `image.tag` | PostgreSQL Image tag | `{TAG_NAME}` | +| `image.pullPolicy` | PostgreSQL Image pull policy | `IfNotPresent` | +| `image.pullSecrets` | Specify Image pull secrets | `nil` (does not add image pull secrets to deployed pods) | +| `image.debug` | Specify if debug values should be set | `false` | +| `nameOverride` | String to partially override common.names.fullname template with a string (will prepend the release name) | `nil` | +| `fullnameOverride` | String to fully override common.names.fullname template with a string | `nil` | +| `volumePermissions.enabled` | Enable init container that changes volume permissions in the data directory (for cases where the default k8s `runAsUser` and `fsUser` values do not work) | `false` | +| `volumePermissions.image.registry` | Init container volume-permissions image registry | `docker.io` | +| `volumePermissions.image.repository` | Init container volume-permissions image name | `bitnami/bitnami-shell` | +| `volumePermissions.image.tag` | Init container volume-permissions image tag | `"10"` | +| `volumePermissions.image.pullPolicy` | Init container volume-permissions image pull policy | `Always` | +| `volumePermissions.securityContext.*` | Other container security context to be included as-is in the container spec | `{}` | +| `volumePermissions.securityContext.runAsUser` | User ID for the init container (when facing issues in OpenShift or uid unknown, try value "auto") | `0` | +| `usePasswordFile` | Have the secrets mounted as a file instead of env vars | `false` | +| `ldap.enabled` | Enable LDAP support | `false` | +| `ldap.existingSecret` | Name of existing secret to use for LDAP passwords | `nil` | +| `ldap.url` | LDAP URL beginning in the form `ldap[s]://host[:port]/basedn[?[attribute][?[scope][?[filter]]]]` | `nil` | +| `ldap.server` | IP address or name of the LDAP server. | `nil` | +| `ldap.port` | Port number on the LDAP server to connect to | `nil` | +| `ldap.scheme` | Set to `ldaps` to use LDAPS. | `nil` | +| `ldap.tls` | Set to `1` to use TLS encryption | `nil` | +| `ldap.prefix` | String to prepend to the user name when forming the DN to bind | `nil` | +| `ldap.suffix` | String to append to the user name when forming the DN to bind | `nil` | +| `ldap.search_attr` | Attribute to match against the user name in the search | `nil` | +| `ldap.search_filter` | The search filter to use when doing search+bind authentication | `nil` | +| `ldap.baseDN` | Root DN to begin the search for the user in | `nil` | +| `ldap.bindDN` | DN of user to bind to LDAP | `nil` | +| `ldap.bind_password` | Password for the user to bind to LDAP | `nil` | +| `replication.enabled` | Enable replication | `false` | +| `replication.user` | Replication user | `repl_user` | +| `replication.password` | Replication user password | `repl_password` | +| `replication.readReplicas` | Number of read replicas replicas | `1` | +| `replication.synchronousCommit` | Set synchronous commit mode. Allowed values: `on`, `remote_apply`, `remote_write`, `local` and `off` | `off` | +| `replication.numSynchronousReplicas` | Number of replicas that will have synchronous replication. Note: Cannot be greater than `replication.readReplicas`. | `0` | +| `replication.applicationName` | Cluster application name. Useful for advanced replication settings | `my_application` | +| `existingSecret` | Name of existing secret to use for PostgreSQL passwords. The secret has to contain the keys `postgresql-password` which is the password for `postgresqlUsername` when it is different of `postgres`, `postgresql-postgres-password` which will override `postgresqlPassword`, `postgresql-replication-password` which will override `replication.password` and `postgresql-ldap-password` which will be used to authenticate on LDAP. The value is evaluated as a template. | `nil` | +| `postgresqlPostgresPassword` | PostgreSQL admin password (used when `postgresqlUsername` is not `postgres`, in which case`postgres` is the admin username). | _random 10 character alphanumeric string_ | +| `postgresqlUsername` | PostgreSQL user (creates a non-admin user when `postgresqlUsername` is not `postgres`) | `postgres` | +| `postgresqlPassword` | PostgreSQL user password | _random 10 character alphanumeric string_ | +| `postgresqlDatabase` | PostgreSQL database | `nil` | +| `postgresqlDataDir` | PostgreSQL data dir folder | `/bitnami/postgresql` (same value as persistence.mountPath) | +| `extraEnv` | Any extra environment variables you would like to pass on to the pod. The value is evaluated as a template. | `[]` | +| `extraEnvVarsCM` | Name of a Config Map containing extra environment variables you would like to pass on to the pod. The value is evaluated as a template. | `nil` | +| `postgresqlInitdbArgs` | PostgreSQL initdb extra arguments | `nil` | +| `postgresqlInitdbWalDir` | PostgreSQL location for transaction log | `nil` | +| `postgresqlConfiguration` | Runtime Config Parameters | `nil` | +| `postgresqlExtendedConf` | Extended Runtime Config Parameters (appended to main or default configuration) | `nil` | +| `pgHbaConfiguration` | Content of pg_hba.conf | `nil (do not create pg_hba.conf)` | +| `postgresqlSharedPreloadLibraries` | Shared preload libraries (comma-separated list) | `pgaudit` | +| `postgresqlMaxConnections` | Maximum total connections | `nil` | +| `postgresqlPostgresConnectionLimit` | Maximum total connections for the postgres user | `nil` | +| `postgresqlDbUserConnectionLimit` | Maximum total connections for the non-admin user | `nil` | +| `postgresqlTcpKeepalivesInterval` | TCP keepalives interval | `nil` | +| `postgresqlTcpKeepalivesIdle` | TCP keepalives idle | `nil` | +| `postgresqlTcpKeepalivesCount` | TCP keepalives count | `nil` | +| `postgresqlStatementTimeout` | Statement timeout | `nil` | +| `postgresqlPghbaRemoveFilters` | Comma-separated list of patterns to remove from the pg_hba.conf file | `nil` | +| `customStartupProbe` | Override default startup probe | `nil` | +| `customLivenessProbe` | Override default liveness probe | `nil` | +| `customReadinessProbe` | Override default readiness probe | `nil` | +| `audit.logHostname` | Add client hostnames to the log file | `false` | +| `audit.logConnections` | Add client log-in operations to the log file | `false` | +| `audit.logDisconnections` | Add client log-outs operations to the log file | `false` | +| `audit.pgAuditLog` | Add operations to log using the pgAudit extension | `nil` | +| `audit.clientMinMessages` | Message log level to share with the user | `nil` | +| `audit.logLinePrefix` | Template string for the log line prefix | `nil` | +| `audit.logTimezone` | Timezone for the log timestamps | `nil` | +| `configurationConfigMap` | ConfigMap with the PostgreSQL configuration files (Note: Overrides `postgresqlConfiguration` and `pgHbaConfiguration`). The value is evaluated as a template. | `nil` | +| `extendedConfConfigMap` | ConfigMap with the extended PostgreSQL configuration files. The value is evaluated as a template. | `nil` | +| `initdbScripts` | Dictionary of initdb scripts | `nil` | +| `initdbUser` | PostgreSQL user to execute the .sql and sql.gz scripts | `nil` | +| `initdbPassword` | Password for the user specified in `initdbUser` | `nil` | +| `initdbScriptsConfigMap` | ConfigMap with the initdb scripts (Note: Overrides `initdbScripts`). The value is evaluated as a template. | `nil` | +| `initdbScriptsSecret` | Secret with initdb scripts that contain sensitive information (Note: can be used with `initdbScriptsConfigMap` or `initdbScripts`). The value is evaluated as a template. | `nil` | +| `service.type` | Kubernetes Service type | `ClusterIP` | +| `service.port` | PostgreSQL port | `5432` | +| `service.nodePort` | Kubernetes Service nodePort | `nil` | +| `service.annotations` | Annotations for PostgreSQL service | `{}` (evaluated as a template) | +| `service.loadBalancerIP` | loadBalancerIP if service type is `LoadBalancer` | `nil` | +| `service.loadBalancerSourceRanges` | Address that are allowed when svc is LoadBalancer | `[]` (evaluated as a template) | +| `schedulerName` | Name of the k8s scheduler (other than default) | `nil` | +| `shmVolume.enabled` | Enable emptyDir volume for /dev/shm for primary and read replica(s) Pod(s) | `true` | +| `shmVolume.chmod.enabled` | Run at init chmod 777 of the /dev/shm (ignored if `volumePermissions.enabled` is `false`) | `true` | +| `persistence.enabled` | Enable persistence using PVC | `true` | +| `persistence.existingClaim` | Provide an existing `PersistentVolumeClaim`, the value is evaluated as a template. | `nil` | +| `persistence.mountPath` | Path to mount the volume at | `/bitnami/postgresql` | +| `persistence.subPath` | Subdirectory of the volume to mount at | `""` | +| `persistence.storageClass` | PVC Storage Class for PostgreSQL volume | `nil` | +| `persistence.accessModes` | PVC Access Mode for PostgreSQL volume | `[ReadWriteOnce]` | +| `persistence.size` | PVC Storage Request for PostgreSQL volume | `8Gi` | +| `persistence.annotations` | Annotations for the PVC | `{}` | +| `persistence.selector` | Selector to match an existing Persistent Volume (this value is evaluated as a template) | `{}` | +| `commonAnnotations` | Annotations to be added to all deployed resources (rendered as a template) | `{}` | +| `primary.podAffinityPreset` | PostgreSQL primary pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `primary.podAntiAffinityPreset` | PostgreSQL primary pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | +| `primary.nodeAffinityPreset.type` | PostgreSQL primary node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `primary.nodeAffinityPreset.key` | PostgreSQL primary node label key to match Ignored if `primary.affinity` is set. | `""` | +| `primary.nodeAffinityPreset.values` | PostgreSQL primary node label values to match. Ignored if `primary.affinity` is set. | `[]` | +| `primary.affinity` | Affinity for PostgreSQL primary pods assignment | `{}` (evaluated as a template) | +| `primary.nodeSelector` | Node labels for PostgreSQL primary pods assignment | `{}` (evaluated as a template) | +| `primary.tolerations` | Tolerations for PostgreSQL primary pods assignment | `[]` (evaluated as a template) | +| `primary.anotations` | Map of annotations to add to the statefulset (postgresql primary) | `{}` | +| `primary.labels` | Map of labels to add to the statefulset (postgresql primary) | `{}` | +| `primary.podAnnotations` | Map of annotations to add to the pods (postgresql primary) | `{}` | +| `primary.podLabels` | Map of labels to add to the pods (postgresql primary) | `{}` | +| `primary.priorityClassName` | Priority Class to use for each pod (postgresql primary) | `nil` | +| `primary.extraInitContainers` | Additional init containers to add to the pods (postgresql primary) | `[]` | +| `primary.extraVolumeMounts` | Additional volume mounts to add to the pods (postgresql primary) | `[]` | +| `primary.extraVolumes` | Additional volumes to add to the pods (postgresql primary) | `[]` | +| `primary.sidecars` | Add additional containers to the pod | `[]` | +| `primary.service.type` | Allows using a different service type for primary | `nil` | +| `primary.service.nodePort` | Allows using a different nodePort for primary | `nil` | +| `primary.service.clusterIP` | Allows using a different clusterIP for primary | `nil` | +| `primaryAsStandBy.enabled` | Whether to enable current cluster's primary as standby server of another cluster or not. | `false` | +| `primaryAsStandBy.primaryHost` | The Host of replication primary in the other cluster. | `nil` | +| `primaryAsStandBy.primaryPort ` | The Port of replication primary in the other cluster. | `nil` | +| `readReplicas.podAffinityPreset` | PostgreSQL read only pod affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `readReplicas.podAntiAffinityPreset` | PostgreSQL read only pod anti-affinity preset. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `soft` | +| `readReplicas.nodeAffinityPreset.type` | PostgreSQL read only node affinity preset type. Ignored if `primary.affinity` is set. Allowed values: `soft` or `hard` | `""` | +| `readReplicas.nodeAffinityPreset.key` | PostgreSQL read only node label key to match Ignored if `primary.affinity` is set. | `""` | +| `readReplicas.nodeAffinityPreset.values` | PostgreSQL read only node label values to match. Ignored if `primary.affinity` is set. | `[]` | +| `readReplicas.affinity` | Affinity for PostgreSQL read only pods assignment | `{}` (evaluated as a template) | +| `readReplicas.nodeSelector` | Node labels for PostgreSQL read only pods assignment | `{}` (evaluated as a template) | +| `readReplicas.anotations` | Map of annotations to add to the statefulsets (postgresql readReplicas) | `{}` | +| `readReplicas.resources` | CPU/Memory resource requests/limits override for readReplicass. Will fallback to `values.resources` if not defined. | `{}` | +| `readReplicas.labels` | Map of labels to add to the statefulsets (postgresql readReplicas) | `{}` | +| `readReplicas.podAnnotations` | Map of annotations to add to the pods (postgresql readReplicas) | `{}` | +| `readReplicas.podLabels` | Map of labels to add to the pods (postgresql readReplicas) | `{}` | +| `readReplicas.priorityClassName` | Priority Class to use for each pod (postgresql readReplicas) | `nil` | +| `readReplicas.extraInitContainers` | Additional init containers to add to the pods (postgresql readReplicas) | `[]` | +| `readReplicas.extraVolumeMounts` | Additional volume mounts to add to the pods (postgresql readReplicas) | `[]` | +| `readReplicas.extraVolumes` | Additional volumes to add to the pods (postgresql readReplicas) | `[]` | +| `readReplicas.sidecars` | Add additional containers to the pod | `[]` | +| `readReplicas.service.type` | Allows using a different service type for readReplicas | `nil` | +| `readReplicas.service.nodePort` | Allows using a different nodePort for readReplicas | `nil` | +| `readReplicas.service.clusterIP` | Allows using a different clusterIP for readReplicas | `nil` | +| `readReplicas.persistence.enabled` | Whether to enable readReplicas replicas persistence | `true` | +| `terminationGracePeriodSeconds` | Seconds the pod needs to terminate gracefully | `nil` | +| `resources` | CPU/Memory resource requests/limits | Memory: `256Mi`, CPU: `250m` | +| `securityContext.*` | Other pod security context to be included as-is in the pod spec | `{}` | +| `securityContext.enabled` | Enable security context | `true` | +| `securityContext.fsGroup` | Group ID for the pod | `1001` | +| `containerSecurityContext.*` | Other container security context to be included as-is in the container spec | `{}` | +| `containerSecurityContext.enabled` | Enable container security context | `true` | +| `containerSecurityContext.runAsUser` | User ID for the container | `1001` | +| `serviceAccount.enabled` | Enable service account (Note: Service Account will only be automatically created if `serviceAccount.name` is not set) | `false` | +| `serviceAccount.name` | Name of existing service account | `nil` | +| `networkPolicy.enabled` | Enable NetworkPolicy | `false` | +| `networkPolicy.allowExternal` | Don't require client label for connections | `true` | +| `networkPolicy.explicitNamespacesSelector` | A Kubernetes LabelSelector to explicitly select namespaces from which ingress traffic could be allowed | `{}` | +| `startupProbe.enabled` | Enable startupProbe | `false` | +| `startupProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 30 | +| `startupProbe.periodSeconds` | How often to perform the probe | 15 | +| `startupProbe.timeoutSeconds` | When the probe times | 5 | +| `startupProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 10 | +| `startupProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed. | 1 | +| `livenessProbe.enabled` | Enable livenessProbe | `true` | +| `livenessProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 30 | +| `livenessProbe.periodSeconds` | How often to perform the probe | 10 | +| `livenessProbe.timeoutSeconds` | When the probe times out | 5 | +| `livenessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `livenessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `readinessProbe.enabled` | Enable readinessProbe | `true` | +| `readinessProbe.initialDelaySeconds` | Delay before readiness probe is initiated | 5 | +| `readinessProbe.periodSeconds` | How often to perform the probe | 10 | +| `readinessProbe.timeoutSeconds` | When the probe times out | 5 | +| `readinessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `readinessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `tls.enabled` | Enable TLS traffic support | `false` | +| `tls.preferServerCiphers` | Whether to use the server's TLS cipher preferences rather than the client's | `true` | +| `tls.certificatesSecret` | Name of an existing secret that contains the certificates | `nil` | +| `tls.certFilename` | Certificate filename | `""` | +| `tls.certKeyFilename` | Certificate key filename | `""` | +| `tls.certCAFilename` | CA Certificate filename. If provided, PostgreSQL will authenticate TLS/SSL clients by requesting them a certificate. | `nil` | +| `tls.crlFilename` | File containing a Certificate Revocation List | `nil` | +| `metrics.enabled` | Start a prometheus exporter | `false` | +| `metrics.service.type` | Kubernetes Service type | `ClusterIP` | +| `service.clusterIP` | Static clusterIP or None for headless services | `nil` | +| `metrics.service.annotations` | Additional annotations for metrics exporter pod | `{ prometheus.io/scrape: "true", prometheus.io/port: "9187"}` | +| `metrics.service.loadBalancerIP` | loadBalancerIP if redis metrics service type is `LoadBalancer` | `nil` | +| `metrics.serviceMonitor.enabled` | Set this to `true` to create ServiceMonitor for Prometheus operator | `false` | +| `metrics.serviceMonitor.additionalLabels` | Additional labels that can be used so ServiceMonitor will be discovered by Prometheus | `{}` | +| `metrics.serviceMonitor.namespace` | Optional namespace in which to create ServiceMonitor | `nil` | +| `metrics.serviceMonitor.interval` | Scrape interval. If not set, the Prometheus default scrape interval is used | `nil` | +| `metrics.serviceMonitor.scrapeTimeout` | Scrape timeout. If not set, the Prometheus default scrape timeout is used | `nil` | +| `metrics.prometheusRule.enabled` | Set this to true to create prometheusRules for Prometheus operator | `false` | +| `metrics.prometheusRule.additionalLabels` | Additional labels that can be used so prometheusRules will be discovered by Prometheus | `{}` | +| `metrics.prometheusRule.namespace` | namespace where prometheusRules resource should be created | the same namespace as postgresql | +| `metrics.prometheusRule.rules` | [rules](https://prometheus.io/docs/prometheus/latest/configuration/alerting_rules/) to be created, check values for an example. | `[]` | +| `metrics.image.registry` | PostgreSQL Exporter Image registry | `docker.io` | +| `metrics.image.repository` | PostgreSQL Exporter Image name | `bitnami/postgres-exporter` | +| `metrics.image.tag` | PostgreSQL Exporter Image tag | `{TAG_NAME}` | +| `metrics.image.pullPolicy` | PostgreSQL Exporter Image pull policy | `IfNotPresent` | +| `metrics.image.pullSecrets` | Specify Image pull secrets | `nil` (does not add image pull secrets to deployed pods) | +| `metrics.customMetrics` | Additional custom metrics | `nil` | +| `metrics.extraEnvVars` | Extra environment variables to add to exporter | `{}` (evaluated as a template) | +| `metrics.securityContext.*` | Other container security context to be included as-is in the container spec | `{}` | +| `metrics.securityContext.enabled` | Enable security context for metrics | `false` | +| `metrics.securityContext.runAsUser` | User ID for the container for metrics | `1001` | +| `metrics.livenessProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 30 | +| `metrics.livenessProbe.periodSeconds` | How often to perform the probe | 10 | +| `metrics.livenessProbe.timeoutSeconds` | When the probe times out | 5 | +| `metrics.livenessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `metrics.livenessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `metrics.readinessProbe.enabled` | would you like a readinessProbe to be enabled | `true` | +| `metrics.readinessProbe.initialDelaySeconds` | Delay before liveness probe is initiated | 5 | +| `metrics.readinessProbe.periodSeconds` | How often to perform the probe | 10 | +| `metrics.readinessProbe.timeoutSeconds` | When the probe times out | 5 | +| `metrics.readinessProbe.failureThreshold` | Minimum consecutive failures for the probe to be considered failed after having succeeded. | 6 | +| `metrics.readinessProbe.successThreshold` | Minimum consecutive successes for the probe to be considered successful after having failed | 1 | +| `updateStrategy` | Update strategy policy | `{type: "RollingUpdate"}` | +| `psp.create` | Create Pod Security Policy | `false` | +| `rbac.create` | Create Role and RoleBinding (required for PSP to work) | `false` | +| `extraDeploy` | Array of extra objects to deploy with the release (evaluated as a template). | `nil` | + +Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example, + +```console +$ helm install my-release \ + --set postgresqlPassword=secretpassword,postgresqlDatabase=my-database \ + bitnami/postgresql +``` + +The above command sets the PostgreSQL `postgres` account password to `secretpassword`. Additionally it creates a database named `my-database`. + +> NOTE: Once this chart is deployed, it is not possible to change the application's access credentials, such as usernames or passwords, using Helm. To change these application credentials after deployment, delete any persistent volumes (PVs) used by the chart and re-deploy it, or use the application's built-in administrative tools if available. + +Alternatively, a YAML file that specifies the values for the parameters can be provided while installing the chart. For example, + +```console +$ helm install my-release -f values.yaml bitnami/postgresql +``` + +> **Tip**: You can use the default [values.yaml](values.yaml) + +## Configuration and installation details + +### [Rolling VS Immutable tags](https://docs.bitnami.com/containers/how-to/understand-rolling-tags-containers/) + +It is strongly recommended to use immutable tags in a production environment. This ensures your deployment does not change automatically if the same tag is updated with a different image. + +Bitnami will release a new chart updating its containers if a new version of the main container, significant changes, or critical vulnerabilities exist. + +### Customizing primary and read replica services in a replicated configuration + +At the top level, there is a service object which defines the services for both primary and readReplicas. For deeper customization, there are service objects for both the primary and read types individually. This allows you to override the values in the top level service object so that the primary and read can be of different service types and with different clusterIPs / nodePorts. Also in the case you want the primary and read to be of type nodePort, you will need to set the nodePorts to different values to prevent a collision. The values that are deeper in the primary.service or readReplicas.service objects will take precedence over the top level service object. + +### Change PostgreSQL version + +To modify the PostgreSQL version used in this chart you can specify a [valid image tag](https://hub.docker.com/r/bitnami/postgresql/tags/) using the `image.tag` parameter. For example, `image.tag=X.Y.Z`. This approach is also applicable to other images like exporters. + +### postgresql.conf / pg_hba.conf files as configMap + +This helm chart also supports to customize the whole configuration file. + +Add your custom file to "files/postgresql.conf" in your working directory. This file will be mounted as configMap to the containers and it will be used for configuring the PostgreSQL server. + +Alternatively, you can add additional PostgreSQL configuration parameters using the `postgresqlExtendedConf` parameter as a dict, using camelCase, e.g. {"sharedBuffers": "500MB"}. Alternatively, to replace the entire default configuration use `postgresqlConfiguration`. + +In addition to these options, you can also set an external ConfigMap with all the configuration files. This is done by setting the `configurationConfigMap` parameter. Note that this will override the two previous options. + +### Allow settings to be loaded from files other than the default `postgresql.conf` + +If you don't want to provide the whole PostgreSQL configuration file and only specify certain parameters, you can add your extended `.conf` files to "files/conf.d/" in your working directory. +Those files will be mounted as configMap to the containers adding/overwriting the default configuration using the `include_dir` directive that allows settings to be loaded from files other than the default `postgresql.conf`. + +Alternatively, you can also set an external ConfigMap with all the extra configuration files. This is done by setting the `extendedConfConfigMap` parameter. Note that this will override the previous option. + +### Initialize a fresh instance + +The [Bitnami PostgreSQL](https://github.com/bitnami/bitnami-docker-postgresql) image allows you to use your custom scripts to initialize a fresh instance. In order to execute the scripts, they must be located inside the chart folder `files/docker-entrypoint-initdb.d` so they can be consumed as a ConfigMap. + +Alternatively, you can specify custom scripts using the `initdbScripts` parameter as dict. + +In addition to these options, you can also set an external ConfigMap with all the initialization scripts. This is done by setting the `initdbScriptsConfigMap` parameter. Note that this will override the two previous options. If your initialization scripts contain sensitive information such as credentials or passwords, you can use the `initdbScriptsSecret` parameter. + +The allowed extensions are `.sh`, `.sql` and `.sql.gz`. + +### Securing traffic using TLS + +TLS support can be enabled in the chart by specifying the `tls.` parameters while creating a release. The following parameters should be configured to properly enable the TLS support in the chart: + +- `tls.enabled`: Enable TLS support. Defaults to `false` +- `tls.certificatesSecret`: Name of an existing secret that contains the certificates. No defaults. +- `tls.certFilename`: Certificate filename. No defaults. +- `tls.certKeyFilename`: Certificate key filename. No defaults. + +For example: + +* First, create the secret with the cetificates files: + + ```console + kubectl create secret generic certificates-tls-secret --from-file=./cert.crt --from-file=./cert.key --from-file=./ca.crt + ``` + +* Then, use the following parameters: + + ```console + volumePermissions.enabled=true + tls.enabled=true + tls.certificatesSecret="certificates-tls-secret" + tls.certFilename="cert.crt" + tls.certKeyFilename="cert.key" + ``` + + > Note TLS and VolumePermissions: PostgreSQL requires certain permissions on sensitive files (such as certificate keys) to start up. Due to an on-going [issue](https://github.com/kubernetes/kubernetes/issues/57923) regarding kubernetes permissions and the use of `containerSecurityContext.runAsUser`, you must enable `volumePermissions` to ensure everything works as expected. + +### Sidecars + +If you need additional containers to run within the same pod as PostgreSQL (e.g. an additional metrics or logging exporter), you can do so via the `sidecars` config parameter. Simply define your container according to the Kubernetes container spec. + +```yaml +# For the PostgreSQL primary +primary: + sidecars: + - name: your-image-name + image: your-image + imagePullPolicy: Always + ports: + - name: portname + containerPort: 1234 +# For the PostgreSQL replicas +readReplicas: + sidecars: + - name: your-image-name + image: your-image + imagePullPolicy: Always + ports: + - name: portname + containerPort: 1234 +``` + +### Metrics + +The chart optionally can start a metrics exporter for [prometheus](https://prometheus.io). The metrics endpoint (port 9187) is not exposed and it is expected that the metrics are collected from inside the k8s cluster using something similar as the described in the [example Prometheus scrape configuration](https://github.com/prometheus/prometheus/blob/master/documentation/examples/prometheus-kubernetes.yml). + +The exporter allows to create custom metrics from additional SQL queries. See the Chart's `values.yaml` for an example and consult the [exporters documentation](https://github.com/wrouesnel/postgres_exporter#adding-new-metrics-via-a-config-file) for more details. + +### Use of global variables + +In more complex scenarios, we may have the following tree of dependencies + +``` + +--------------+ + | | + +------------+ Chart 1 +-----------+ + | | | | + | --------+------+ | + | | | + | | | + | | | + | | | + v v v ++-------+------+ +--------+------+ +--------+------+ +| | | | | | +| PostgreSQL | | Sub-chart 1 | | Sub-chart 2 | +| | | | | | ++--------------+ +---------------+ +---------------+ +``` + +The three charts below depend on the parent chart Chart 1. However, subcharts 1 and 2 may need to connect to PostgreSQL as well. In order to do so, subcharts 1 and 2 need to know the PostgreSQL credentials, so one option for deploying could be deploy Chart 1 with the following parameters: + +``` +postgresql.postgresqlPassword=testtest +subchart1.postgresql.postgresqlPassword=testtest +subchart2.postgresql.postgresqlPassword=testtest +postgresql.postgresqlDatabase=db1 +subchart1.postgresql.postgresqlDatabase=db1 +subchart2.postgresql.postgresqlDatabase=db1 +``` + +If the number of dependent sub-charts increases, installing the chart with parameters can become increasingly difficult. An alternative would be to set the credentials using global variables as follows: + +``` +global.postgresql.postgresqlPassword=testtest +global.postgresql.postgresqlDatabase=db1 +``` + +This way, the credentials will be available in all of the subcharts. + +## Persistence + +The [Bitnami PostgreSQL](https://github.com/bitnami/bitnami-docker-postgresql) image stores the PostgreSQL data and configurations at the `/bitnami/postgresql` path of the container. + +Persistent Volume Claims are used to keep the data across deployments. This is known to work in GCE, AWS, and minikube. +See the [Parameters](#parameters) section to configure the PVC or to disable persistence. + +If you already have data in it, you will fail to sync to standby nodes for all commits, details can refer to [code](https://github.com/bitnami/bitnami-docker-postgresql/blob/8725fe1d7d30ebe8d9a16e9175d05f7ad9260c93/9.6/debian-9/rootfs/libpostgresql.sh#L518-L556). If you need to use those data, please covert them to sql and import after `helm install` finished. + +## NetworkPolicy + +To enable network policy for PostgreSQL, install [a networking plugin that implements the Kubernetes NetworkPolicy spec](https://kubernetes.io/docs/tasks/administer-cluster/declare-network-policy#before-you-begin), and set `networkPolicy.enabled` to `true`. + +For Kubernetes v1.5 & v1.6, you must also turn on NetworkPolicy by setting the DefaultDeny namespace annotation. Note: this will enforce policy for _all_ pods in the namespace: + +```console +$ kubectl annotate namespace default "net.beta.kubernetes.io/network-policy={\"ingress\":{\"isolation\":\"DefaultDeny\"}}" +``` + +With NetworkPolicy enabled, traffic will be limited to just port 5432. + +For more precise policy, set `networkPolicy.allowExternal=false`. This will only allow pods with the generated client label to connect to PostgreSQL. +This label will be displayed in the output of a successful install. + +## Differences between Bitnami PostgreSQL image and [Docker Official](https://hub.docker.com/_/postgres) image + +- The Docker Official PostgreSQL image does not support replication. If you pass any replication environment variable, this would be ignored. The only environment variables supported by the Docker Official image are POSTGRES_USER, POSTGRES_DB, POSTGRES_PASSWORD, POSTGRES_INITDB_ARGS, POSTGRES_INITDB_WALDIR and PGDATA. All the remaining environment variables are specific to the Bitnami PostgreSQL image. +- The Bitnami PostgreSQL image is non-root by default. This requires that you run the pod with `securityContext` and updates the permissions of the volume with an `initContainer`. A key benefit of this configuration is that the pod follows security best practices and is prepared to run on Kubernetes distributions with hard security constraints like OpenShift. +- For OpenShift, one may either define the runAsUser and fsGroup accordingly, or try this more dynamic option: volumePermissions.securityContext.runAsUser="auto",securityContext.enabled=false,containerSecurityContext.enabled=false,shmVolume.chmod.enabled=false + +### Deploy chart using Docker Official PostgreSQL Image + +From chart version 4.0.0, it is possible to use this chart with the Docker Official PostgreSQL image. +Besides specifying the new Docker repository and tag, it is important to modify the PostgreSQL data directory and volume mount point. Basically, the PostgreSQL data dir cannot be the mount point directly, it has to be a subdirectory. + +``` +image.repository=postgres +image.tag=10.6 +postgresqlDataDir=/data/pgdata +persistence.mountPath=/data/ +``` + +### Setting Pod's affinity + +This chart allows you to set your custom affinity using the `XXX.affinity` paremeter(s). Find more infomation about Pod's affinity in the [kubernetes documentation](https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity). + +As an alternative, you can use of the preset configurations for pod affinity, pod anti-affinity, and node affinity available at the [bitnami/common](https://github.com/bitnami/charts/tree/master/bitnami/common#affinities) chart. To do so, set the `XXX.podAffinityPreset`, `XXX.podAntiAffinityPreset`, or `XXX.nodeAffinityPreset` parameters. + +## Troubleshooting + +Find more information about how to deal with common errors related to Bitnami’s Helm charts in [this troubleshooting guide](https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues). + +## Upgrading + +It's necessary to specify the existing passwords while performing an upgrade to ensure the secrets are not updated with invalid randomly generated passwords. Remember to specify the existing values of the `postgresqlPassword` and `replication.password` parameters when upgrading the chart: + +```bash +$ helm upgrade my-release bitnami/postgresql \ + --set postgresqlPassword=[POSTGRESQL_PASSWORD] \ + --set replication.password=[REPLICATION_PASSWORD] +``` + +> Note: you need to substitute the placeholders _[POSTGRESQL_PASSWORD]_, and _[REPLICATION_PASSWORD]_ with the values obtained from instructions in the installation notes. + +### To 10.0.0 + +[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. + +**What changes were introduced in this major version?** + +- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. +- Move dependency information from the *requirements.yaml* to the *Chart.yaml* +- After running `helm dependency update`, a *Chart.lock* file is generated containing the same structure used in the previous *requirements.lock* +- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Chart. + +**Considerations when upgrading to this version** + +- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore +- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 + +**Useful links** + +- https://docs.bitnami.com/tutorials/resolve-helm2-helm3-post-migration-issues/ +- https://helm.sh/docs/topics/v2_v3_migration/ +- https://helm.sh/blog/migrate-from-helm-v2-to-helm-v3/ + +#### Breaking changes + +- The term `master` has been replaced with `primary` and `slave` with `readReplicas` throughout the chart. Role names have changed from `master` and `slave` to `primary` and `read`. + +To upgrade to `10.0.0`, it should be done reusing the PVCs used to hold the PostgreSQL data on your previous release. To do so, follow the instructions below (the following example assumes that the release name is `postgresql`): + +> NOTE: Please, create a backup of your database before running any of those actions. + +Obtain the credentials and the names of the PVCs used to hold the PostgreSQL data on your current release: + +```console +$ export POSTGRESQL_PASSWORD=$(kubectl get secret --namespace default postgresql -o jsonpath="{.data.postgresql-password}" | base64 --decode) +$ export POSTGRESQL_PVC=$(kubectl get pvc -l app.kubernetes.io/instance=postgresql,role=master -o jsonpath="{.items[0].metadata.name}") +``` + +Delete the PostgreSQL statefulset. Notice the option `--cascade=false`: + +```console +$ kubectl delete statefulsets.apps postgresql-postgresql --cascade=false +``` + +Now the upgrade works: + +```console +$ helm upgrade postgresql bitnami/postgresql --set postgresqlPassword=$POSTGRESQL_PASSWORD --set persistence.existingClaim=$POSTGRESQL_PVC +``` + +You will have to delete the existing PostgreSQL pod and the new statefulset is going to create a new one + +```console +$ kubectl delete pod postgresql-postgresql-0 +``` + +Finally, you should see the lines below in PostgreSQL container logs: + +```console +$ kubectl logs $(kubectl get pods -l app.kubernetes.io/instance=postgresql,app.kubernetes.io/name=postgresql,role=primary -o jsonpath="{.items[0].metadata.name}") +... +postgresql 08:05:12.59 INFO ==> Deploying PostgreSQL with persisted data... +... +``` + +### To 9.0.0 + +In this version the chart was adapted to follow the Helm label best practices, see [PR 3021](https://github.com/bitnami/charts/pull/3021). That means the backward compatibility is not guarantee when upgrading the chart to this major version. + +As a workaround, you can delete the existing statefulset (using the `--cascade=false` flag pods are not deleted) before upgrade the chart. For example, this can be a valid workflow: + +- Deploy an old version (8.X.X) + +```console +$ helm install postgresql bitnami/postgresql --version 8.10.14 +``` + +- Old version is up and running + +```console +$ helm ls +NAME NAMESPACE REVISION UPDATED STATUS CHART APP VERSION +postgresql default 1 2020-08-04 13:39:54.783480286 +0000 UTC deployed postgresql-8.10.14 11.8.0 + +$ kubectl get pods +NAME READY STATUS RESTARTS AGE +postgresql-postgresql-0 1/1 Running 0 76s +``` + +- The upgrade to the latest one (9.X.X) is going to fail + +```console +$ helm upgrade postgresql bitnami/postgresql +Error: UPGRADE FAILED: cannot patch "postgresql-postgresql" with kind StatefulSet: StatefulSet.apps "postgresql-postgresql" is invalid: spec: Forbidden: updates to statefulset spec for fields other than 'replicas', 'template', and 'updateStrategy' are forbidden +``` + +- Delete the statefulset + +```console +$ kubectl delete statefulsets.apps --cascade=false postgresql-postgresql +statefulset.apps "postgresql-postgresql" deleted +``` + +- Now the upgrade works + +```console +$ helm upgrade postgresql bitnami/postgresql +$ helm ls +NAME NAMESPACE REVISION UPDATED STATUS CHART APP VERSION +postgresql default 3 2020-08-04 13:42:08.020385884 +0000 UTC deployed postgresql-9.1.2 11.8.0 +``` + +- We can kill the existing pod and the new statefulset is going to create a new one: + +```console +$ kubectl delete pod postgresql-postgresql-0 +pod "postgresql-postgresql-0" deleted + +$ kubectl get pods +NAME READY STATUS RESTARTS AGE +postgresql-postgresql-0 1/1 Running 0 19s +``` + +Please, note that without the `--cascade=false` both objects (statefulset and pod) are going to be removed and both objects will be deployed again with the `helm upgrade` command + +### To 8.0.0 + +Prefixes the port names with their protocols to comply with Istio conventions. + +If you depend on the port names in your setup, make sure to update them to reflect this change. + +### To 7.1.0 + +Adds support for LDAP configuration. + +### To 7.0.0 + +Helm performs a lookup for the object based on its group (apps), version (v1), and kind (Deployment). Also known as its GroupVersionKind, or GVK. Changing the GVK is considered a compatibility breaker from Kubernetes' point of view, so you cannot "upgrade" those objects to the new GVK in-place. Earlier versions of Helm 3 did not perform the lookup correctly which has since been fixed to match the spec. + +In https://github.com/helm/charts/pull/17281 the `apiVersion` of the statefulset resources was updated to `apps/v1` in tune with the api's deprecated, resulting in compatibility breakage. + +This major version bump signifies this change. + +### To 6.5.7 + +In this version, the chart will use PostgreSQL with the Postgis extension included. The version used with Postgresql version 10, 11 and 12 is Postgis 2.5. It has been compiled with the following dependencies: + +- protobuf +- protobuf-c +- json-c +- geos +- proj + +### To 5.0.0 + +In this version, the **chart is using PostgreSQL 11 instead of PostgreSQL 10**. You can find the main difference and notable changes in the following links: [https://www.postgresql.org/about/news/1894/](https://www.postgresql.org/about/news/1894/) and [https://www.postgresql.org/about/featurematrix/](https://www.postgresql.org/about/featurematrix/). + +For major releases of PostgreSQL, the internal data storage format is subject to change, thus complicating upgrades, you can see some errors like the following one in the logs: + +```console +Welcome to the Bitnami postgresql container +Subscribe to project updates by watching https://github.com/bitnami/bitnami-docker-postgresql +Submit issues and feature requests at https://github.com/bitnami/bitnami-docker-postgresql/issues +Send us your feedback at containers@bitnami.com + +INFO ==> ** Starting PostgreSQL setup ** +NFO ==> Validating settings in POSTGRESQL_* env vars.. +INFO ==> Initializing PostgreSQL database... +INFO ==> postgresql.conf file not detected. Generating it... +INFO ==> pg_hba.conf file not detected. Generating it... +INFO ==> Deploying PostgreSQL with persisted data... +INFO ==> Configuring replication parameters +INFO ==> Loading custom scripts... +INFO ==> Enabling remote connections +INFO ==> Stopping PostgreSQL... +INFO ==> ** PostgreSQL setup finished! ** + +INFO ==> ** Starting PostgreSQL ** + [1] FATAL: database files are incompatible with server + [1] DETAIL: The data directory was initialized by PostgreSQL version 10, which is not compatible with this version 11.3. +``` + +In this case, you should migrate the data from the old chart to the new one following an approach similar to that described in [this section](https://www.postgresql.org/docs/current/upgrading.html#UPGRADING-VIA-PGDUMPALL) from the official documentation. Basically, create a database dump in the old chart, move and restore it in the new one. + +### To 4.0.0 + +This chart will use by default the Bitnami PostgreSQL container starting from version `10.7.0-r68`. This version moves the initialization logic from node.js to bash. This new version of the chart requires setting the `POSTGRES_PASSWORD` in the slaves as well, in order to properly configure the `pg_hba.conf` file. Users from previous versions of the chart are advised to upgrade immediately. + +IMPORTANT: If you do not want to upgrade the chart version then make sure you use the `10.7.0-r68` version of the container. Otherwise, you will get this error + +``` +The POSTGRESQL_PASSWORD environment variable is empty or not set. Set the environment variable ALLOW_EMPTY_PASSWORD=yes to allow the container to be started with blank passwords. This is recommended only for development +``` + +### To 3.0.0 + +This releases make it possible to specify different nodeSelector, affinity and tolerations for master and slave pods. +It also fixes an issue with `postgresql.master.fullname` helper template not obeying fullnameOverride. + +#### Breaking changes + +- `affinty` has been renamed to `master.affinity` and `slave.affinity`. +- `tolerations` has been renamed to `master.tolerations` and `slave.tolerations`. +- `nodeSelector` has been renamed to `master.nodeSelector` and `slave.nodeSelector`. + +### To 2.0.0 + +In order to upgrade from the `0.X.X` branch to `1.X.X`, you should follow the below steps: + +- Obtain the service name (`SERVICE_NAME`) and password (`OLD_PASSWORD`) of the existing postgresql chart. You can find the instructions to obtain the password in the NOTES.txt, the service name can be obtained by running + +```console +$ kubectl get svc +``` + +- Install (not upgrade) the new version + +```console +$ helm repo update +$ helm install my-release bitnami/postgresql +``` + +- Connect to the new pod (you can obtain the name by running `kubectl get pods`): + +```console +$ kubectl exec -it NAME bash +``` + +- Once logged in, create a dump file from the previous database using `pg_dump`, for that we should connect to the previous postgresql chart: + +```console +$ pg_dump -h SERVICE_NAME -U postgres DATABASE_NAME > /tmp/backup.sql +``` + +After run above command you should be prompted for a password, this password is the previous chart password (`OLD_PASSWORD`). +This operation could take some time depending on the database size. + +- Once you have the backup file, you can restore it with a command like the one below: + +```console +$ psql -U postgres DATABASE_NAME < /tmp/backup.sql +``` + +In this case, you are accessing to the local postgresql, so the password should be the new one (you can find it in NOTES.txt). + +If you want to restore the database and the database schema does not exist, it is necessary to first follow the steps described below. + +```console +$ psql -U postgres +postgres=# drop database DATABASE_NAME; +postgres=# create database DATABASE_NAME; +postgres=# create user USER_NAME; +postgres=# alter role USER_NAME with password 'BITNAMI_USER_PASSWORD'; +postgres=# grant all privileges on database DATABASE_NAME to USER_NAME; +postgres=# alter database DATABASE_NAME owner to USER_NAME; +``` diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/.helmignore b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/.helmignore new file mode 100644 index 0000000000..50af031725 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/Chart.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/Chart.yaml new file mode 100644 index 0000000000..bcc3808d08 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/Chart.yaml @@ -0,0 +1,23 @@ +annotations: + category: Infrastructure +apiVersion: v2 +appVersion: 1.4.2 +description: A Library Helm Chart for grouping common logic between bitnami charts. + This chart is not deployable by itself. +home: https://github.com/bitnami/charts/tree/master/bitnami/common +icon: https://bitnami.com/downloads/logos/bitnami-mark.png +keywords: +- common +- helper +- template +- function +- bitnami +maintainers: +- email: containers@bitnami.com + name: Bitnami +name: common +sources: +- https://github.com/bitnami/charts +- http://www.bitnami.com/ +type: library +version: 1.4.2 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/README.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/README.md new file mode 100644 index 0000000000..7287cbb5fc --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/README.md @@ -0,0 +1,322 @@ +# Bitnami Common Library Chart + +A [Helm Library Chart](https://helm.sh/docs/topics/library_charts/#helm) for grouping common logic between bitnami charts. + +## TL;DR + +```yaml +dependencies: + - name: common + version: 0.x.x + repository: https://charts.bitnami.com/bitnami +``` + +```bash +$ helm dependency update +``` + +```yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "common.names.fullname" . }} +data: + myvalue: "Hello World" +``` + +## Introduction + +This chart provides a common template helpers which can be used to develop new charts using [Helm](https://helm.sh) package manager. + +Bitnami charts can be used with [Kubeapps](https://kubeapps.com/) for deployment and management of Helm Charts in clusters. This Helm chart has been tested on top of [Bitnami Kubernetes Production Runtime](https://kubeprod.io/) (BKPR). Deploy BKPR to get automated TLS certificates, logging and monitoring for your applications. + +## Prerequisites + +- Kubernetes 1.12+ +- Helm 3.1.0 + +## Parameters + +The following table lists the helpers available in the library which are scoped in different sections. + +### Affinities + +| Helper identifier | Description | Expected Input | +|-------------------------------|------------------------------------------------------|------------------------------------------------| +| `common.affinities.node.soft` | Return a soft nodeAffinity definition | `dict "key" "FOO" "values" (list "BAR" "BAZ")` | +| `common.affinities.node.hard` | Return a hard nodeAffinity definition | `dict "key" "FOO" "values" (list "BAR" "BAZ")` | +| `common.affinities.pod.soft` | Return a soft podAffinity/podAntiAffinity definition | `dict "component" "FOO" "context" $` | +| `common.affinities.pod.hard` | Return a hard podAffinity/podAntiAffinity definition | `dict "component" "FOO" "context" $` | + +### Capabilities + +| Helper identifier | Description | Expected Input | +|----------------------------------------------|------------------------------------------------------------------------------------------------|-------------------| +| `common.capabilities.kubeVersion` | Return the target Kubernetes version (using client default if .Values.kubeVersion is not set). | `.` Chart context | +| `common.capabilities.deployment.apiVersion` | Return the appropriate apiVersion for deployment. | `.` Chart context | +| `common.capabilities.statefulset.apiVersion` | Return the appropriate apiVersion for statefulset. | `.` Chart context | +| `common.capabilities.ingress.apiVersion` | Return the appropriate apiVersion for ingress. | `.` Chart context | +| `common.capabilities.rbac.apiVersion` | Return the appropriate apiVersion for RBAC resources. | `.` Chart context | +| `common.capabilities.crd.apiVersion` | Return the appropriate apiVersion for CRDs. | `.` Chart context | +| `common.capabilities.supportsHelmVersion` | Returns true if the used Helm version is 3.3+ | `.` Chart context | + +### Errors + +| Helper identifier | Description | Expected Input | +|-----------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------| +| `common.errors.upgrade.passwords.empty` | It will ensure required passwords are given when we are upgrading a chart. If `validationErrors` is not empty it will throw an error and will stop the upgrade action. | `dict "validationErrors" (list $validationError00 $validationError01) "context" $` | + +### Images + +| Helper identifier | Description | Expected Input | +|-----------------------------|------------------------------------------------------|---------------------------------------------------------------------------------------------------------| +| `common.images.image` | Return the proper and full image name | `dict "imageRoot" .Values.path.to.the.image "global" $`, see [ImageRoot](#imageroot) for the structure. | +| `common.images.pullSecrets` | Return the proper Docker Image Registry Secret Names | `dict "images" (list .Values.path.to.the.image1, .Values.path.to.the.image2) "global" .Values.global` | + +### Ingress + +| Helper identifier | Description | Expected Input | +|--------------------------|----------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.ingress.backend` | Generate a proper Ingress backend entry depending on the API version | `dict "serviceName" "foo" "servicePort" "bar"`, see the [Ingress deprecation notice](https://kubernetes.io/blog/2019/07/18/api-deprecations-in-1-16/) for the syntax differences | + +### Labels + +| Helper identifier | Description | Expected Input | +|-----------------------------|------------------------------------------------------|-------------------| +| `common.labels.standard` | Return Kubernetes standard labels | `.` Chart context | +| `common.labels.matchLabels` | Return the proper Docker Image Registry Secret Names | `.` Chart context | + +### Names + +| Helper identifier | Description | Expected Inpput | +|-------------------------|------------------------------------------------------------|-------------------| +| `common.names.name` | Expand the name of the chart or use `.Values.nameOverride` | `.` Chart context | +| `common.names.fullname` | Create a default fully qualified app name. | `.` Chart context | +| `common.names.chart` | Chart name plus version | `.` Chart context | + +### Secrets + +| Helper identifier | Description | Expected Input | +|---------------------------|--------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.secrets.name` | Generate the name of the secret. | `dict "existingSecret" .Values.path.to.the.existingSecret "defaultNameSuffix" "mySuffix" "context" $` see [ExistingSecret](#existingsecret) for the structure. | +| `common.secrets.key` | Generate secret key. | `dict "existingSecret" .Values.path.to.the.existingSecret "key" "keyName"` see [ExistingSecret](#existingsecret) for the structure. | +| `common.passwords.manage` | Generate secret password or retrieve one if already created. | `dict "secret" "secret-name" "key" "keyName" "providedValues" (list "path.to.password1" "path.to.password2") "length" 10 "strong" false "chartName" "chartName" "context" $`, length, strong and chartNAme fields are optional. | +| `common.secrets.exists` | Returns whether a previous generated secret already exists. | `dict "secret" "secret-name" "context" $` | + +### Storage + +| Helper identifier | Description | Expected Input | +|-------------------------------|---------------------------------------|---------------------------------------------------------------------------------------------------------------------| +| `common.affinities.node.soft` | Return a soft nodeAffinity definition | `dict "persistence" .Values.path.to.the.persistence "global" $`, see [Persistence](#persistence) for the structure. | + +### TplValues + +| Helper identifier | Description | Expected Input | +|---------------------------|----------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.tplvalues.render` | Renders a value that contains template | `dict "value" .Values.path.to.the.Value "context" $`, value is the value should rendered as template, context frequently is the chart context `$` or `.` | + +### Utils + +| Helper identifier | Description | Expected Input | +|--------------------------------|------------------------------------------------------------------------------------------|------------------------------------------------------------------------| +| `common.utils.fieldToEnvVar` | Build environment variable name given a field. | `dict "field" "my-password"` | +| `common.utils.secret.getvalue` | Print instructions to get a secret value. | `dict "secret" "secret-name" "field" "secret-value-field" "context" $` | +| `common.utils.getValueFromKey` | Gets a value from `.Values` object given its key path | `dict "key" "path.to.key" "context" $` | +| `common.utils.getKeyFromList` | Returns first `.Values` key with a defined value or first of the list if all non-defined | `dict "keys" (list "path.to.key1" "path.to.key2") "context" $` | + +### Validations + +| Helper identifier | Description | Expected Input | +|--------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| `common.validations.values.single.empty` | Validate a value must not be empty. | `dict "valueKey" "path.to.value" "secret" "secret.name" "field" "my-password" "subchart" "subchart" "context" $` secret, field and subchart are optional. In case they are given, the helper will generate a how to get instruction. See [ValidateValue](#validatevalue) | +| `common.validations.values.multiple.empty` | Validate a multiple values must not be empty. It returns a shared error for all the values. | `dict "required" (list $validateValueConf00 $validateValueConf01) "context" $`. See [ValidateValue](#validatevalue) | +| `common.validations.values.mariadb.passwords` | This helper will ensure required password for MariaDB are not empty. It returns a shared error for all the values. | `dict "secret" "mariadb-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use mariadb chart and the helper. | +| `common.validations.values.postgresql.passwords` | This helper will ensure required password for PostgreSQL are not empty. It returns a shared error for all the values. | `dict "secret" "postgresql-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use postgresql chart and the helper. | +| `common.validations.values.redis.passwords` | This helper will ensure required password for RedisTM are not empty. It returns a shared error for all the values. | `dict "secret" "redis-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use redis chart and the helper. | +| `common.validations.values.cassandra.passwords` | This helper will ensure required password for Cassandra are not empty. It returns a shared error for all the values. | `dict "secret" "cassandra-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use cassandra chart and the helper. | +| `common.validations.values.mongodb.passwords` | This helper will ensure required password for MongoDB® are not empty. It returns a shared error for all the values. | `dict "secret" "mongodb-secret" "subchart" "true" "context" $` subchart field is optional and could be true or false it depends on where you will use mongodb chart and the helper. | + +### Warnings + +| Helper identifier | Description | Expected Input | +|------------------------------|----------------------------------|------------------------------------------------------------| +| `common.warnings.rollingTag` | Warning about using rolling tag. | `ImageRoot` see [ImageRoot](#imageroot) for the structure. | + +## Special input schemas + +### ImageRoot + +```yaml +registry: + type: string + description: Docker registry where the image is located + example: docker.io + +repository: + type: string + description: Repository and image name + example: bitnami/nginx + +tag: + type: string + description: image tag + example: 1.16.1-debian-10-r63 + +pullPolicy: + type: string + description: Specify a imagePullPolicy. Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' + +pullSecrets: + type: array + items: + type: string + description: Optionally specify an array of imagePullSecrets. + +debug: + type: boolean + description: Set to true if you would like to see extra information on logs + example: false + +## An instance would be: +# registry: docker.io +# repository: bitnami/nginx +# tag: 1.16.1-debian-10-r63 +# pullPolicy: IfNotPresent +# debug: false +``` + +### Persistence + +```yaml +enabled: + type: boolean + description: Whether enable persistence. + example: true + +storageClass: + type: string + description: Ghost data Persistent Volume Storage Class, If set to "-", storageClassName: "" which disables dynamic provisioning. + example: "-" + +accessMode: + type: string + description: Access mode for the Persistent Volume Storage. + example: ReadWriteOnce + +size: + type: string + description: Size the Persistent Volume Storage. + example: 8Gi + +path: + type: string + description: Path to be persisted. + example: /bitnami + +## An instance would be: +# enabled: true +# storageClass: "-" +# accessMode: ReadWriteOnce +# size: 8Gi +# path: /bitnami +``` + +### ExistingSecret + +```yaml +name: + type: string + description: Name of the existing secret. + example: mySecret +keyMapping: + description: Mapping between the expected key name and the name of the key in the existing secret. + type: object + +## An instance would be: +# name: mySecret +# keyMapping: +# password: myPasswordKey +``` + +#### Example of use + +When we store sensitive data for a deployment in a secret, some times we want to give to users the possibility of using theirs existing secrets. + +```yaml +# templates/secret.yaml +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "common.names.fullname" . }} + labels: + app: {{ include "common.names.fullname" . }} +type: Opaque +data: + password: {{ .Values.password | b64enc | quote }} + +# templates/dpl.yaml +--- +... + env: + - name: PASSWORD + valueFrom: + secretKeyRef: + name: {{ include "common.secrets.name" (dict "existingSecret" .Values.existingSecret "context" $) }} + key: {{ include "common.secrets.key" (dict "existingSecret" .Values.existingSecret "key" "password") }} +... + +# values.yaml +--- +name: mySecret +keyMapping: + password: myPasswordKey +``` + +### ValidateValue + +#### NOTES.txt + +```console +{{- $validateValueConf00 := (dict "valueKey" "path.to.value00" "secret" "secretName" "field" "password-00") -}} +{{- $validateValueConf01 := (dict "valueKey" "path.to.value01" "secret" "secretName" "field" "password-01") -}} + +{{ include "common.validations.values.multiple.empty" (dict "required" (list $validateValueConf00 $validateValueConf01) "context" $) }} +``` + +If we force those values to be empty we will see some alerts + +```console +$ helm install test mychart --set path.to.value00="",path.to.value01="" + 'path.to.value00' must not be empty, please add '--set path.to.value00=$PASSWORD_00' to the command. To get the current value: + + export PASSWORD_00=$(kubectl get secret --namespace default secretName -o jsonpath="{.data.password-00}" | base64 --decode) + + 'path.to.value01' must not be empty, please add '--set path.to.value01=$PASSWORD_01' to the command. To get the current value: + + export PASSWORD_01=$(kubectl get secret --namespace default secretName -o jsonpath="{.data.password-01}" | base64 --decode) +``` + +## Upgrading + +### To 1.0.0 + +[On November 13, 2020, Helm v2 support was formally finished](https://github.com/helm/charts#status-of-the-project), this major version is the result of the required changes applied to the Helm Chart to be able to incorporate the different features added in Helm v3 and to be consistent with the Helm project itself regarding the Helm v2 EOL. + +**What changes were introduced in this major version?** + +- Previous versions of this Helm Chart use `apiVersion: v1` (installable by both Helm 2 and 3), this Helm Chart was updated to `apiVersion: v2` (installable by Helm 3 only). [Here](https://helm.sh/docs/topics/charts/#the-apiversion-field) you can find more information about the `apiVersion` field. +- Use `type: library`. [Here](https://v3.helm.sh/docs/faq/#library-chart-support) you can find more information. +- The different fields present in the *Chart.yaml* file has been ordered alphabetically in a homogeneous way for all the Bitnami Helm Charts + +**Considerations when upgrading to this version** + +- If you want to upgrade to this version from a previous one installed with Helm v3, you shouldn't face any issues +- If you want to upgrade to this version using Helm v2, this scenario is not supported as this version doesn't support Helm v2 anymore +- If you installed the previous version with Helm v2 and wants to upgrade to this version with Helm v3, please refer to the [official Helm documentation](https://helm.sh/docs/topics/v2_v3_migration/#migration-use-cases) about migrating from Helm v2 to v3 + +**Useful links** + +- https://docs.bitnami.com/tutorials/resolve-helm2-helm3-post-migration-issues/ +- https://helm.sh/docs/topics/v2_v3_migration/ +- https://helm.sh/blog/migrate-from-helm-v2-to-helm-v3/ diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_affinities.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_affinities.tpl new file mode 100644 index 0000000000..493a6dc7e4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_affinities.tpl @@ -0,0 +1,94 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Return a soft nodeAffinity definition +{{ include "common.affinities.nodes.soft" (dict "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.nodes.soft" -}} +preferredDuringSchedulingIgnoredDuringExecution: + - preference: + matchExpressions: + - key: {{ .key }} + operator: In + values: + {{- range .values }} + - {{ . }} + {{- end }} + weight: 1 +{{- end -}} + +{{/* +Return a hard nodeAffinity definition +{{ include "common.affinities.nodes.hard" (dict "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.nodes.hard" -}} +requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: {{ .key }} + operator: In + values: + {{- range .values }} + - {{ . }} + {{- end }} +{{- end -}} + +{{/* +Return a nodeAffinity definition +{{ include "common.affinities.nodes" (dict "type" "soft" "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.nodes" -}} + {{- if eq .type "soft" }} + {{- include "common.affinities.nodes.soft" . -}} + {{- else if eq .type "hard" }} + {{- include "common.affinities.nodes.hard" . -}} + {{- end -}} +{{- end -}} + +{{/* +Return a soft podAffinity/podAntiAffinity definition +{{ include "common.affinities.pods.soft" (dict "component" "FOO" "context" $) -}} +*/}} +{{- define "common.affinities.pods.soft" -}} +{{- $component := default "" .component -}} +preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + labelSelector: + matchLabels: {{- (include "common.labels.matchLabels" .context) | nindent 10 }} + {{- if not (empty $component) }} + {{ printf "app.kubernetes.io/component: %s" $component }} + {{- end }} + namespaces: + - {{ .context.Release.Namespace | quote }} + topologyKey: kubernetes.io/hostname + weight: 1 +{{- end -}} + +{{/* +Return a hard podAffinity/podAntiAffinity definition +{{ include "common.affinities.pods.hard" (dict "component" "FOO" "context" $) -}} +*/}} +{{- define "common.affinities.pods.hard" -}} +{{- $component := default "" .component -}} +requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchLabels: {{- (include "common.labels.matchLabels" .context) | nindent 8 }} + {{- if not (empty $component) }} + {{ printf "app.kubernetes.io/component: %s" $component }} + {{- end }} + namespaces: + - {{ .context.Release.Namespace | quote }} + topologyKey: kubernetes.io/hostname +{{- end -}} + +{{/* +Return a podAffinity/podAntiAffinity definition +{{ include "common.affinities.pods" (dict "type" "soft" "key" "FOO" "values" (list "BAR" "BAZ")) -}} +*/}} +{{- define "common.affinities.pods" -}} + {{- if eq .type "soft" }} + {{- include "common.affinities.pods.soft" . -}} + {{- else if eq .type "hard" }} + {{- include "common.affinities.pods.hard" . -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_capabilities.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_capabilities.tpl new file mode 100644 index 0000000000..4dde56a38d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_capabilities.tpl @@ -0,0 +1,95 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Return the target Kubernetes version +*/}} +{{- define "common.capabilities.kubeVersion" -}} +{{- if .Values.global }} + {{- if .Values.global.kubeVersion }} + {{- .Values.global.kubeVersion -}} + {{- else }} + {{- default .Capabilities.KubeVersion.Version .Values.kubeVersion -}} + {{- end -}} +{{- else }} +{{- default .Capabilities.KubeVersion.Version .Values.kubeVersion -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for deployment. +*/}} +{{- define "common.capabilities.deployment.apiVersion" -}} +{{- if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "extensions/v1beta1" -}} +{{- else -}} +{{- print "apps/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for statefulset. +*/}} +{{- define "common.capabilities.statefulset.apiVersion" -}} +{{- if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "apps/v1beta1" -}} +{{- else -}} +{{- print "apps/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for ingress. +*/}} +{{- define "common.capabilities.ingress.apiVersion" -}} +{{- if .Values.ingress -}} +{{- if .Values.ingress.apiVersion -}} +{{- .Values.ingress.apiVersion -}} +{{- else if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "extensions/v1beta1" -}} +{{- else if semverCompare "<1.19-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "networking.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "networking.k8s.io/v1" -}} +{{- end }} +{{- else if semverCompare "<1.14-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "extensions/v1beta1" -}} +{{- else if semverCompare "<1.19-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "networking.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "networking.k8s.io/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for RBAC resources. +*/}} +{{- define "common.capabilities.rbac.apiVersion" -}} +{{- if semverCompare "<1.17-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "rbac.authorization.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "rbac.authorization.k8s.io/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for CRDs. +*/}} +{{- define "common.capabilities.crd.apiVersion" -}} +{{- if semverCompare "<1.19-0" (include "common.capabilities.kubeVersion" .) -}} +{{- print "apiextensions.k8s.io/v1beta1" -}} +{{- else -}} +{{- print "apiextensions.k8s.io/v1" -}} +{{- end -}} +{{- end -}} + +{{/* +Returns true if the used Helm version is 3.3+. +A way to check the used Helm version was not introduced until version 3.3.0 with .Capabilities.HelmVersion, which contains an additional "{}}" structure. +This check is introduced as a regexMatch instead of {{ if .Capabilities.HelmVersion }} because checking for the key HelmVersion in <3.3 results in a "interface not found" error. +**To be removed when the catalog's minimun Helm version is 3.3** +*/}} +{{- define "common.capabilities.supportsHelmVersion" -}} +{{- if regexMatch "{(v[0-9])*[^}]*}}$" (.Capabilities | toString ) }} + {{- true -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_errors.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_errors.tpl new file mode 100644 index 0000000000..a79cc2e322 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_errors.tpl @@ -0,0 +1,23 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Through error when upgrading using empty passwords values that must not be empty. + +Usage: +{{- $validationError00 := include "common.validations.values.single.empty" (dict "valueKey" "path.to.password00" "secret" "secretName" "field" "password-00") -}} +{{- $validationError01 := include "common.validations.values.single.empty" (dict "valueKey" "path.to.password01" "secret" "secretName" "field" "password-01") -}} +{{ include "common.errors.upgrade.passwords.empty" (dict "validationErrors" (list $validationError00 $validationError01) "context" $) }} + +Required password params: + - validationErrors - String - Required. List of validation strings to be return, if it is empty it won't throw error. + - context - Context - Required. Parent context. +*/}} +{{- define "common.errors.upgrade.passwords.empty" -}} + {{- $validationErrors := join "" .validationErrors -}} + {{- if and $validationErrors .context.Release.IsUpgrade -}} + {{- $errorString := "\nPASSWORDS ERROR: You must provide your current passwords when upgrading the release." -}} + {{- $errorString = print $errorString "\n Note that even after reinstallation, old credentials may be needed as they may be kept in persistent volume claims." -}} + {{- $errorString = print $errorString "\n Further information can be obtained at https://docs.bitnami.com/general/how-to/troubleshoot-helm-chart-issues/#credential-errors-while-upgrading-chart-releases" -}} + {{- $errorString = print $errorString "\n%s" -}} + {{- printf $errorString $validationErrors | fail -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_images.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_images.tpl new file mode 100644 index 0000000000..60f04fd6e2 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_images.tpl @@ -0,0 +1,47 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Return the proper image name +{{ include "common.images.image" ( dict "imageRoot" .Values.path.to.the.image "global" $) }} +*/}} +{{- define "common.images.image" -}} +{{- $registryName := .imageRoot.registry -}} +{{- $repositoryName := .imageRoot.repository -}} +{{- $tag := .imageRoot.tag | toString -}} +{{- if .global }} + {{- if .global.imageRegistry }} + {{- $registryName = .global.imageRegistry -}} + {{- end -}} +{{- end -}} +{{- if $registryName }} +{{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} +{{- else -}} +{{- printf "%s:%s" $repositoryName $tag -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper Docker Image Registry Secret Names +{{ include "common.images.pullSecrets" ( dict "images" (list .Values.path.to.the.image1, .Values.path.to.the.image2) "global" .Values.global) }} +*/}} +{{- define "common.images.pullSecrets" -}} + {{- $pullSecrets := list }} + + {{- if .global }} + {{- range .global.imagePullSecrets -}} + {{- $pullSecrets = append $pullSecrets . -}} + {{- end -}} + {{- end -}} + + {{- range .images -}} + {{- range .pullSecrets -}} + {{- $pullSecrets = append $pullSecrets . -}} + {{- end -}} + {{- end -}} + + {{- if (not (empty $pullSecrets)) }} +imagePullSecrets: + {{- range $pullSecrets }} + - name: {{ . }} + {{- end }} + {{- end }} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_ingress.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_ingress.tpl new file mode 100644 index 0000000000..622ef50e3c --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_ingress.tpl @@ -0,0 +1,42 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Generate backend entry that is compatible with all Kubernetes API versions. + +Usage: +{{ include "common.ingress.backend" (dict "serviceName" "backendName" "servicePort" "backendPort" "context" $) }} + +Params: + - serviceName - String. Name of an existing service backend + - servicePort - String/Int. Port name (or number) of the service. It will be translated to different yaml depending if it is a string or an integer. + - context - Dict - Required. The context for the template evaluation. +*/}} +{{- define "common.ingress.backend" -}} +{{- $apiVersion := (include "common.capabilities.ingress.apiVersion" .context) -}} +{{- if or (eq $apiVersion "extensions/v1beta1") (eq $apiVersion "networking.k8s.io/v1beta1") -}} +serviceName: {{ .serviceName }} +servicePort: {{ .servicePort }} +{{- else -}} +service: + name: {{ .serviceName }} + port: + {{- if typeIs "string" .servicePort }} + name: {{ .servicePort }} + {{- else if typeIs "int" .servicePort }} + number: {{ .servicePort }} + {{- end }} +{{- end -}} +{{- end -}} + +{{/* +Print "true" if the API pathType field is supported +Usage: +{{ include "common.ingress.supportsPathType" . }} +*/}} +{{- define "common.ingress.supportsPathType" -}} +{{- if (semverCompare "<1.18-0" (include "common.capabilities.kubeVersion" .)) -}} +{{- print "false" -}} +{{- else -}} +{{- print "true" -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_labels.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_labels.tpl new file mode 100644 index 0000000000..252066c7e2 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_labels.tpl @@ -0,0 +1,18 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Kubernetes standard labels +*/}} +{{- define "common.labels.standard" -}} +app.kubernetes.io/name: {{ include "common.names.name" . }} +helm.sh/chart: {{ include "common.names.chart" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} + +{{/* +Labels to use on deploy.spec.selector.matchLabels and svc.spec.selector +*/}} +{{- define "common.labels.matchLabels" -}} +app.kubernetes.io/name: {{ include "common.names.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_names.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_names.tpl new file mode 100644 index 0000000000..adf2a74f48 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_names.tpl @@ -0,0 +1,32 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "common.names.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "common.names.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | 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 "common.names.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 -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_secrets.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_secrets.tpl new file mode 100644 index 0000000000..60b84a7019 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_secrets.tpl @@ -0,0 +1,129 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Generate secret name. + +Usage: +{{ include "common.secrets.name" (dict "existingSecret" .Values.path.to.the.existingSecret "defaultNameSuffix" "mySuffix" "context" $) }} + +Params: + - existingSecret - ExistingSecret/String - Optional. The path to the existing secrets in the values.yaml given by the user + to be used instead of the default one. Allows for it to be of type String (just the secret name) for backwards compatibility. + +info: https://github.com/bitnami/charts/tree/master/bitnami/common#existingsecret + - defaultNameSuffix - String - Optional. It is used only if we have several secrets in the same deployment. + - context - Dict - Required. The context for the template evaluation. +*/}} +{{- define "common.secrets.name" -}} +{{- $name := (include "common.names.fullname" .context) -}} + +{{- if .defaultNameSuffix -}} +{{- $name = printf "%s-%s" $name .defaultNameSuffix | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{- with .existingSecret -}} +{{- if not (typeIs "string" .) -}} +{{- with .name -}} +{{- $name = . -}} +{{- end -}} +{{- else -}} +{{- $name = . -}} +{{- end -}} +{{- end -}} + +{{- printf "%s" $name -}} +{{- end -}} + +{{/* +Generate secret key. + +Usage: +{{ include "common.secrets.key" (dict "existingSecret" .Values.path.to.the.existingSecret "key" "keyName") }} + +Params: + - existingSecret - ExistingSecret/String - Optional. The path to the existing secrets in the values.yaml given by the user + to be used instead of the default one. Allows for it to be of type String (just the secret name) for backwards compatibility. + +info: https://github.com/bitnami/charts/tree/master/bitnami/common#existingsecret + - key - String - Required. Name of the key in the secret. +*/}} +{{- define "common.secrets.key" -}} +{{- $key := .key -}} + +{{- if .existingSecret -}} + {{- if not (typeIs "string" .existingSecret) -}} + {{- if .existingSecret.keyMapping -}} + {{- $key = index .existingSecret.keyMapping $.key -}} + {{- end -}} + {{- end }} +{{- end -}} + +{{- printf "%s" $key -}} +{{- end -}} + +{{/* +Generate secret password or retrieve one if already created. + +Usage: +{{ include "common.secrets.passwords.manage" (dict "secret" "secret-name" "key" "keyName" "providedValues" (list "path.to.password1" "path.to.password2") "length" 10 "strong" false "chartName" "chartName" "context" $) }} + +Params: + - secret - String - Required - Name of the 'Secret' resource where the password is stored. + - key - String - Required - Name of the key in the secret. + - providedValues - List - Required - The path to the validating value in the values.yaml, e.g: "mysql.password". Will pick first parameter with a defined value. + - length - int - Optional - Length of the generated random password. + - strong - Boolean - Optional - Whether to add symbols to the generated random password. + - chartName - String - Optional - Name of the chart used when said chart is deployed as a subchart. + - context - Context - Required - Parent context. +*/}} +{{- define "common.secrets.passwords.manage" -}} + +{{- $password := "" }} +{{- $subchart := "" }} +{{- $chartName := default "" .chartName }} +{{- $passwordLength := default 10 .length }} +{{- $providedPasswordKey := include "common.utils.getKeyFromList" (dict "keys" .providedValues "context" $.context) }} +{{- $providedPasswordValue := include "common.utils.getValueFromKey" (dict "key" $providedPasswordKey "context" $.context) }} +{{- $secret := (lookup "v1" "Secret" $.context.Release.Namespace .secret) }} +{{- if $secret }} + {{- if index $secret.data .key }} + {{- $password = index $secret.data .key }} + {{- end -}} +{{- else if $providedPasswordValue }} + {{- $password = $providedPasswordValue | toString | b64enc | quote }} +{{- else }} + + {{- if .context.Values.enabled }} + {{- $subchart = $chartName }} + {{- end -}} + + {{- $requiredPassword := dict "valueKey" $providedPasswordKey "secret" .secret "field" .key "subchart" $subchart "context" $.context -}} + {{- $requiredPasswordError := include "common.validations.values.single.empty" $requiredPassword -}} + {{- $passwordValidationErrors := list $requiredPasswordError -}} + {{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" $passwordValidationErrors "context" $.context) -}} + + {{- if .strong }} + {{- $subStr := list (lower (randAlpha 1)) (randNumeric 1) (upper (randAlpha 1)) | join "_" }} + {{- $password = randAscii $passwordLength }} + {{- $password = regexReplaceAllLiteral "\\W" $password "@" | substr 5 $passwordLength }} + {{- $password = printf "%s%s" $subStr $password | toString | shuffle | b64enc | quote }} + {{- else }} + {{- $password = randAlphaNum $passwordLength | b64enc | quote }} + {{- end }} +{{- end -}} +{{- printf "%s" $password -}} +{{- end -}} + +{{/* +Returns whether a previous generated secret already exists + +Usage: +{{ include "common.secrets.exists" (dict "secret" "secret-name" "context" $) }} + +Params: + - secret - String - Required - Name of the 'Secret' resource where the password is stored. + - context - Context - Required - Parent context. +*/}} +{{- define "common.secrets.exists" -}} +{{- $secret := (lookup "v1" "Secret" $.context.Release.Namespace .secret) }} +{{- if $secret }} + {{- true -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_storage.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_storage.tpl new file mode 100644 index 0000000000..60e2a844f6 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_storage.tpl @@ -0,0 +1,23 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Return the proper Storage Class +{{ include "common.storage.class" ( dict "persistence" .Values.path.to.the.persistence "global" $) }} +*/}} +{{- define "common.storage.class" -}} + +{{- $storageClass := .persistence.storageClass -}} +{{- if .global -}} + {{- if .global.storageClass -}} + {{- $storageClass = .global.storageClass -}} + {{- end -}} +{{- end -}} + +{{- if $storageClass -}} + {{- if (eq "-" $storageClass) -}} + {{- printf "storageClassName: \"\"" -}} + {{- else }} + {{- printf "storageClassName: %s" $storageClass -}} + {{- end -}} +{{- end -}} + +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_tplvalues.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_tplvalues.tpl new file mode 100644 index 0000000000..2db166851b --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_tplvalues.tpl @@ -0,0 +1,13 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Renders a value that contains template. +Usage: +{{ include "common.tplvalues.render" ( dict "value" .Values.path.to.the.Value "context" $) }} +*/}} +{{- define "common.tplvalues.render" -}} + {{- if typeIs "string" .value }} + {{- tpl .value .context }} + {{- else }} + {{- tpl (.value | toYaml) .context }} + {{- end }} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_utils.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_utils.tpl new file mode 100644 index 0000000000..ea083a249f --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_utils.tpl @@ -0,0 +1,62 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Print instructions to get a secret value. +Usage: +{{ include "common.utils.secret.getvalue" (dict "secret" "secret-name" "field" "secret-value-field" "context" $) }} +*/}} +{{- define "common.utils.secret.getvalue" -}} +{{- $varname := include "common.utils.fieldToEnvVar" . -}} +export {{ $varname }}=$(kubectl get secret --namespace {{ .context.Release.Namespace | quote }} {{ .secret }} -o jsonpath="{.data.{{ .field }}}" | base64 --decode) +{{- end -}} + +{{/* +Build env var name given a field +Usage: +{{ include "common.utils.fieldToEnvVar" dict "field" "my-password" }} +*/}} +{{- define "common.utils.fieldToEnvVar" -}} + {{- $fieldNameSplit := splitList "-" .field -}} + {{- $upperCaseFieldNameSplit := list -}} + + {{- range $fieldNameSplit -}} + {{- $upperCaseFieldNameSplit = append $upperCaseFieldNameSplit ( upper . ) -}} + {{- end -}} + + {{ join "_" $upperCaseFieldNameSplit }} +{{- end -}} + +{{/* +Gets a value from .Values given +Usage: +{{ include "common.utils.getValueFromKey" (dict "key" "path.to.key" "context" $) }} +*/}} +{{- define "common.utils.getValueFromKey" -}} +{{- $splitKey := splitList "." .key -}} +{{- $value := "" -}} +{{- $latestObj := $.context.Values -}} +{{- range $splitKey -}} + {{- if not $latestObj -}} + {{- printf "please review the entire path of '%s' exists in values" $.key | fail -}} + {{- end -}} + {{- $value = ( index $latestObj . ) -}} + {{- $latestObj = $value -}} +{{- end -}} +{{- printf "%v" (default "" $value) -}} +{{- end -}} + +{{/* +Returns first .Values key with a defined value or first of the list if all non-defined +Usage: +{{ include "common.utils.getKeyFromList" (dict "keys" (list "path.to.key1" "path.to.key2") "context" $) }} +*/}} +{{- define "common.utils.getKeyFromList" -}} +{{- $key := first .keys -}} +{{- $reverseKeys := reverse .keys }} +{{- range $reverseKeys }} + {{- $value := include "common.utils.getValueFromKey" (dict "key" . "context" $.context ) }} + {{- if $value -}} + {{- $key = . }} + {{- end -}} +{{- end -}} +{{- printf "%s" $key -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_warnings.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_warnings.tpl new file mode 100644 index 0000000000..ae10fa41ee --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/_warnings.tpl @@ -0,0 +1,14 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Warning about using rolling tag. +Usage: +{{ include "common.warnings.rollingTag" .Values.path.to.the.imageRoot }} +*/}} +{{- define "common.warnings.rollingTag" -}} + +{{- if and (contains "bitnami/" .repository) (not (.tag | toString | regexFind "-r\\d+$|sha256:")) }} +WARNING: Rolling tag detected ({{ .repository }}:{{ .tag }}), please note that it is strongly recommended to avoid using rolling tags in a production environment. ++info https://docs.bitnami.com/containers/how-to/understand-rolling-tags-containers/ +{{- end }} + +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_cassandra.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_cassandra.tpl new file mode 100644 index 0000000000..8679ddffb1 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_cassandra.tpl @@ -0,0 +1,72 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate Cassandra required passwords are not empty. + +Usage: +{{ include "common.validations.values.cassandra.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where Cassandra values are stored, e.g: "cassandra-passwords-secret" + - subchart - Boolean - Optional. Whether Cassandra is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.cassandra.passwords" -}} + {{- $existingSecret := include "common.cassandra.values.existingSecret" . -}} + {{- $enabled := include "common.cassandra.values.enabled" . -}} + {{- $dbUserPrefix := include "common.cassandra.values.key.dbUser" . -}} + {{- $valueKeyPassword := printf "%s.password" $dbUserPrefix -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredPassword := dict "valueKey" $valueKeyPassword "secret" .secret "field" "cassandra-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPassword -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.cassandra.values.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether Cassandra is used as subchart or not. Default: false +*/}} +{{- define "common.cassandra.values.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.cassandra.dbUser.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.dbUser.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled cassandra. + +Usage: +{{ include "common.cassandra.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.cassandra.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.cassandra.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key dbUser + +Usage: +{{ include "common.cassandra.values.key.dbUser" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether Cassandra is used as subchart or not. Default: false +*/}} +{{- define "common.cassandra.values.key.dbUser" -}} + {{- if .subchart -}} + cassandra.dbUser + {{- else -}} + dbUser + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mariadb.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mariadb.tpl new file mode 100644 index 0000000000..bb5ed7253d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mariadb.tpl @@ -0,0 +1,103 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate MariaDB required passwords are not empty. + +Usage: +{{ include "common.validations.values.mariadb.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where MariaDB values are stored, e.g: "mysql-passwords-secret" + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.mariadb.passwords" -}} + {{- $existingSecret := include "common.mariadb.values.auth.existingSecret" . -}} + {{- $enabled := include "common.mariadb.values.enabled" . -}} + {{- $architecture := include "common.mariadb.values.architecture" . -}} + {{- $authPrefix := include "common.mariadb.values.key.auth" . -}} + {{- $valueKeyRootPassword := printf "%s.rootPassword" $authPrefix -}} + {{- $valueKeyUsername := printf "%s.username" $authPrefix -}} + {{- $valueKeyPassword := printf "%s.password" $authPrefix -}} + {{- $valueKeyReplicationPassword := printf "%s.replicationPassword" $authPrefix -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredRootPassword := dict "valueKey" $valueKeyRootPassword "secret" .secret "field" "mariadb-root-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredRootPassword -}} + + {{- $valueUsername := include "common.utils.getValueFromKey" (dict "key" $valueKeyUsername "context" .context) }} + {{- if not (empty $valueUsername) -}} + {{- $requiredPassword := dict "valueKey" $valueKeyPassword "secret" .secret "field" "mariadb-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPassword -}} + {{- end -}} + + {{- if (eq $architecture "replication") -}} + {{- $requiredReplicationPassword := dict "valueKey" $valueKeyReplicationPassword "secret" .secret "field" "mariadb-replication-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredReplicationPassword -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.mariadb.values.auth.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mariadb.values.auth.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.mariadb.auth.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.auth.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled mariadb. + +Usage: +{{ include "common.mariadb.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.mariadb.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.mariadb.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for architecture + +Usage: +{{ include "common.mariadb.values.architecture" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mariadb.values.architecture" -}} + {{- if .subchart -}} + {{- .context.Values.mariadb.architecture -}} + {{- else -}} + {{- .context.Values.architecture -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key auth + +Usage: +{{ include "common.mariadb.values.key.auth" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mariadb.values.key.auth" -}} + {{- if .subchart -}} + mariadb.auth + {{- else -}} + auth + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mongodb.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mongodb.tpl new file mode 100644 index 0000000000..7d5ecbccb4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_mongodb.tpl @@ -0,0 +1,108 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate MongoDB(R) required passwords are not empty. + +Usage: +{{ include "common.validations.values.mongodb.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where MongoDB(R) values are stored, e.g: "mongodb-passwords-secret" + - subchart - Boolean - Optional. Whether MongoDB(R) is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.mongodb.passwords" -}} + {{- $existingSecret := include "common.mongodb.values.auth.existingSecret" . -}} + {{- $enabled := include "common.mongodb.values.enabled" . -}} + {{- $authPrefix := include "common.mongodb.values.key.auth" . -}} + {{- $architecture := include "common.mongodb.values.architecture" . -}} + {{- $valueKeyRootPassword := printf "%s.rootPassword" $authPrefix -}} + {{- $valueKeyUsername := printf "%s.username" $authPrefix -}} + {{- $valueKeyDatabase := printf "%s.database" $authPrefix -}} + {{- $valueKeyPassword := printf "%s.password" $authPrefix -}} + {{- $valueKeyReplicaSetKey := printf "%s.replicaSetKey" $authPrefix -}} + {{- $valueKeyAuthEnabled := printf "%s.enabled" $authPrefix -}} + + {{- $authEnabled := include "common.utils.getValueFromKey" (dict "key" $valueKeyAuthEnabled "context" .context) -}} + + {{- if and (not $existingSecret) (eq $enabled "true") (eq $authEnabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredRootPassword := dict "valueKey" $valueKeyRootPassword "secret" .secret "field" "mongodb-root-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredRootPassword -}} + + {{- $valueUsername := include "common.utils.getValueFromKey" (dict "key" $valueKeyUsername "context" .context) }} + {{- $valueDatabase := include "common.utils.getValueFromKey" (dict "key" $valueKeyDatabase "context" .context) }} + {{- if and $valueUsername $valueDatabase -}} + {{- $requiredPassword := dict "valueKey" $valueKeyPassword "secret" .secret "field" "mongodb-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPassword -}} + {{- end -}} + + {{- if (eq $architecture "replicaset") -}} + {{- $requiredReplicaSetKey := dict "valueKey" $valueKeyReplicaSetKey "secret" .secret "field" "mongodb-replica-set-key" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredReplicaSetKey -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.mongodb.values.auth.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MongoDb is used as subchart or not. Default: false +*/}} +{{- define "common.mongodb.values.auth.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.mongodb.auth.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.auth.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled mongodb. + +Usage: +{{ include "common.mongodb.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.mongodb.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.mongodb.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key auth + +Usage: +{{ include "common.mongodb.values.key.auth" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MongoDB(R) is used as subchart or not. Default: false +*/}} +{{- define "common.mongodb.values.key.auth" -}} + {{- if .subchart -}} + mongodb.auth + {{- else -}} + auth + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for architecture + +Usage: +{{ include "common.mongodb.values.architecture" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether MariaDB is used as subchart or not. Default: false +*/}} +{{- define "common.mongodb.values.architecture" -}} + {{- if .subchart -}} + {{- .context.Values.mongodb.architecture -}} + {{- else -}} + {{- .context.Values.architecture -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_postgresql.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_postgresql.tpl new file mode 100644 index 0000000000..992bcd3908 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_postgresql.tpl @@ -0,0 +1,131 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate PostgreSQL required passwords are not empty. + +Usage: +{{ include "common.validations.values.postgresql.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where postgresql values are stored, e.g: "postgresql-passwords-secret" + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.postgresql.passwords" -}} + {{- $existingSecret := include "common.postgresql.values.existingSecret" . -}} + {{- $enabled := include "common.postgresql.values.enabled" . -}} + {{- $valueKeyPostgresqlPassword := include "common.postgresql.values.key.postgressPassword" . -}} + {{- $valueKeyPostgresqlReplicationEnabled := include "common.postgresql.values.key.replicationPassword" . -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $requiredPostgresqlPassword := dict "valueKey" $valueKeyPostgresqlPassword "secret" .secret "field" "postgresql-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPostgresqlPassword -}} + + {{- $enabledReplication := include "common.postgresql.values.enabled.replication" . -}} + {{- if (eq $enabledReplication "true") -}} + {{- $requiredPostgresqlReplicationPassword := dict "valueKey" $valueKeyPostgresqlReplicationEnabled "secret" .secret "field" "postgresql-replication-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredPostgresqlReplicationPassword -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to decide whether evaluate global values. + +Usage: +{{ include "common.postgresql.values.use.global" (dict "key" "key-of-global" "context" $) }} +Params: + - key - String - Required. Field to be evaluated within global, e.g: "existingSecret" +*/}} +{{- define "common.postgresql.values.use.global" -}} + {{- if .context.Values.global -}} + {{- if .context.Values.global.postgresql -}} + {{- index .context.Values.global.postgresql .key | quote -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.postgresql.values.existingSecret" (dict "context" $) }} +*/}} +{{- define "common.postgresql.values.existingSecret" -}} + {{- $globalValue := include "common.postgresql.values.use.global" (dict "key" "existingSecret" "context" .context) -}} + + {{- if .subchart -}} + {{- default (.context.Values.postgresql.existingSecret | quote) $globalValue -}} + {{- else -}} + {{- default (.context.Values.existingSecret | quote) $globalValue -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled postgresql. + +Usage: +{{ include "common.postgresql.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.postgresql.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.postgresql.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key postgressPassword. + +Usage: +{{ include "common.postgresql.values.key.postgressPassword" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.postgresql.values.key.postgressPassword" -}} + {{- $globalValue := include "common.postgresql.values.use.global" (dict "key" "postgresqlUsername" "context" .context) -}} + + {{- if not $globalValue -}} + {{- if .subchart -}} + postgresql.postgresqlPassword + {{- else -}} + postgresqlPassword + {{- end -}} + {{- else -}} + global.postgresql.postgresqlPassword + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled.replication. + +Usage: +{{ include "common.postgresql.values.enabled.replication" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.postgresql.values.enabled.replication" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.postgresql.replication.enabled -}} + {{- else -}} + {{- printf "%v" .context.Values.replication.enabled -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for the key replication.password. + +Usage: +{{ include "common.postgresql.values.key.replicationPassword" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether postgresql is used as subchart or not. Default: false +*/}} +{{- define "common.postgresql.values.key.replicationPassword" -}} + {{- if .subchart -}} + postgresql.replication.password + {{- else -}} + replication.password + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_redis.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_redis.tpl new file mode 100644 index 0000000000..3e2a47c039 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_redis.tpl @@ -0,0 +1,72 @@ + +{{/* vim: set filetype=mustache: */}} +{{/* +Validate Redis(TM) required passwords are not empty. + +Usage: +{{ include "common.validations.values.redis.passwords" (dict "secret" "secretName" "subchart" false "context" $) }} +Params: + - secret - String - Required. Name of the secret where redis values are stored, e.g: "redis-passwords-secret" + - subchart - Boolean - Optional. Whether redis is used as subchart or not. Default: false +*/}} +{{- define "common.validations.values.redis.passwords" -}} + {{- $existingSecret := include "common.redis.values.existingSecret" . -}} + {{- $enabled := include "common.redis.values.enabled" . -}} + {{- $valueKeyPrefix := include "common.redis.values.keys.prefix" . -}} + {{- $valueKeyRedisPassword := printf "%s%s" $valueKeyPrefix "password" -}} + {{- $valueKeyRedisUsePassword := printf "%s%s" $valueKeyPrefix "usePassword" -}} + + {{- if and (not $existingSecret) (eq $enabled "true") -}} + {{- $requiredPasswords := list -}} + + {{- $usePassword := include "common.utils.getValueFromKey" (dict "key" $valueKeyRedisUsePassword "context" .context) -}} + {{- if eq $usePassword "true" -}} + {{- $requiredRedisPassword := dict "valueKey" $valueKeyRedisPassword "secret" .secret "field" "redis-password" -}} + {{- $requiredPasswords = append $requiredPasswords $requiredRedisPassword -}} + {{- end -}} + + {{- include "common.validations.values.multiple.empty" (dict "required" $requiredPasswords "context" .context) -}} + {{- end -}} +{{- end -}} + +{{/* +Redis Auxiliary function to get the right value for existingSecret. + +Usage: +{{ include "common.redis.values.existingSecret" (dict "context" $) }} +Params: + - subchart - Boolean - Optional. Whether Redis(TM) is used as subchart or not. Default: false +*/}} +{{- define "common.redis.values.existingSecret" -}} + {{- if .subchart -}} + {{- .context.Values.redis.existingSecret | quote -}} + {{- else -}} + {{- .context.Values.existingSecret | quote -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right value for enabled redis. + +Usage: +{{ include "common.redis.values.enabled" (dict "context" $) }} +*/}} +{{- define "common.redis.values.enabled" -}} + {{- if .subchart -}} + {{- printf "%v" .context.Values.redis.enabled -}} + {{- else -}} + {{- printf "%v" (not .context.Values.enabled) -}} + {{- end -}} +{{- end -}} + +{{/* +Auxiliary function to get the right prefix path for the values + +Usage: +{{ include "common.redis.values.key.prefix" (dict "subchart" "true" "context" $) }} +Params: + - subchart - Boolean - Optional. Whether redis is used as subchart or not. Default: false +*/}} +{{- define "common.redis.values.keys.prefix" -}} + {{- if .subchart -}}redis.{{- else -}}{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_validations.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_validations.tpl new file mode 100644 index 0000000000..9a814cf40d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/templates/validations/_validations.tpl @@ -0,0 +1,46 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Validate values must not be empty. + +Usage: +{{- $validateValueConf00 := (dict "valueKey" "path.to.value" "secret" "secretName" "field" "password-00") -}} +{{- $validateValueConf01 := (dict "valueKey" "path.to.value" "secret" "secretName" "field" "password-01") -}} +{{ include "common.validations.values.empty" (dict "required" (list $validateValueConf00 $validateValueConf01) "context" $) }} + +Validate value params: + - valueKey - String - Required. The path to the validating value in the values.yaml, e.g: "mysql.password" + - secret - String - Optional. Name of the secret where the validating value is generated/stored, e.g: "mysql-passwords-secret" + - field - String - Optional. Name of the field in the secret data, e.g: "mysql-password" +*/}} +{{- define "common.validations.values.multiple.empty" -}} + {{- range .required -}} + {{- include "common.validations.values.single.empty" (dict "valueKey" .valueKey "secret" .secret "field" .field "context" $.context) -}} + {{- end -}} +{{- end -}} + +{{/* +Validate a value must not be empty. + +Usage: +{{ include "common.validations.value.empty" (dict "valueKey" "mariadb.password" "secret" "secretName" "field" "my-password" "subchart" "subchart" "context" $) }} + +Validate value params: + - valueKey - String - Required. The path to the validating value in the values.yaml, e.g: "mysql.password" + - secret - String - Optional. Name of the secret where the validating value is generated/stored, e.g: "mysql-passwords-secret" + - field - String - Optional. Name of the field in the secret data, e.g: "mysql-password" + - subchart - String - Optional - Name of the subchart that the validated password is part of. +*/}} +{{- define "common.validations.values.single.empty" -}} + {{- $value := include "common.utils.getValueFromKey" (dict "key" .valueKey "context" .context) }} + {{- $subchart := ternary "" (printf "%s." .subchart) (empty .subchart) }} + + {{- if not $value -}} + {{- $varname := "my-value" -}} + {{- $getCurrentValue := "" -}} + {{- if and .secret .field -}} + {{- $varname = include "common.utils.fieldToEnvVar" . -}} + {{- $getCurrentValue = printf " To get the current value:\n\n %s\n" (include "common.utils.secret.getvalue" .) -}} + {{- end -}} + {{- printf "\n '%s' must not be empty, please add '--set %s%s=$%s' to the command.%s" .valueKey $subchart .valueKey $varname $getCurrentValue -}} + {{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/values.yaml new file mode 100644 index 0000000000..9ecdc93f58 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/charts/common/values.yaml @@ -0,0 +1,3 @@ +## bitnami/common +## It is required by CI/CD tools and processes. +exampleValue: common-chart diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/commonAnnotations.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/commonAnnotations.yaml new file mode 100644 index 0000000000..97e18a4cc0 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/commonAnnotations.yaml @@ -0,0 +1,3 @@ +commonAnnotations: + helm.sh/hook: "\"pre-install, pre-upgrade\"" + helm.sh/hook-weight: "-1" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/default-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/default-values.yaml new file mode 100644 index 0000000000..fc2ba605ad --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/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/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/shmvolume-disabled-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/shmvolume-disabled-values.yaml new file mode 100644 index 0000000000..347d3b40a8 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/ci/shmvolume-disabled-values.yaml @@ -0,0 +1,2 @@ +shmVolume: + enabled: false diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/README.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/README.md new file mode 100644 index 0000000000..1813a2feaa --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/README.md @@ -0,0 +1 @@ +Copy here your postgresql.conf and/or pg_hba.conf files to use it as a config map. diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/conf.d/README.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/conf.d/README.md new file mode 100644 index 0000000000..184c1875d5 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/conf.d/README.md @@ -0,0 +1,4 @@ +If you don't want to provide the whole configuration file and only specify certain parameters, you can copy here your extended `.conf` files. +These files will be injected as a config maps and add/overwrite the default configuration using the `include_dir` directive that allows settings to be loaded from files other than the default `postgresql.conf`. + +More info in the [bitnami-docker-postgresql README](https://github.com/bitnami/bitnami-docker-postgresql#configuration-file). diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/docker-entrypoint-initdb.d/README.md b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/docker-entrypoint-initdb.d/README.md new file mode 100644 index 0000000000..cba38091e0 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/files/docker-entrypoint-initdb.d/README.md @@ -0,0 +1,3 @@ +You can copy here your custom `.sh`, `.sql` or `.sql.gz` file so they are executed during the first boot of the image. + +More info in the [bitnami-docker-postgresql](https://github.com/bitnami/bitnami-docker-postgresql#initializing-a-new-instance) repository. \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/NOTES.txt b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/NOTES.txt new file mode 100644 index 0000000000..4e98958c13 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/NOTES.txt @@ -0,0 +1,59 @@ +** Please be patient while the chart is being deployed ** + +PostgreSQL can be accessed via port {{ template "postgresql.port" . }} on the following DNS name from within your cluster: + + {{ template "common.names.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local - Read/Write connection +{{- if .Values.replication.enabled }} + {{ template "common.names.fullname" . }}-read.{{ .Release.Namespace }}.svc.cluster.local - Read only connection +{{- end }} + +{{- if not (eq (include "postgresql.username" .) "postgres") }} + +To get the password for "postgres" run: + + export POSTGRES_ADMIN_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "postgresql.secretName" . }} -o jsonpath="{.data.postgresql-postgres-password}" | base64 --decode) +{{- end }} + +To get the password for "{{ template "postgresql.username" . }}" run: + + export POSTGRES_PASSWORD=$(kubectl get secret --namespace {{ .Release.Namespace }} {{ template "postgresql.secretName" . }} -o jsonpath="{.data.postgresql-password}" | base64 --decode) + +To connect to your database run the following command: + + kubectl run {{ template "common.names.fullname" . }}-client --rm --tty -i --restart='Never' --namespace {{ .Release.Namespace }} --image {{ template "postgresql.image" . }} --env="PGPASSWORD=$POSTGRES_PASSWORD" {{- if and (.Values.networkPolicy.enabled) (not .Values.networkPolicy.allowExternal) }} + --labels="{{ template "common.names.fullname" . }}-client=true" {{- end }} --command -- psql --host {{ template "common.names.fullname" . }} -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} -p {{ template "postgresql.port" . }} + +{{ if and (.Values.networkPolicy.enabled) (not .Values.networkPolicy.allowExternal) }} +Note: Since NetworkPolicy is enabled, only pods with label {{ template "common.names.fullname" . }}-client=true" will be able to connect to this PostgreSQL cluster. +{{- end }} + +To connect to your database from outside the cluster execute the following commands: + +{{- if contains "NodePort" .Values.service.type }} + + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "common.names.fullname" . }}) + {{ if (include "postgresql.password" . ) }}PGPASSWORD="$POSTGRES_PASSWORD" {{ end }}psql --host $NODE_IP --port $NODE_PORT -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} + +{{- else if contains "LoadBalancer" .Values.service.type }} + + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + Watch the status with: 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "common.names.fullname" . }}' + + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "common.names.fullname" . }} --template "{{"{{ range (index .status.loadBalancer.ingress 0) }}{{.}}{{ end }}"}}") + {{ if (include "postgresql.password" . ) }}PGPASSWORD="$POSTGRES_PASSWORD" {{ end }}psql --host $SERVICE_IP --port {{ template "postgresql.port" . }} -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} + +{{- else if contains "ClusterIP" .Values.service.type }} + + kubectl port-forward --namespace {{ .Release.Namespace }} svc/{{ template "common.names.fullname" . }} {{ template "postgresql.port" . }}:{{ template "postgresql.port" . }} & + {{ if (include "postgresql.password" . ) }}PGPASSWORD="$POSTGRES_PASSWORD" {{ end }}psql --host 127.0.0.1 -U {{ .Values.postgresqlUsername }} -d {{- if .Values.postgresqlDatabase }} {{ .Values.postgresqlDatabase }}{{- else }} postgres{{- end }} -p {{ template "postgresql.port" . }} + +{{- end }} + +{{- include "postgresql.validateValues" . -}} + +{{- include "common.warnings.rollingTag" .Values.image -}} + +{{- $passwordValidationErrors := include "common.validations.values.postgresql.passwords" (dict "secret" (include "common.names.fullname" .) "context" $) -}} + +{{- include "common.errors.upgrade.passwords.empty" (dict "validationErrors" (list $passwordValidationErrors) "context" $) -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/_helpers.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/_helpers.tpl new file mode 100644 index 0000000000..1f98efe789 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/_helpers.tpl @@ -0,0 +1,337 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Expand the name of the chart. +*/}} +{{- define "postgresql.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 "postgresql.primary.fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- $fullname := default (printf "%s-%s" .Release.Name $name) .Values.fullnameOverride -}} +{{- if .Values.replication.enabled -}} +{{- printf "%s-%s" $fullname "primary" | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s" $fullname | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper PostgreSQL image name +*/}} +{{- define "postgresql.image" -}} +{{ include "common.images.image" (dict "imageRoot" .Values.image "global" .Values.global) }} +{{- end -}} + +{{/* +Return the proper PostgreSQL metrics image name +*/}} +{{- define "postgresql.metrics.image" -}} +{{ include "common.images.image" (dict "imageRoot" .Values.metrics.image "global" .Values.global) }} +{{- end -}} + +{{/* +Return the proper image name (for the init container volume-permissions image) +*/}} +{{- define "postgresql.volumePermissions.image" -}} +{{ include "common.images.image" (dict "imageRoot" .Values.volumePermissions.image "global" .Values.global) }} +{{- end -}} + +{{/* +Return the proper Docker Image Registry Secret Names +*/}} +{{- define "postgresql.imagePullSecrets" -}} +{{ include "common.images.pullSecrets" (dict "images" (list .Values.image .Values.metrics.image .Values.volumePermissions.image) "global" .Values.global) }} +{{- end -}} + +{{/* +Return PostgreSQL postgres user password +*/}} +{{- define "postgresql.postgres.password" -}} +{{- if .Values.global.postgresql.postgresqlPostgresPassword }} + {{- .Values.global.postgresql.postgresqlPostgresPassword -}} +{{- else if .Values.postgresqlPostgresPassword -}} + {{- .Values.postgresqlPostgresPassword -}} +{{- else -}} + {{- randAlphaNum 10 -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL password +*/}} +{{- define "postgresql.password" -}} +{{- if .Values.global.postgresql.postgresqlPassword }} + {{- .Values.global.postgresql.postgresqlPassword -}} +{{- else if .Values.postgresqlPassword -}} + {{- .Values.postgresqlPassword -}} +{{- else -}} + {{- randAlphaNum 10 -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL replication password +*/}} +{{- define "postgresql.replication.password" -}} +{{- if .Values.global.postgresql.replicationPassword }} + {{- .Values.global.postgresql.replicationPassword -}} +{{- else if .Values.replication.password -}} + {{- .Values.replication.password -}} +{{- else -}} + {{- randAlphaNum 10 -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL username +*/}} +{{- define "postgresql.username" -}} +{{- if .Values.global.postgresql.postgresqlUsername }} + {{- .Values.global.postgresql.postgresqlUsername -}} +{{- else -}} + {{- .Values.postgresqlUsername -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL replication username +*/}} +{{- define "postgresql.replication.username" -}} +{{- if .Values.global.postgresql.replicationUser }} + {{- .Values.global.postgresql.replicationUser -}} +{{- else -}} + {{- .Values.replication.user -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL port +*/}} +{{- define "postgresql.port" -}} +{{- if .Values.global.postgresql.servicePort }} + {{- .Values.global.postgresql.servicePort -}} +{{- else -}} + {{- .Values.service.port -}} +{{- end -}} +{{- end -}} + +{{/* +Return PostgreSQL created database +*/}} +{{- define "postgresql.database" -}} +{{- if .Values.global.postgresql.postgresqlDatabase }} + {{- .Values.global.postgresql.postgresqlDatabase -}} +{{- else if .Values.postgresqlDatabase -}} + {{- .Values.postgresqlDatabase -}} +{{- end -}} +{{- end -}} + +{{/* +Get the password secret. +*/}} +{{- define "postgresql.secretName" -}} +{{- if .Values.global.postgresql.existingSecret }} + {{- printf "%s" (tpl .Values.global.postgresql.existingSecret $) -}} +{{- else if .Values.existingSecret -}} + {{- printf "%s" (tpl .Values.existingSecret $) -}} +{{- else -}} + {{- printf "%s" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Return true if we should use an existingSecret. +*/}} +{{- define "postgresql.useExistingSecret" -}} +{{- if or .Values.global.postgresql.existingSecret .Values.existingSecret -}} + {{- true -}} +{{- end -}} +{{- end -}} + +{{/* +Return true if a secret object should be created +*/}} +{{- define "postgresql.createSecret" -}} +{{- if not (include "postgresql.useExistingSecret" .) -}} + {{- true -}} +{{- end -}} +{{- end -}} + +{{/* +Get the configuration ConfigMap name. +*/}} +{{- define "postgresql.configurationCM" -}} +{{- if .Values.configurationConfigMap -}} +{{- printf "%s" (tpl .Values.configurationConfigMap $) -}} +{{- else -}} +{{- printf "%s-configuration" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Get the extended configuration ConfigMap name. +*/}} +{{- define "postgresql.extendedConfigurationCM" -}} +{{- if .Values.extendedConfConfigMap -}} +{{- printf "%s" (tpl .Values.extendedConfConfigMap $) -}} +{{- else -}} +{{- printf "%s-extended-configuration" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Return true if a configmap should be mounted with PostgreSQL configuration +*/}} +{{- define "postgresql.mountConfigurationCM" -}} +{{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap }} + {{- true -}} +{{- end -}} +{{- end -}} + +{{/* +Get the initialization scripts ConfigMap name. +*/}} +{{- define "postgresql.initdbScriptsCM" -}} +{{- if .Values.initdbScriptsConfigMap -}} +{{- printf "%s" (tpl .Values.initdbScriptsConfigMap $) -}} +{{- else -}} +{{- printf "%s-init-scripts" (include "common.names.fullname" .) -}} +{{- end -}} +{{- end -}} + +{{/* +Get the initialization scripts Secret name. +*/}} +{{- define "postgresql.initdbScriptsSecret" -}} +{{- printf "%s" (tpl .Values.initdbScriptsSecret $) -}} +{{- end -}} + +{{/* +Get the metrics ConfigMap name. +*/}} +{{- define "postgresql.metricsCM" -}} +{{- printf "%s-metrics" (include "common.names.fullname" .) -}} +{{- end -}} + +{{/* +Get the readiness probe command +*/}} +{{- define "postgresql.readinessProbeCommand" -}} +- | +{{- if (include "postgresql.database" .) }} + exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} +{{- else }} + exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} +{{- end }} +{{- if contains "bitnami/" .Values.image.repository }} + [ -f /opt/bitnami/postgresql/tmp/.initialized ] || [ -f /bitnami/postgresql/.initialized ] +{{- end -}} +{{- end -}} + +{{/* +Compile all warnings into a single message, and call fail. +*/}} +{{- define "postgresql.validateValues" -}} +{{- $messages := list -}} +{{- $messages := append $messages (include "postgresql.validateValues.ldapConfigurationMethod" .) -}} +{{- $messages := append $messages (include "postgresql.validateValues.psp" .) -}} +{{- $messages := append $messages (include "postgresql.validateValues.tls" .) -}} +{{- $messages := without $messages "" -}} +{{- $message := join "\n" $messages -}} + +{{- if $message -}} +{{- printf "\nVALUES VALIDATION:\n%s" $message | fail -}} +{{- end -}} +{{- end -}} + +{{/* +Validate values of Postgresql - If ldap.url is used then you don't need the other settings for ldap +*/}} +{{- define "postgresql.validateValues.ldapConfigurationMethod" -}} +{{- if and .Values.ldap.enabled (and (not (empty .Values.ldap.url)) (not (empty .Values.ldap.server))) }} +postgresql: ldap.url, ldap.server + You cannot set both `ldap.url` and `ldap.server` at the same time. + Please provide a unique way to configure LDAP. + More info at https://www.postgresql.org/docs/current/auth-ldap.html +{{- end -}} +{{- end -}} + +{{/* +Validate values of Postgresql - If PSP is enabled RBAC should be enabled too +*/}} +{{- define "postgresql.validateValues.psp" -}} +{{- if and .Values.psp.create (not .Values.rbac.create) }} +postgresql: psp.create, rbac.create + RBAC should be enabled if PSP is enabled in order for PSP to work. + More info at https://kubernetes.io/docs/concepts/policy/pod-security-policy/#authorizing-policies +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for podsecuritypolicy. +*/}} +{{- define "podsecuritypolicy.apiVersion" -}} +{{- if semverCompare "<1.10-0" .Capabilities.KubeVersion.GitVersion -}} +{{- print "extensions/v1beta1" -}} +{{- else -}} +{{- print "policy/v1beta1" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the appropriate apiVersion for networkpolicy. +*/}} +{{- define "postgresql.networkPolicy.apiVersion" -}} +{{- if semverCompare ">=1.4-0, <1.7-0" .Capabilities.KubeVersion.GitVersion -}} +"extensions/v1beta1" +{{- else if semverCompare "^1.7-0" .Capabilities.KubeVersion.GitVersion -}} +"networking.k8s.io/v1" +{{- end -}} +{{- end -}} + +{{/* +Validate values of Postgresql TLS - When TLS is enabled, so must be VolumePermissions +*/}} +{{- define "postgresql.validateValues.tls" -}} +{{- if and .Values.tls.enabled (not .Values.volumePermissions.enabled) }} +postgresql: tls.enabled, volumePermissions.enabled + When TLS is enabled you must enable volumePermissions as well to ensure certificates files have + the right permissions. +{{- end -}} +{{- end -}} + +{{/* +Return the path to the cert file. +*/}} +{{- define "postgresql.tlsCert" -}} +{{- required "Certificate filename is required when TLS in enabled" .Values.tls.certFilename | printf "/opt/bitnami/postgresql/certs/%s" -}} +{{- end -}} + +{{/* +Return the path to the cert key file. +*/}} +{{- define "postgresql.tlsCertKey" -}} +{{- required "Certificate Key filename is required when TLS in enabled" .Values.tls.certKeyFilename | printf "/opt/bitnami/postgresql/certs/%s" -}} +{{- end -}} + +{{/* +Return the path to the CA cert file. +*/}} +{{- define "postgresql.tlsCACert" -}} +{{- printf "/opt/bitnami/postgresql/certs/%s" .Values.tls.certCAFilename -}} +{{- end -}} + +{{/* +Return the path to the CRL file. +*/}} +{{- define "postgresql.tlsCRL" -}} +{{- if .Values.tls.crlFilename -}} +{{- printf "/opt/bitnami/postgresql/certs/%s" .Values.tls.crlFilename -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/configmap.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/configmap.yaml new file mode 100644 index 0000000000..3a5ea18ae9 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/configmap.yaml @@ -0,0 +1,31 @@ +{{ if and (or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration) (not .Values.configurationConfigMap) }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "common.names.fullname" . }}-configuration + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +data: +{{- if (.Files.Glob "files/postgresql.conf") }} +{{ (.Files.Glob "files/postgresql.conf").AsConfig | indent 2 }} +{{- else if .Values.postgresqlConfiguration }} + postgresql.conf: | +{{- range $key, $value := default dict .Values.postgresqlConfiguration }} + {{- if kindIs "string" $value }} + {{ $key | snakecase }} = '{{ $value }}' + {{- else }} + {{ $key | snakecase }} = {{ $value }} + {{- end }} +{{- end }} +{{- end }} +{{- if (.Files.Glob "files/pg_hba.conf") }} +{{ (.Files.Glob "files/pg_hba.conf").AsConfig | indent 2 }} +{{- else if .Values.pgHbaConfiguration }} + pg_hba.conf: | +{{ .Values.pgHbaConfiguration | indent 4 }} +{{- end }} +{{ end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extended-config-configmap.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extended-config-configmap.yaml new file mode 100644 index 0000000000..b0dad253b5 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extended-config-configmap.yaml @@ -0,0 +1,26 @@ +{{- if and (or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf) (not .Values.extendedConfConfigMap)}} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "common.names.fullname" . }}-extended-configuration + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +data: +{{- with .Files.Glob "files/conf.d/*.conf" }} +{{ .AsConfig | indent 2 }} +{{- end }} +{{ with .Values.postgresqlExtendedConf }} + override.conf: | +{{- range $key, $value := . }} + {{- if kindIs "string" $value }} + {{ $key | snakecase }} = '{{ $value }}' + {{- else }} + {{ $key | snakecase }} = {{ $value }} + {{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extra-list.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extra-list.yaml new file mode 100644 index 0000000000..9ac65f9e16 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/extra-list.yaml @@ -0,0 +1,4 @@ +{{- range .Values.extraDeploy }} +--- +{{ include "common.tplvalues.render" (dict "value" . "context" $) }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/initialization-configmap.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/initialization-configmap.yaml new file mode 100644 index 0000000000..7796c67a93 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/initialization-configmap.yaml @@ -0,0 +1,25 @@ +{{- if and (or (.Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql,sql.gz}") .Values.initdbScripts) (not .Values.initdbScriptsConfigMap) }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "common.names.fullname" . }}-init-scripts + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +{{- with .Files.Glob "files/docker-entrypoint-initdb.d/*.sql.gz" }} +binaryData: +{{- range $path, $bytes := . }} + {{ base $path }}: {{ $.Files.Get $path | b64enc | quote }} +{{- end }} +{{- end }} +data: +{{- with .Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql}" }} +{{ .AsConfig | indent 2 }} +{{- end }} +{{- with .Values.initdbScripts }} +{{ toYaml . | indent 2 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-configmap.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-configmap.yaml new file mode 100644 index 0000000000..fa539582bb --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-configmap.yaml @@ -0,0 +1,14 @@ +{{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "postgresql.metricsCM" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +data: + custom-metrics.yaml: {{ toYaml .Values.metrics.customMetrics | quote }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-svc.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-svc.yaml new file mode 100644 index 0000000000..af8b67e2ff --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/metrics-svc.yaml @@ -0,0 +1,26 @@ +{{- if .Values.metrics.enabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }}-metrics + labels: + {{- include "common.labels.standard" . | nindent 4 }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- toYaml .Values.metrics.service.annotations | nindent 4 }} + namespace: {{ .Release.Namespace }} +spec: + type: {{ .Values.metrics.service.type }} + {{- if and (eq .Values.metrics.service.type "LoadBalancer") .Values.metrics.service.loadBalancerIP }} + loadBalancerIP: {{ .Values.metrics.service.loadBalancerIP }} + {{- end }} + ports: + - name: http-metrics + port: 9187 + targetPort: http-metrics + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} + role: primary +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/networkpolicy.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/networkpolicy.yaml new file mode 100644 index 0000000000..4f2740ea0c --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/networkpolicy.yaml @@ -0,0 +1,39 @@ +{{- if .Values.networkPolicy.enabled }} +kind: NetworkPolicy +apiVersion: {{ template "postgresql.networkPolicy.apiVersion" . }} +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + podSelector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} + ingress: + # Allow inbound connections + - ports: + - port: {{ template "postgresql.port" . }} + {{- if not .Values.networkPolicy.allowExternal }} + from: + - podSelector: + matchLabels: + {{ template "common.names.fullname" . }}-client: "true" + {{- if .Values.networkPolicy.explicitNamespacesSelector }} + namespaceSelector: +{{ toYaml .Values.networkPolicy.explicitNamespacesSelector | indent 12 }} + {{- end }} + - podSelector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 14 }} + role: read + {{- end }} + {{- if .Values.metrics.enabled }} + # Allow prometheus scrapes + - ports: + - port: 9187 + {{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/podsecuritypolicy.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..0c49694fad --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/podsecuritypolicy.yaml @@ -0,0 +1,38 @@ +{{- if .Values.psp.create }} +apiVersion: {{ include "podsecuritypolicy.apiVersion" . }} +kind: PodSecurityPolicy +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + privileged: false + volumes: + - 'configMap' + - 'secret' + - 'persistentVolumeClaim' + - 'emptyDir' + - 'projected' + hostNetwork: false + hostIPC: false + hostPID: false + runAsUser: + rule: 'RunAsAny' + seLinux: + rule: 'RunAsAny' + supplementalGroups: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + - min: 1 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/prometheusrule.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/prometheusrule.yaml new file mode 100644 index 0000000000..d0f408c78f --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/prometheusrule.yaml @@ -0,0 +1,23 @@ +{{- if and .Values.metrics.enabled .Values.metrics.prometheusRule.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: PrometheusRule +metadata: + name: {{ template "common.names.fullname" . }} +{{- with .Values.metrics.prometheusRule.namespace }} + namespace: {{ . }} +{{- end }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- with .Values.metrics.prometheusRule.additionalLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} +spec: +{{- with .Values.metrics.prometheusRule.rules }} + groups: + - name: {{ template "postgresql.name" $ }} + rules: {{ tpl (toYaml .) $ | nindent 8 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/role.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/role.yaml new file mode 100644 index 0000000000..017a5716b1 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/role.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create }} +kind: Role +apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +rules: + {{- if .Values.psp.create }} + - apiGroups: ["extensions"] + resources: ["podsecuritypolicies"] + verbs: ["use"] + resourceNames: + - {{ template "common.names.fullname" . }} + {{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/rolebinding.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/rolebinding.yaml new file mode 100644 index 0000000000..189775a15a --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create }} +kind: RoleBinding +apiVersion: {{ include "common.capabilities.rbac.apiVersion" . }} +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +roleRef: + kind: Role + name: {{ template "common.names.fullname" . }} + apiGroup: rbac.authorization.k8s.io +subjects: + - kind: ServiceAccount + name: {{ default (include "common.names.fullname" . ) .Values.serviceAccount.name }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/secrets.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/secrets.yaml new file mode 100644 index 0000000000..d492cd593b --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/secrets.yaml @@ -0,0 +1,24 @@ +{{- if (include "postgresql.createSecret" .) }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +type: Opaque +data: + {{- if not (eq (include "postgresql.username" .) "postgres") }} + postgresql-postgres-password: {{ include "postgresql.postgres.password" . | b64enc | quote }} + {{- end }} + postgresql-password: {{ include "postgresql.password" . | b64enc | quote }} + {{- if .Values.replication.enabled }} + postgresql-replication-password: {{ include "postgresql.replication.password" . | b64enc | quote }} + {{- end }} + {{- if (and .Values.ldap.enabled .Values.ldap.bind_password)}} + postgresql-ldap-password: {{ .Values.ldap.bind_password | b64enc | quote }} + {{- end }} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/serviceaccount.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/serviceaccount.yaml new file mode 100644 index 0000000000..03f0f50e7d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/serviceaccount.yaml @@ -0,0 +1,12 @@ +{{- if and (.Values.serviceAccount.enabled) (not .Values.serviceAccount.name) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "common.labels.standard" . | nindent 4 }} + name: {{ template "common.names.fullname" . }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/servicemonitor.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/servicemonitor.yaml new file mode 100644 index 0000000000..587ce85b87 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/servicemonitor.yaml @@ -0,0 +1,33 @@ +{{- if and .Values.metrics.enabled .Values.metrics.serviceMonitor.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "common.names.fullname" . }} + {{- if .Values.metrics.serviceMonitor.namespace }} + namespace: {{ .Values.metrics.serviceMonitor.namespace }} + {{- end }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.metrics.serviceMonitor.additionalLabels }} + {{- toYaml .Values.metrics.serviceMonitor.additionalLabels | nindent 4 }} + {{- end }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + +spec: + endpoints: + - port: http-metrics + {{- if .Values.metrics.serviceMonitor.interval }} + interval: {{ .Values.metrics.serviceMonitor.interval }} + {{- end }} + {{- if .Values.metrics.serviceMonitor.scrapeTimeout }} + scrapeTimeout: {{ .Values.metrics.serviceMonitor.scrapeTimeout }} + {{- end }} + namespaceSelector: + matchNames: + - {{ .Release.Namespace }} + selector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset-readreplicas.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset-readreplicas.yaml new file mode 100644 index 0000000000..b038299bf6 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset-readreplicas.yaml @@ -0,0 +1,411 @@ +{{- if .Values.replication.enabled }} +{{- $readReplicasResources := coalesce .Values.readReplicas.resources .Values.resources -}} +apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} +kind: StatefulSet +metadata: + name: "{{ template "common.names.fullname" . }}-read" + labels: {{- include "common.labels.standard" . | nindent 4 }} + app.kubernetes.io/component: read +{{- with .Values.readReplicas.labels }} +{{ toYaml . | indent 4 }} +{{- end }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- with .Values.readReplicas.annotations }} + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + serviceName: {{ template "common.names.fullname" . }}-headless + replicas: {{ .Values.replication.readReplicas }} + selector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} + role: read + template: + metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 8 }} + app.kubernetes.io/component: read + role: read +{{- with .Values.readReplicas.podLabels }} +{{ toYaml . | indent 8 }} +{{- end }} +{{- with .Values.readReplicas.podAnnotations }} + annotations: +{{ toYaml . | indent 8 }} +{{- end }} + spec: + {{- if .Values.schedulerName }} + schedulerName: "{{ .Values.schedulerName }}" + {{- end }} +{{- include "postgresql.imagePullSecrets" . | indent 6 }} + {{- if .Values.readReplicas.affinity }} + affinity: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.affinity "context" $) | nindent 8 }} + {{- else }} + affinity: + podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.readReplicas.podAffinityPreset "component" "read" "context" $) | nindent 10 }} + podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.readReplicas.podAntiAffinityPreset "component" "read" "context" $) | nindent 10 }} + nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.readReplicas.nodeAffinityPreset.type "key" .Values.readReplicas.nodeAffinityPreset.key "values" .Values.readReplicas.nodeAffinityPreset.values) | nindent 10 }} + {{- end }} + {{- if .Values.readReplicas.nodeSelector }} + nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.nodeSelector "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.readReplicas.tolerations }} + tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.readReplicas.tolerations "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.securityContext.enabled }} + securityContext: {{- omit .Values.securityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.serviceAccount.enabled }} + serviceAccountName: {{ default (include "common.names.fullname" . ) .Values.serviceAccount.name}} + {{- end }} + {{- if or .Values.readReplicas.extraInitContainers (and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled))) }} + initContainers: + {{- if and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled) .Values.tls.enabled) }} + - name: init-chmod-data + image: {{ template "postgresql.volumePermissions.image" . }} + imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} + {{- if .Values.resources }} + resources: {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + command: + - /bin/sh + - -cx + - | + {{- if .Values.persistence.enabled }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown `id -u`:`id -G | cut -d " " -f2` {{ .Values.persistence.mountPath }} + {{- else }} + chown {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} {{ .Values.persistence.mountPath }} + {{- end }} + mkdir -p {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + chmod 700 {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + find {{ .Values.persistence.mountPath }} -mindepth 1 -maxdepth 1 {{- if not (include "postgresql.mountConfigurationCM" .) }} -not -name "conf" {{- end }} -not -name ".snapshot" -not -name "lost+found" | \ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + xargs chown -R `id -u`:`id -G | cut -d " " -f2` + {{- else }} + xargs chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} + {{- end }} + {{- end }} + {{- if and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled }} + chmod -R 777 /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + cp /tmp/certs/* /opt/bitnami/postgresql/certs/ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown -R `id -u`:`id -G | cut -d " " -f2` /opt/bitnami/postgresql/certs/ + {{- else }} + chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} /opt/bitnami/postgresql/certs/ + {{- end }} + chmod 600 {{ template "postgresql.tlsCertKey" . }} + {{- end }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + securityContext: {{- omit .Values.volumePermissions.securityContext "runAsUser" | toYaml | nindent 12 }} + {{- else }} + securityContext: {{- .Values.volumePermissions.securityContext | toYaml | nindent 12 }} + {{- end }} + volumeMounts: + {{ if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + mountPath: /tmp/certs + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + {{- end }} + {{- end }} + {{- if .Values.readReplicas.extraInitContainers }} + {{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.extraInitContainers "context" $ ) | nindent 8 }} + {{- end }} + {{- end }} + {{- if .Values.readReplicas.priorityClassName }} + priorityClassName: {{ .Values.readReplicas.priorityClassName }} + {{- end }} + containers: + - name: {{ template "common.names.fullname" . }} + image: {{ template "postgresql.image" . }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + {{- if $readReplicasResources }} + resources: {{- toYaml $readReplicasResources | nindent 12 }} + {{- end }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 12 }} + {{- end }} + env: + - name: BITNAMI_DEBUG + value: {{ ternary "true" "false" .Values.image.debug | quote }} + - name: POSTGRESQL_VOLUME_DIR + value: "{{ .Values.persistence.mountPath }}" + - name: POSTGRESQL_PORT_NUMBER + value: "{{ template "postgresql.port" . }}" + {{- if .Values.persistence.mountPath }} + - name: PGDATA + value: {{ .Values.postgresqlDataDir | quote }} + {{- end }} + - name: POSTGRES_REPLICATION_MODE + value: "slave" + - name: POSTGRES_REPLICATION_USER + value: {{ include "postgresql.replication.username" . | quote }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_REPLICATION_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-replication-password" + {{- else }} + - name: POSTGRES_REPLICATION_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-replication-password + {{- end }} + - name: POSTGRES_CLUSTER_APP_NAME + value: {{ .Values.replication.applicationName }} + - name: POSTGRES_MASTER_HOST + value: {{ template "common.names.fullname" . }} + - name: POSTGRES_MASTER_PORT_NUMBER + value: {{ include "postgresql.port" . | quote }} + {{- if not (eq (include "postgresql.username" .) "postgres") }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-postgres-password" + {{- else }} + - name: POSTGRES_POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-postgres-password + {{- end }} + {{- end }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-password" + {{- else }} + - name: POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-password + {{- end }} + - name: POSTGRESQL_ENABLE_TLS + value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} + {{- if .Values.tls.enabled }} + - name: POSTGRESQL_TLS_PREFER_SERVER_CIPHERS + value: {{ ternary "yes" "no" .Values.tls.preferServerCiphers | quote }} + - name: POSTGRESQL_TLS_CERT_FILE + value: {{ template "postgresql.tlsCert" . }} + - name: POSTGRESQL_TLS_KEY_FILE + value: {{ template "postgresql.tlsCertKey" . }} + {{- if .Values.tls.certCAFilename }} + - name: POSTGRESQL_TLS_CA_FILE + value: {{ template "postgresql.tlsCACert" . }} + {{- end }} + {{- if .Values.tls.crlFilename }} + - name: POSTGRESQL_TLS_CRL_FILE + value: {{ template "postgresql.tlsCRL" . }} + {{- end }} + {{- end }} + - name: POSTGRESQL_LOG_HOSTNAME + value: {{ .Values.audit.logHostname | quote }} + - name: POSTGRESQL_LOG_CONNECTIONS + value: {{ .Values.audit.logConnections | quote }} + - name: POSTGRESQL_LOG_DISCONNECTIONS + value: {{ .Values.audit.logDisconnections | quote }} + {{- if .Values.audit.logLinePrefix }} + - name: POSTGRESQL_LOG_LINE_PREFIX + value: {{ .Values.audit.logLinePrefix | quote }} + {{- end }} + {{- if .Values.audit.logTimezone }} + - name: POSTGRESQL_LOG_TIMEZONE + value: {{ .Values.audit.logTimezone | quote }} + {{- end }} + {{- if .Values.audit.pgAuditLog }} + - name: POSTGRESQL_PGAUDIT_LOG + value: {{ .Values.audit.pgAuditLog | quote }} + {{- end }} + - name: POSTGRESQL_PGAUDIT_LOG_CATALOG + value: {{ .Values.audit.pgAuditLogCatalog | quote }} + - name: POSTGRESQL_CLIENT_MIN_MESSAGES + value: {{ .Values.audit.clientMinMessages | quote }} + - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES + value: {{ .Values.postgresqlSharedPreloadLibraries | quote }} + {{- if .Values.postgresqlMaxConnections }} + - name: POSTGRESQL_MAX_CONNECTIONS + value: {{ .Values.postgresqlMaxConnections | quote }} + {{- end }} + {{- if .Values.postgresqlPostgresConnectionLimit }} + - name: POSTGRESQL_POSTGRES_CONNECTION_LIMIT + value: {{ .Values.postgresqlPostgresConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlDbUserConnectionLimit }} + - name: POSTGRESQL_USERNAME_CONNECTION_LIMIT + value: {{ .Values.postgresqlDbUserConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesInterval }} + - name: POSTGRESQL_TCP_KEEPALIVES_INTERVAL + value: {{ .Values.postgresqlTcpKeepalivesInterval | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesIdle }} + - name: POSTGRESQL_TCP_KEEPALIVES_IDLE + value: {{ .Values.postgresqlTcpKeepalivesIdle | quote }} + {{- end }} + {{- if .Values.postgresqlStatementTimeout }} + - name: POSTGRESQL_STATEMENT_TIMEOUT + value: {{ .Values.postgresqlStatementTimeout | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesCount }} + - name: POSTGRESQL_TCP_KEEPALIVES_COUNT + value: {{ .Values.postgresqlTcpKeepalivesCount | quote }} + {{- end }} + {{- if .Values.postgresqlPghbaRemoveFilters }} + - name: POSTGRESQL_PGHBA_REMOVE_FILTERS + value: {{ .Values.postgresqlPghbaRemoveFilters | quote }} + {{- end }} + ports: + - name: tcp-postgresql + containerPort: {{ template "postgresql.port" . }} + {{- if .Values.livenessProbe.enabled }} + livenessProbe: + exec: + command: + - /bin/sh + - -c + {{- if (include "postgresql.database" .) }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- else }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- end }} + initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.livenessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} + successThreshold: {{ .Values.livenessProbe.successThreshold }} + failureThreshold: {{ .Values.livenessProbe.failureThreshold }} + {{- else if .Values.customLivenessProbe }} + livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customLivenessProbe "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.readinessProbe.enabled }} + readinessProbe: + exec: + command: + - /bin/sh + - -c + - -e + {{- include "postgresql.readinessProbeCommand" . | nindent 16 }} + initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.readinessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} + successThreshold: {{ .Values.readinessProbe.successThreshold }} + failureThreshold: {{ .Values.readinessProbe.failureThreshold }} + {{- else if .Values.customReadinessProbe }} + readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customReadinessProbe "context" $) | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.usePasswordFile }} + - name: postgresql-password + mountPath: /opt/bitnami/postgresql/secrets/ + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{ end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + mountPath: /bitnami/postgresql/conf/conf.d/ + {{- end }} + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap }} + - name: postgresql-config + mountPath: /bitnami/postgresql/conf + {{- end }} + {{- if .Values.tls.enabled }} + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + readOnly: true + {{- end }} + {{- if .Values.readReplicas.extraVolumeMounts }} + {{- toYaml .Values.readReplicas.extraVolumeMounts | nindent 12 }} + {{- end }} +{{- if .Values.readReplicas.sidecars }} +{{- include "common.tplvalues.render" ( dict "value" .Values.readReplicas.sidecars "context" $ ) | nindent 8 }} +{{- end }} + volumes: + {{- if .Values.usePasswordFile }} + - name: postgresql-password + secret: + secretName: {{ template "postgresql.secretName" . }} + {{- end }} + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap}} + - name: postgresql-config + configMap: + name: {{ template "postgresql.configurationCM" . }} + {{- end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + configMap: + name: {{ template "postgresql.extendedConfigurationCM" . }} + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + secret: + secretName: {{ required "A secret containing TLS certificates is required when TLS is enabled" .Values.tls.certificatesSecret }} + - name: postgresql-certificates + emptyDir: {} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + emptyDir: + medium: Memory + sizeLimit: 1Gi + {{- end }} + {{- if or (not .Values.persistence.enabled) (not .Values.readReplicas.persistence.enabled) }} + - name: data + emptyDir: {} + {{- end }} + {{- if .Values.readReplicas.extraVolumes }} + {{- toYaml .Values.readReplicas.extraVolumes | nindent 8 }} + {{- end }} + updateStrategy: + type: {{ .Values.updateStrategy.type }} + {{- if (eq "Recreate" .Values.updateStrategy.type) }} + rollingUpdate: null + {{- end }} +{{- if and .Values.persistence.enabled .Values.readReplicas.persistence.enabled }} + volumeClaimTemplates: + - metadata: + name: data + {{- with .Values.persistence.annotations }} + annotations: + {{- range $key, $value := . }} + {{ $key }}: {{ $value }} + {{- end }} + {{- end }} + spec: + accessModes: + {{- range .Values.persistence.accessModes }} + - {{ . | quote }} + {{- end }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + {{ include "common.storage.class" (dict "persistence" .Values.persistence "global" .Values.global) }} + + {{- if .Values.persistence.selector }} + selector: {{- include "common.tplvalues.render" (dict "value" .Values.persistence.selector "context" $) | nindent 10 }} + {{- end -}} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset.yaml new file mode 100644 index 0000000000..f8163fd99f --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/statefulset.yaml @@ -0,0 +1,609 @@ +apiVersion: {{ include "common.capabilities.statefulset.apiVersion" . }} +kind: StatefulSet +metadata: + name: {{ template "postgresql.primary.fullname" . }} + labels: {{- include "common.labels.standard" . | nindent 4 }} + app.kubernetes.io/component: primary + {{- with .Values.primary.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- with .Values.primary.annotations }} + {{- toYaml . | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + serviceName: {{ template "common.names.fullname" . }}-headless + replicas: 1 + updateStrategy: + type: {{ .Values.updateStrategy.type }} + {{- if (eq "Recreate" .Values.updateStrategy.type) }} + rollingUpdate: null + {{- end }} + selector: + matchLabels: + {{- include "common.labels.matchLabels" . | nindent 6 }} + role: primary + template: + metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 8 }} + role: primary + app.kubernetes.io/component: primary + {{- with .Values.primary.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.primary.podAnnotations }} + annotations: {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- if .Values.schedulerName }} + schedulerName: "{{ .Values.schedulerName }}" + {{- end }} +{{- include "postgresql.imagePullSecrets" . | indent 6 }} + {{- if .Values.primary.affinity }} + affinity: {{- include "common.tplvalues.render" (dict "value" .Values.primary.affinity "context" $) | nindent 8 }} + {{- else }} + affinity: + podAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAffinityPreset "component" "primary" "context" $) | nindent 10 }} + podAntiAffinity: {{- include "common.affinities.pods" (dict "type" .Values.primary.podAntiAffinityPreset "component" "primary" "context" $) | nindent 10 }} + nodeAffinity: {{- include "common.affinities.nodes" (dict "type" .Values.primary.nodeAffinityPreset.type "key" .Values.primary.nodeAffinityPreset.key "values" .Values.primary.nodeAffinityPreset.values) | nindent 10 }} + {{- end }} + {{- if .Values.primary.nodeSelector }} + nodeSelector: {{- include "common.tplvalues.render" (dict "value" .Values.primary.nodeSelector "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.primary.tolerations }} + tolerations: {{- include "common.tplvalues.render" (dict "value" .Values.primary.tolerations "context" $) | nindent 8 }} + {{- end }} + {{- if .Values.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.securityContext.enabled }} + securityContext: {{- omit .Values.securityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.serviceAccount.enabled }} + serviceAccountName: {{ default (include "common.names.fullname" . ) .Values.serviceAccount.name }} + {{- end }} + {{- if or .Values.primary.extraInitContainers (and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled))) }} + initContainers: + {{- if and .Values.volumePermissions.enabled (or .Values.persistence.enabled (and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled) .Values.tls.enabled) }} + - name: init-chmod-data + image: {{ template "postgresql.volumePermissions.image" . }} + imagePullPolicy: {{ .Values.volumePermissions.image.pullPolicy | quote }} + {{- if .Values.resources }} + resources: {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + command: + - /bin/sh + - -cx + - | + {{- if .Values.persistence.enabled }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown `id -u`:`id -G | cut -d " " -f2` {{ .Values.persistence.mountPath }} + {{- else }} + chown {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} {{ .Values.persistence.mountPath }} + {{- end }} + mkdir -p {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + chmod 700 {{ .Values.persistence.mountPath }}/data {{- if (include "postgresql.mountConfigurationCM" .) }} {{ .Values.persistence.mountPath }}/conf {{- end }} + find {{ .Values.persistence.mountPath }} -mindepth 1 -maxdepth 1 {{- if not (include "postgresql.mountConfigurationCM" .) }} -not -name "conf" {{- end }} -not -name ".snapshot" -not -name "lost+found" | \ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + xargs chown -R `id -u`:`id -G | cut -d " " -f2` + {{- else }} + xargs chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} + {{- end }} + {{- end }} + {{- if and .Values.shmVolume.enabled .Values.shmVolume.chmod.enabled }} + chmod -R 777 /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + cp /tmp/certs/* /opt/bitnami/postgresql/certs/ + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + chown -R `id -u`:`id -G | cut -d " " -f2` /opt/bitnami/postgresql/certs/ + {{- else }} + chown -R {{ .Values.containerSecurityContext.runAsUser }}:{{ .Values.securityContext.fsGroup }} /opt/bitnami/postgresql/certs/ + {{- end }} + chmod 600 {{ template "postgresql.tlsCertKey" . }} + {{- end }} + {{- if eq ( toString ( .Values.volumePermissions.securityContext.runAsUser )) "auto" }} + securityContext: {{- omit .Values.volumePermissions.securityContext "runAsUser" | toYaml | nindent 12 }} + {{- else }} + securityContext: {{- .Values.volumePermissions.securityContext | toYaml | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + mountPath: /tmp/certs + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + {{- end }} + {{- end }} + {{- if .Values.primary.extraInitContainers }} + {{- include "common.tplvalues.render" ( dict "value" .Values.primary.extraInitContainers "context" $ ) | nindent 8 }} + {{- end }} + {{- end }} + {{- if .Values.primary.priorityClassName }} + priorityClassName: {{ .Values.primary.priorityClassName }} + {{- end }} + containers: + - name: {{ template "common.names.fullname" . }} + image: {{ template "postgresql.image" . }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + {{- if .Values.resources }} + resources: {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 12 }} + {{- end }} + env: + - name: BITNAMI_DEBUG + value: {{ ternary "true" "false" .Values.image.debug | quote }} + - name: POSTGRESQL_PORT_NUMBER + value: "{{ template "postgresql.port" . }}" + - name: POSTGRESQL_VOLUME_DIR + value: "{{ .Values.persistence.mountPath }}" + {{- if .Values.postgresqlInitdbArgs }} + - name: POSTGRES_INITDB_ARGS + value: {{ .Values.postgresqlInitdbArgs | quote }} + {{- end }} + {{- if .Values.postgresqlInitdbWalDir }} + - name: POSTGRES_INITDB_WALDIR + value: {{ .Values.postgresqlInitdbWalDir | quote }} + {{- end }} + {{- if .Values.initdbUser }} + - name: POSTGRESQL_INITSCRIPTS_USERNAME + value: {{ .Values.initdbUser }} + {{- end }} + {{- if .Values.initdbPassword }} + - name: POSTGRESQL_INITSCRIPTS_PASSWORD + value: {{ .Values.initdbPassword }} + {{- end }} + {{- if .Values.persistence.mountPath }} + - name: PGDATA + value: {{ .Values.postgresqlDataDir | quote }} + {{- end }} + {{- if .Values.primaryAsStandBy.enabled }} + - name: POSTGRES_MASTER_HOST + value: {{ .Values.primaryAsStandBy.primaryHost }} + - name: POSTGRES_MASTER_PORT_NUMBER + value: {{ .Values.primaryAsStandBy.primaryPort | quote }} + {{- end }} + {{- if or .Values.replication.enabled .Values.primaryAsStandBy.enabled }} + - name: POSTGRES_REPLICATION_MODE + {{- if .Values.primaryAsStandBy.enabled }} + value: "slave" + {{- else }} + value: "master" + {{- end }} + - name: POSTGRES_REPLICATION_USER + value: {{ include "postgresql.replication.username" . | quote }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_REPLICATION_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-replication-password" + {{- else }} + - name: POSTGRES_REPLICATION_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-replication-password + {{- end }} + {{- if not (eq .Values.replication.synchronousCommit "off")}} + - name: POSTGRES_SYNCHRONOUS_COMMIT_MODE + value: {{ .Values.replication.synchronousCommit | quote }} + - name: POSTGRES_NUM_SYNCHRONOUS_REPLICAS + value: {{ .Values.replication.numSynchronousReplicas | quote }} + {{- end }} + - name: POSTGRES_CLUSTER_APP_NAME + value: {{ .Values.replication.applicationName }} + {{- end }} + {{- if not (eq (include "postgresql.username" .) "postgres") }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-postgres-password" + {{- else }} + - name: POSTGRES_POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-postgres-password + {{- end }} + {{- end }} + - name: POSTGRES_USER + value: {{ include "postgresql.username" . | quote }} + {{- if .Values.usePasswordFile }} + - name: POSTGRES_PASSWORD_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-password" + {{- else }} + - name: POSTGRES_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-password + {{- end }} + {{- if (include "postgresql.database" .) }} + - name: POSTGRES_DB + value: {{ (include "postgresql.database" .) | quote }} + {{- end }} + {{- if .Values.extraEnv }} + {{- include "common.tplvalues.render" (dict "value" .Values.extraEnv "context" $) | nindent 12 }} + {{- end }} + - name: POSTGRESQL_ENABLE_LDAP + value: {{ ternary "yes" "no" .Values.ldap.enabled | quote }} + {{- if .Values.ldap.enabled }} + - name: POSTGRESQL_LDAP_SERVER + value: {{ .Values.ldap.server }} + - name: POSTGRESQL_LDAP_PORT + value: {{ .Values.ldap.port | quote }} + - name: POSTGRESQL_LDAP_SCHEME + value: {{ .Values.ldap.scheme }} + {{- if .Values.ldap.tls }} + - name: POSTGRESQL_LDAP_TLS + value: "1" + {{- end }} + - name: POSTGRESQL_LDAP_PREFIX + value: {{ .Values.ldap.prefix | quote }} + - name: POSTGRESQL_LDAP_SUFFIX + value: {{ .Values.ldap.suffix | quote }} + - name: POSTGRESQL_LDAP_BASE_DN + value: {{ .Values.ldap.baseDN }} + - name: POSTGRESQL_LDAP_BIND_DN + value: {{ .Values.ldap.bindDN }} + {{- if (not (empty .Values.ldap.bind_password)) }} + - name: POSTGRESQL_LDAP_BIND_PASSWORD + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-ldap-password + {{- end}} + - name: POSTGRESQL_LDAP_SEARCH_ATTR + value: {{ .Values.ldap.search_attr }} + - name: POSTGRESQL_LDAP_SEARCH_FILTER + value: {{ .Values.ldap.search_filter }} + - name: POSTGRESQL_LDAP_URL + value: {{ .Values.ldap.url }} + {{- end}} + - name: POSTGRESQL_ENABLE_TLS + value: {{ ternary "yes" "no" .Values.tls.enabled | quote }} + {{- if .Values.tls.enabled }} + - name: POSTGRESQL_TLS_PREFER_SERVER_CIPHERS + value: {{ ternary "yes" "no" .Values.tls.preferServerCiphers | quote }} + - name: POSTGRESQL_TLS_CERT_FILE + value: {{ template "postgresql.tlsCert" . }} + - name: POSTGRESQL_TLS_KEY_FILE + value: {{ template "postgresql.tlsCertKey" . }} + {{- if .Values.tls.certCAFilename }} + - name: POSTGRESQL_TLS_CA_FILE + value: {{ template "postgresql.tlsCACert" . }} + {{- end }} + {{- if .Values.tls.crlFilename }} + - name: POSTGRESQL_TLS_CRL_FILE + value: {{ template "postgresql.tlsCRL" . }} + {{- end }} + {{- end }} + - name: POSTGRESQL_LOG_HOSTNAME + value: {{ .Values.audit.logHostname | quote }} + - name: POSTGRESQL_LOG_CONNECTIONS + value: {{ .Values.audit.logConnections | quote }} + - name: POSTGRESQL_LOG_DISCONNECTIONS + value: {{ .Values.audit.logDisconnections | quote }} + {{- if .Values.audit.logLinePrefix }} + - name: POSTGRESQL_LOG_LINE_PREFIX + value: {{ .Values.audit.logLinePrefix | quote }} + {{- end }} + {{- if .Values.audit.logTimezone }} + - name: POSTGRESQL_LOG_TIMEZONE + value: {{ .Values.audit.logTimezone | quote }} + {{- end }} + {{- if .Values.audit.pgAuditLog }} + - name: POSTGRESQL_PGAUDIT_LOG + value: {{ .Values.audit.pgAuditLog | quote }} + {{- end }} + - name: POSTGRESQL_PGAUDIT_LOG_CATALOG + value: {{ .Values.audit.pgAuditLogCatalog | quote }} + - name: POSTGRESQL_CLIENT_MIN_MESSAGES + value: {{ .Values.audit.clientMinMessages | quote }} + - name: POSTGRESQL_SHARED_PRELOAD_LIBRARIES + value: {{ .Values.postgresqlSharedPreloadLibraries | quote }} + {{- if .Values.postgresqlMaxConnections }} + - name: POSTGRESQL_MAX_CONNECTIONS + value: {{ .Values.postgresqlMaxConnections | quote }} + {{- end }} + {{- if .Values.postgresqlPostgresConnectionLimit }} + - name: POSTGRESQL_POSTGRES_CONNECTION_LIMIT + value: {{ .Values.postgresqlPostgresConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlDbUserConnectionLimit }} + - name: POSTGRESQL_USERNAME_CONNECTION_LIMIT + value: {{ .Values.postgresqlDbUserConnectionLimit | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesInterval }} + - name: POSTGRESQL_TCP_KEEPALIVES_INTERVAL + value: {{ .Values.postgresqlTcpKeepalivesInterval | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesIdle }} + - name: POSTGRESQL_TCP_KEEPALIVES_IDLE + value: {{ .Values.postgresqlTcpKeepalivesIdle | quote }} + {{- end }} + {{- if .Values.postgresqlStatementTimeout }} + - name: POSTGRESQL_STATEMENT_TIMEOUT + value: {{ .Values.postgresqlStatementTimeout | quote }} + {{- end }} + {{- if .Values.postgresqlTcpKeepalivesCount }} + - name: POSTGRESQL_TCP_KEEPALIVES_COUNT + value: {{ .Values.postgresqlTcpKeepalivesCount | quote }} + {{- end }} + {{- if .Values.postgresqlPghbaRemoveFilters }} + - name: POSTGRESQL_PGHBA_REMOVE_FILTERS + value: {{ .Values.postgresqlPghbaRemoveFilters | quote }} + {{- end }} + {{- if .Values.extraEnvVarsCM }} + envFrom: + - configMapRef: + name: {{ tpl .Values.extraEnvVarsCM . }} + {{- end }} + ports: + - name: tcp-postgresql + containerPort: {{ template "postgresql.port" . }} + {{- if .Values.startupProbe.enabled }} + startupProbe: + exec: + command: + - /bin/sh + - -c + {{- if (include "postgresql.database" .) }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- else }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- end }} + initialDelaySeconds: {{ .Values.startupProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.startupProbe.periodSeconds }} + timeoutSeconds: {{ .Values.startupProbe.timeoutSeconds }} + successThreshold: {{ .Values.startupProbe.successThreshold }} + failureThreshold: {{ .Values.startupProbe.failureThreshold }} + {{- else if .Values.customStartupProbe }} + startupProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customStartupProbe "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.livenessProbe.enabled }} + livenessProbe: + exec: + command: + - /bin/sh + - -c + {{- if (include "postgresql.database" .) }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} -d "dbname={{ include "postgresql.database" . }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}{{- end }}" -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- else }} + - exec pg_isready -U {{ include "postgresql.username" . | quote }} {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} -d "sslcert={{ include "postgresql.tlsCert" . }} sslkey={{ include "postgresql.tlsCertKey" . }}"{{- end }} -h 127.0.0.1 -p {{ template "postgresql.port" . }} + {{- end }} + initialDelaySeconds: {{ .Values.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.livenessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.livenessProbe.timeoutSeconds }} + successThreshold: {{ .Values.livenessProbe.successThreshold }} + failureThreshold: {{ .Values.livenessProbe.failureThreshold }} + {{- else if .Values.customLivenessProbe }} + livenessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customLivenessProbe "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.readinessProbe.enabled }} + readinessProbe: + exec: + command: + - /bin/sh + - -c + - -e + {{- include "postgresql.readinessProbeCommand" . | nindent 16 }} + initialDelaySeconds: {{ .Values.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.readinessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.readinessProbe.timeoutSeconds }} + successThreshold: {{ .Values.readinessProbe.successThreshold }} + failureThreshold: {{ .Values.readinessProbe.failureThreshold }} + {{- else if .Values.customReadinessProbe }} + readinessProbe: {{- include "common.tplvalues.render" (dict "value" .Values.customReadinessProbe "context" $) | nindent 12 }} + {{- end }} + volumeMounts: + {{- if or (.Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql,sql.gz}") .Values.initdbScriptsConfigMap .Values.initdbScripts }} + - name: custom-init-scripts + mountPath: /docker-entrypoint-initdb.d/ + {{- end }} + {{- if .Values.initdbScriptsSecret }} + - name: custom-init-scripts-secret + mountPath: /docker-entrypoint-initdb.d/secret + {{- end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + mountPath: /bitnami/postgresql/conf/conf.d/ + {{- end }} + {{- if .Values.usePasswordFile }} + - name: postgresql-password + mountPath: /opt/bitnami/postgresql/secrets/ + {{- end }} + {{- if .Values.tls.enabled }} + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + readOnly: true + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + mountPath: /dev/shm + {{- end }} + {{- if .Values.persistence.enabled }} + - name: data + mountPath: {{ .Values.persistence.mountPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap }} + - name: postgresql-config + mountPath: /bitnami/postgresql/conf + {{- end }} + {{- if .Values.primary.extraVolumeMounts }} + {{- toYaml .Values.primary.extraVolumeMounts | nindent 12 }} + {{- end }} +{{- if .Values.primary.sidecars }} +{{- include "common.tplvalues.render" ( dict "value" .Values.primary.sidecars "context" $ ) | nindent 8 }} +{{- end }} +{{- if .Values.metrics.enabled }} + - name: metrics + image: {{ template "postgresql.metrics.image" . }} + imagePullPolicy: {{ .Values.metrics.image.pullPolicy | quote }} + {{- if .Values.metrics.securityContext.enabled }} + securityContext: {{- omit .Values.metrics.securityContext "enabled" | toYaml | nindent 12 }} + {{- end }} + env: + {{- $database := required "In order to enable metrics you need to specify a database (.Values.postgresqlDatabase or .Values.global.postgresql.postgresqlDatabase)" (include "postgresql.database" .) }} + {{- $sslmode := ternary "require" "disable" .Values.tls.enabled }} + {{- if and .Values.tls.enabled .Values.tls.certCAFilename }} + - name: DATA_SOURCE_NAME + value: {{ printf "host=127.0.0.1 port=%d user=%s sslmode=%s sslcert=%s sslkey=%s" (int (include "postgresql.port" .)) (include "postgresql.username" .) $sslmode (include "postgresql.tlsCert" .) (include "postgresql.tlsCertKey" .) }} + {{- else }} + - name: DATA_SOURCE_URI + value: {{ printf "127.0.0.1:%d/%s?sslmode=%s" (int (include "postgresql.port" .)) $database $sslmode }} + {{- end }} + {{- if .Values.usePasswordFile }} + - name: DATA_SOURCE_PASS_FILE + value: "/opt/bitnami/postgresql/secrets/postgresql-password" + {{- else }} + - name: DATA_SOURCE_PASS + valueFrom: + secretKeyRef: + name: {{ template "postgresql.secretName" . }} + key: postgresql-password + {{- end }} + - name: DATA_SOURCE_USER + value: {{ template "postgresql.username" . }} + {{- if .Values.metrics.extraEnvVars }} + {{- include "common.tplvalues.render" (dict "value" .Values.metrics.extraEnvVars "context" $) | nindent 12 }} + {{- end }} + {{- if .Values.livenessProbe.enabled }} + livenessProbe: + httpGet: + path: / + port: http-metrics + initialDelaySeconds: {{ .Values.metrics.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.metrics.livenessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.metrics.livenessProbe.timeoutSeconds }} + successThreshold: {{ .Values.metrics.livenessProbe.successThreshold }} + failureThreshold: {{ .Values.metrics.livenessProbe.failureThreshold }} + {{- end }} + {{- if .Values.readinessProbe.enabled }} + readinessProbe: + httpGet: + path: / + port: http-metrics + initialDelaySeconds: {{ .Values.metrics.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.metrics.readinessProbe.periodSeconds }} + timeoutSeconds: {{ .Values.metrics.readinessProbe.timeoutSeconds }} + successThreshold: {{ .Values.metrics.readinessProbe.successThreshold }} + failureThreshold: {{ .Values.metrics.readinessProbe.failureThreshold }} + {{- end }} + volumeMounts: + {{- if .Values.usePasswordFile }} + - name: postgresql-password + mountPath: /opt/bitnami/postgresql/secrets/ + {{- end }} + {{- if .Values.tls.enabled }} + - name: postgresql-certificates + mountPath: /opt/bitnami/postgresql/certs + readOnly: true + {{- end }} + {{- if .Values.metrics.customMetrics }} + - name: custom-metrics + mountPath: /conf + readOnly: true + args: ["--extend.query-path", "/conf/custom-metrics.yaml"] + {{- end }} + ports: + - name: http-metrics + containerPort: 9187 + {{- if .Values.metrics.resources }} + resources: {{- toYaml .Values.metrics.resources | nindent 12 }} + {{- end }} +{{- end }} + volumes: + {{- if or (.Files.Glob "files/postgresql.conf") (.Files.Glob "files/pg_hba.conf") .Values.postgresqlConfiguration .Values.pgHbaConfiguration .Values.configurationConfigMap}} + - name: postgresql-config + configMap: + name: {{ template "postgresql.configurationCM" . }} + {{- end }} + {{- if or (.Files.Glob "files/conf.d/*.conf") .Values.postgresqlExtendedConf .Values.extendedConfConfigMap }} + - name: postgresql-extended-config + configMap: + name: {{ template "postgresql.extendedConfigurationCM" . }} + {{- end }} + {{- if .Values.usePasswordFile }} + - name: postgresql-password + secret: + secretName: {{ template "postgresql.secretName" . }} + {{- end }} + {{- if or (.Files.Glob "files/docker-entrypoint-initdb.d/*.{sh,sql,sql.gz}") .Values.initdbScriptsConfigMap .Values.initdbScripts }} + - name: custom-init-scripts + configMap: + name: {{ template "postgresql.initdbScriptsCM" . }} + {{- end }} + {{- if .Values.initdbScriptsSecret }} + - name: custom-init-scripts-secret + secret: + secretName: {{ template "postgresql.initdbScriptsSecret" . }} + {{- end }} + {{- if .Values.tls.enabled }} + - name: raw-certificates + secret: + secretName: {{ required "A secret containing TLS certificates is required when TLS is enabled" .Values.tls.certificatesSecret }} + - name: postgresql-certificates + emptyDir: {} + {{- end }} + {{- if .Values.primary.extraVolumes }} + {{- toYaml .Values.primary.extraVolumes | nindent 8 }} + {{- end }} + {{- if and .Values.metrics.enabled .Values.metrics.customMetrics }} + - name: custom-metrics + configMap: + name: {{ template "postgresql.metricsCM" . }} + {{- end }} + {{- if .Values.shmVolume.enabled }} + - name: dshm + emptyDir: + medium: Memory + sizeLimit: 1Gi + {{- end }} +{{- if and .Values.persistence.enabled .Values.persistence.existingClaim }} + - name: data + persistentVolumeClaim: +{{- with .Values.persistence.existingClaim }} + claimName: {{ tpl . $ }} +{{- end }} +{{- else if not .Values.persistence.enabled }} + - name: data + emptyDir: {} +{{- else if and .Values.persistence.enabled (not .Values.persistence.existingClaim) }} + volumeClaimTemplates: + - metadata: + name: data + {{- with .Values.persistence.annotations }} + annotations: + {{- range $key, $value := . }} + {{ $key }}: {{ $value }} + {{- end }} + {{- end }} + spec: + accessModes: + {{- range .Values.persistence.accessModes }} + - {{ . | quote }} + {{- end }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} + {{ include "common.storage.class" (dict "persistence" .Values.persistence "global" .Values.global) }} + {{- if .Values.persistence.selector }} + selector: {{- include "common.tplvalues.render" (dict "value" .Values.persistence.selector "context" $) | nindent 10 }} + {{- end -}} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-headless.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-headless.yaml new file mode 100644 index 0000000000..6f5f3b9ee4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-headless.yaml @@ -0,0 +1,28 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }}-headless + labels: + {{- include "common.labels.standard" . | nindent 4 }} + {{- if .Values.commonAnnotations }} + annotations: {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + # Use this annotation in addition to the actual publishNotReadyAddresses + # field below because the annotation will stop being respected soon but the + # field is broken in some versions of Kubernetes: + # https://github.com/kubernetes/kubernetes/issues/58662 + service.alpha.kubernetes.io/tolerate-unready-endpoints: "true" + namespace: {{ .Release.Namespace }} +spec: + type: ClusterIP + clusterIP: None + # We want all pods in the StatefulSet to have their addresses published for + # the sake of the other Postgresql pods even before they're ready, since they + # have to be able to talk to each other in order to become ready. + publishNotReadyAddresses: true + ports: + - name: tcp-postgresql + port: {{ template "postgresql.port" . }} + targetPort: tcp-postgresql + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-read.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-read.yaml new file mode 100644 index 0000000000..56195ea1e6 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc-read.yaml @@ -0,0 +1,43 @@ +{{- if .Values.replication.enabled }} +{{- $serviceAnnotations := coalesce .Values.readReplicas.service.annotations .Values.service.annotations -}} +{{- $serviceType := coalesce .Values.readReplicas.service.type .Values.service.type -}} +{{- $serviceLoadBalancerIP := coalesce .Values.readReplicas.service.loadBalancerIP .Values.service.loadBalancerIP -}} +{{- $serviceLoadBalancerSourceRanges := coalesce .Values.readReplicas.service.loadBalancerSourceRanges .Values.service.loadBalancerSourceRanges -}} +{{- $serviceClusterIP := coalesce .Values.readReplicas.service.clusterIP .Values.service.clusterIP -}} +{{- $serviceNodePort := coalesce .Values.readReplicas.service.nodePort .Values.service.nodePort -}} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }}-read + labels: + {{- include "common.labels.standard" . | nindent 4 }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- if $serviceAnnotations }} + {{- include "common.tplvalues.render" (dict "value" $serviceAnnotations "context" $) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + type: {{ $serviceType }} + {{- if and $serviceLoadBalancerIP (eq $serviceType "LoadBalancer") }} + loadBalancerIP: {{ $serviceLoadBalancerIP }} + {{- end }} + {{- if and (eq $serviceType "LoadBalancer") $serviceLoadBalancerSourceRanges }} + loadBalancerSourceRanges: {{- include "common.tplvalues.render" (dict "value" $serviceLoadBalancerSourceRanges "context" $) | nindent 4 }} + {{- end }} + {{- if and (eq $serviceType "ClusterIP") $serviceClusterIP }} + clusterIP: {{ $serviceClusterIP }} + {{- end }} + ports: + - name: tcp-postgresql + port: {{ template "postgresql.port" . }} + targetPort: tcp-postgresql + {{- if $serviceNodePort }} + nodePort: {{ $serviceNodePort }} + {{- end }} + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} + role: read +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc.yaml new file mode 100644 index 0000000000..a29431b6a4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/templates/svc.yaml @@ -0,0 +1,41 @@ +{{- $serviceAnnotations := coalesce .Values.primary.service.annotations .Values.service.annotations -}} +{{- $serviceType := coalesce .Values.primary.service.type .Values.service.type -}} +{{- $serviceLoadBalancerIP := coalesce .Values.primary.service.loadBalancerIP .Values.service.loadBalancerIP -}} +{{- $serviceLoadBalancerSourceRanges := coalesce .Values.primary.service.loadBalancerSourceRanges .Values.service.loadBalancerSourceRanges -}} +{{- $serviceClusterIP := coalesce .Values.primary.service.clusterIP .Values.service.clusterIP -}} +{{- $serviceNodePort := coalesce .Values.primary.service.nodePort .Values.service.nodePort -}} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "common.names.fullname" . }} + labels: + {{- include "common.labels.standard" . | nindent 4 }} + annotations: + {{- if .Values.commonAnnotations }} + {{- include "common.tplvalues.render" ( dict "value" .Values.commonAnnotations "context" $ ) | nindent 4 }} + {{- end }} + {{- if $serviceAnnotations }} + {{- include "common.tplvalues.render" (dict "value" $serviceAnnotations "context" $) | nindent 4 }} + {{- end }} + namespace: {{ .Release.Namespace }} +spec: + type: {{ $serviceType }} + {{- if and $serviceLoadBalancerIP (eq $serviceType "LoadBalancer") }} + loadBalancerIP: {{ $serviceLoadBalancerIP }} + {{- end }} + {{- if and (eq $serviceType "LoadBalancer") $serviceLoadBalancerSourceRanges }} + loadBalancerSourceRanges: {{- include "common.tplvalues.render" (dict "value" $serviceLoadBalancerSourceRanges "context" $) | nindent 4 }} + {{- end }} + {{- if and (eq $serviceType "ClusterIP") $serviceClusterIP }} + clusterIP: {{ $serviceClusterIP }} + {{- end }} + ports: + - name: tcp-postgresql + port: {{ template "postgresql.port" . }} + targetPort: tcp-postgresql + {{- if $serviceNodePort }} + nodePort: {{ $serviceNodePort }} + {{- end }} + selector: + {{- include "common.labels.matchLabels" . | nindent 4 }} + role: primary diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.schema.json b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.schema.json new file mode 100644 index 0000000000..66a2a9dd06 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.schema.json @@ -0,0 +1,103 @@ +{ + "$schema": "http://json-schema.org/schema#", + "type": "object", + "properties": { + "postgresqlUsername": { + "type": "string", + "title": "Admin user", + "form": true + }, + "postgresqlPassword": { + "type": "string", + "title": "Password", + "form": true + }, + "persistence": { + "type": "object", + "properties": { + "size": { + "type": "string", + "title": "Persistent Volume Size", + "form": true, + "render": "slider", + "sliderMin": 1, + "sliderMax": 100, + "sliderUnit": "Gi" + } + } + }, + "resources": { + "type": "object", + "title": "Required Resources", + "description": "Configure resource requests", + "form": true, + "properties": { + "requests": { + "type": "object", + "properties": { + "memory": { + "type": "string", + "form": true, + "render": "slider", + "title": "Memory Request", + "sliderMin": 10, + "sliderMax": 2048, + "sliderUnit": "Mi" + }, + "cpu": { + "type": "string", + "form": true, + "render": "slider", + "title": "CPU Request", + "sliderMin": 10, + "sliderMax": 2000, + "sliderUnit": "m" + } + } + } + } + }, + "replication": { + "type": "object", + "form": true, + "title": "Replication Details", + "properties": { + "enabled": { + "type": "boolean", + "title": "Enable Replication", + "form": true + }, + "readReplicas": { + "type": "integer", + "title": "read Replicas", + "form": true, + "hidden": { + "value": false, + "path": "replication/enabled" + } + } + } + }, + "volumePermissions": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "form": true, + "title": "Enable Init Containers", + "description": "Change the owner of the persist volume mountpoint to RunAsUser:fsGroup" + } + } + }, + "metrics": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean", + "title": "Configure metrics exporter", + "form": true + } + } + } + } +} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.yaml new file mode 100644 index 0000000000..82ce092344 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/charts/postgresql/values.yaml @@ -0,0 +1,824 @@ +## Global Docker image parameters +## Please, note that this will override the image parameters, including dependencies, configured to use the global value +## Current available global Docker image parameters: imageRegistry and imagePullSecrets +## +global: + postgresql: {} +# imageRegistry: myRegistryName +# imagePullSecrets: +# - myRegistryKeySecretName +# storageClass: myStorageClass + +## Bitnami PostgreSQL image version +## ref: https://hub.docker.com/r/bitnami/postgresql/tags/ +## +image: + registry: docker.io + repository: bitnami/postgresql + tag: 11.11.0-debian-10-r71 + ## Specify a imagePullPolicy + ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' + ## ref: http://kubernetes.io/docs/user-guide/images/#pre-pulling-images + ## + 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 + + ## Set to true if you would like to see extra information on logs + ## It turns BASH and/or NAMI debugging in the image + ## + debug: false + +## String to partially override common.names.fullname template (will maintain the release name) +## +# nameOverride: + +## String to fully override common.names.fullname template +## +# fullnameOverride: + +## +## Init containers parameters: +## volumePermissions: Change the owner of the persist volume mountpoint to RunAsUser:fsGroup +## +volumePermissions: + enabled: false + image: + registry: docker.io + repository: bitnami/bitnami-shell + tag: "10" + ## Specify a imagePullPolicy + ## Defaults to 'Always' if image tag is 'latest', else set to 'IfNotPresent' + ## ref: http://kubernetes.io/docs/user-guide/images/#pre-pulling-images + ## + pullPolicy: Always + ## 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 + ## Init container Security Context + ## Note: the chown of the data folder is done to securityContext.runAsUser + ## and not the below volumePermissions.securityContext.runAsUser + ## When runAsUser is set to special value "auto", init container will try to chwon the + ## data folder to autodetermined user&group, using commands: `id -u`:`id -G | cut -d" " -f2` + ## "auto" is especially useful for OpenShift which has scc with dynamic userids (and 0 is not allowed). + ## You may want to use this volumePermissions.securityContext.runAsUser="auto" in combination with + ## pod securityContext.enabled=false and shmVolume.chmod.enabled=false + ## + securityContext: + runAsUser: 0 + +## Use an alternate scheduler, e.g. "stork". +## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ +## +# schedulerName: + +## Pod Security Context +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ +## +securityContext: + enabled: true + fsGroup: 1001 + +## Container Security Context +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ +## +containerSecurityContext: + enabled: true + runAsUser: 1001 + +## Pod Service Account +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/ +## +serviceAccount: + enabled: false + ## Name of an already existing service account. Setting this value disables the automatic service account creation. + # name: + +## Pod Security Policy +## ref: https://kubernetes.io/docs/concepts/policy/pod-security-policy/ +## +psp: + create: false + +## Creates role for ServiceAccount +## Required for PSP +## +rbac: + create: false + +replication: + enabled: false + user: repl_user + password: repl_password + readReplicas: 1 + ## Set synchronous commit mode: on, off, remote_apply, remote_write and local + ## ref: https://www.postgresql.org/docs/9.6/runtime-config-wal.html#GUC-WAL-LEVEL + synchronousCommit: 'off' + ## From the number of `readReplicas` defined above, set the number of those that will have synchronous replication + ## NOTE: It cannot be > readReplicas + numSynchronousReplicas: 0 + ## Replication Cluster application name. Useful for defining multiple replication policies + ## + applicationName: my_application + +## PostgreSQL admin password (used when `postgresqlUsername` is not `postgres`) +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#creating-a-database-user-on-first-run (see note!) +# postgresqlPostgresPassword: + +## PostgreSQL user (has superuser privileges if username is `postgres`) +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#setting-the-root-password-on-first-run +## +postgresqlUsername: postgres + +## PostgreSQL password +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#setting-the-root-password-on-first-run +## +# postgresqlPassword: + +## PostgreSQL password using existing secret +## existingSecret: secret +## + +## Mount PostgreSQL secret as a file instead of passing environment variable +# usePasswordFile: false + +## Create a database +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md#creating-a-database-on-first-run +## +# postgresqlDatabase: + +## PostgreSQL data dir +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md +## +postgresqlDataDir: /bitnami/postgresql/data + +## An array to add extra environment variables +## For example: +## extraEnv: +## - name: FOO +## value: "bar" +## +# extraEnv: +extraEnv: [] + +## Name of a ConfigMap containing extra env vars +## +# extraEnvVarsCM: + +## Specify extra initdb args +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md +## +# postgresqlInitdbArgs: + +## Specify a custom location for the PostgreSQL transaction log +## ref: https://github.com/bitnami/bitnami-docker-postgresql/blob/master/README.md +## +# postgresqlInitdbWalDir: + +## PostgreSQL configuration +## Specify runtime configuration parameters as a dict, using camelCase, e.g. +## {"sharedBuffers": "500MB"} +## Alternatively, you can put your postgresql.conf under the files/ directory +## ref: https://www.postgresql.org/docs/current/static/runtime-config.html +## +# postgresqlConfiguration: + +## PostgreSQL extended configuration +## As above, but _appended_ to the main configuration +## Alternatively, you can put your *.conf under the files/conf.d/ directory +## https://github.com/bitnami/bitnami-docker-postgresql#allow-settings-to-be-loaded-from-files-other-than-the-default-postgresqlconf +## +# postgresqlExtendedConf: + +## Configure current cluster's primary server to be the standby server in other cluster. +## This will allow cross cluster replication and provide cross cluster high availability. +## You will need to configure pgHbaConfiguration if you want to enable this feature with local cluster replication enabled. +## +primaryAsStandBy: + enabled: false + # primaryHost: + # primaryPort: + +## PostgreSQL client authentication configuration +## Specify content for pg_hba.conf +## Default: do not create pg_hba.conf +## Alternatively, you can put your pg_hba.conf under the files/ directory +# pgHbaConfiguration: |- +# local all all trust +# host all all localhost trust +# host mydatabase mysuser 192.168.0.0/24 md5 + +## ConfigMap with PostgreSQL configuration +## NOTE: This will override postgresqlConfiguration and pgHbaConfiguration +# configurationConfigMap: + +## ConfigMap with PostgreSQL extended configuration +# extendedConfConfigMap: + +## initdb scripts +## Specify dictionary of scripts to be run at first boot +## Alternatively, you can put your scripts under the files/docker-entrypoint-initdb.d directory +## +# initdbScripts: +# my_init_script.sh: | +# #!/bin/sh +# echo "Do something." + +## ConfigMap with scripts to be run at first boot +## NOTE: This will override initdbScripts +# initdbScriptsConfigMap: + +## Secret with scripts to be run at first boot (in case it contains sensitive information) +## NOTE: This can work along initdbScripts or initdbScriptsConfigMap +# initdbScriptsSecret: + +## Specify the PostgreSQL username and password to execute the initdb scripts +# initdbUser: +# initdbPassword: + +## Audit settings +## https://github.com/bitnami/bitnami-docker-postgresql#auditing +## +audit: + ## Log client hostnames + ## + logHostname: false + ## Log connections to the server + ## + logConnections: false + ## Log disconnections + ## + logDisconnections: false + ## Operation to audit using pgAudit (default if not set) + ## + pgAuditLog: "" + ## Log catalog using pgAudit + ## + pgAuditLogCatalog: "off" + ## Log level for clients + ## + clientMinMessages: error + ## Template for log line prefix (default if not set) + ## + logLinePrefix: "" + ## Log timezone + ## + logTimezone: "" + +## Shared preload libraries +## +postgresqlSharedPreloadLibraries: "pgaudit" + +## Maximum total connections +## +postgresqlMaxConnections: + +## Maximum connections for the postgres user +## +postgresqlPostgresConnectionLimit: + +## Maximum connections for the created user +## +postgresqlDbUserConnectionLimit: + +## TCP keepalives interval +## +postgresqlTcpKeepalivesInterval: + +## TCP keepalives idle +## +postgresqlTcpKeepalivesIdle: + +## TCP keepalives count +## +postgresqlTcpKeepalivesCount: + +## Statement timeout +## +postgresqlStatementTimeout: + +## Remove pg_hba.conf lines with the following comma-separated patterns +## (cannot be used with custom pg_hba.conf) +## +postgresqlPghbaRemoveFilters: + +## Optional duration in seconds the pod needs to terminate gracefully. +## ref: https://kubernetes.io/docs/concepts/workloads/pods/pod/#termination-of-pods +## +# terminationGracePeriodSeconds: 30 + +## LDAP configuration +## +ldap: + enabled: false + url: '' + server: '' + port: '' + prefix: '' + suffix: '' + baseDN: '' + bindDN: '' + bind_password: + search_attr: '' + search_filter: '' + scheme: '' + tls: {} + +## PostgreSQL service configuration +## +service: + ## PosgresSQL service type + ## + type: ClusterIP + # clusterIP: None + port: 5432 + + ## Specify the nodePort value for the LoadBalancer and NodePort service types. + ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport + ## + # nodePort: + + ## Provide any additional annotations which may be required. Evaluated as a template. + ## + annotations: {} + ## Set the LoadBalancer service type to internal only. + ## ref: https://kubernetes.io/docs/concepts/services-networking/service/#internal-load-balancer + ## + # loadBalancerIP: + ## Load Balancer sources. Evaluated as a template. + ## https://kubernetes.io/docs/tasks/access-application-cluster/configure-cloud-provider-firewall/#restrict-access-for-loadbalancer-service + ## + # loadBalancerSourceRanges: + # - 10.10.10.0/24 + +## Start primary and read(s) pod(s) without limitations on shm memory. +## By default docker and containerd (and possibly other container runtimes) +## limit `/dev/shm` to `64M` (see e.g. the +## [docker issue](https://github.com/docker-library/postgres/issues/416) and the +## [containerd issue](https://github.com/containerd/containerd/issues/3654), +## which could be not enough if PostgreSQL uses parallel workers heavily. +## +shmVolume: + ## Set `shmVolume.enabled` to `true` to mount a new tmpfs volume to remove + ## this limitation. + ## + enabled: true + ## Set to `true` to `chmod 777 /dev/shm` on a initContainer. + ## This option is ignored if `volumePermissions.enabled` is `false` + ## + chmod: + enabled: true + +## PostgreSQL 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) +## +persistence: + enabled: true + ## A manually managed Persistent Volume and Claim + ## If defined, PVC must be created manually before volume will be bound + ## The value is evaluated as a template, so, for example, the name can depend on .Release or .Chart + ## + # existingClaim: + + ## The path the volume will be mounted at, useful when using different + ## PostgreSQL images. + ## + mountPath: /bitnami/postgresql + + ## The subdirectory of the volume to mount to, useful in dev environments + ## and one PV for multiple services. + ## + subPath: '' + + # storageClass: "-" + accessModes: + - ReadWriteOnce + size: 8Gi + annotations: {} + ## selector can be used to match an existing PersistentVolume + ## selector: + ## matchLabels: + ## app: my-app + selector: {} + +## updateStrategy for PostgreSQL StatefulSet and its reads StatefulSets +## ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies +## +updateStrategy: + type: RollingUpdate + +## +## PostgreSQL Primary parameters +## +primary: + ## PostgreSQL Primary pod affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAffinityPreset: "" + + ## PostgreSQL Primary pod anti-affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAntiAffinityPreset: soft + + ## PostgreSQL Primary node affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity + ## Allowed values: soft, hard + ## + nodeAffinityPreset: + ## Node affinity type + ## Allowed values: soft, hard + type: "" + ## Node label key to match + ## E.g. + ## key: "kubernetes.io/e2e-az-name" + ## + key: "" + ## Node label values to match + ## E.g. + ## values: + ## - e2e-az1 + ## - e2e-az2 + ## + values: [] + + ## Affinity for PostgreSQL primary pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity + ## Note: primary.podAffinityPreset, primary.podAntiAffinityPreset, and primary.nodeAffinityPreset will be ignored when it's set + ## + affinity: {} + + ## Node labels for PostgreSQL primary pods assignment + ## ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Tolerations for PostgreSQL primary pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + ## + tolerations: [] + + labels: {} + annotations: {} + podLabels: {} + podAnnotations: {} + priorityClassName: '' + ## Extra init containers + ## Example + ## + ## extraInitContainers: + ## - name: do-something + ## image: busybox + ## command: ['do', 'something'] + ## + extraInitContainers: [] + + ## Additional PostgreSQL primary Volume mounts + ## + extraVolumeMounts: [] + ## Additional PostgreSQL primary Volumes + ## + extraVolumes: [] + ## Add sidecars to the pod + ## + ## For example: + ## sidecars: + ## - name: your-image-name + ## image: your-image + ## imagePullPolicy: Always + ## ports: + ## - name: portname + ## containerPort: 1234 + ## + sidecars: [] + + ## Override the service configuration for primary + ## + service: {} + # type: + # nodePort: + # clusterIP: + +## +## PostgreSQL read only replica parameters +## +readReplicas: + ## PostgreSQL read only pod affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAffinityPreset: "" + + ## PostgreSQL read only pod anti-affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity + ## Allowed values: soft, hard + ## + podAntiAffinityPreset: soft + + ## PostgreSQL read only node affinity preset + ## ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity + ## Allowed values: soft, hard + ## + nodeAffinityPreset: + ## Node affinity type + ## Allowed values: soft, hard + type: "" + ## Node label key to match + ## E.g. + ## key: "kubernetes.io/e2e-az-name" + ## + key: "" + ## Node label values to match + ## E.g. + ## values: + ## - e2e-az1 + ## - e2e-az2 + ## + values: [] + + ## Affinity for PostgreSQL read only pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#affinity-and-anti-affinity + ## Note: readReplicas.podAffinityPreset, readReplicas.podAntiAffinityPreset, and readReplicas.nodeAffinityPreset will be ignored when it's set + ## + affinity: {} + + ## Node labels for PostgreSQL read only pods assignment + ## ref: https://kubernetes.io/docs/user-guide/node-selection/ + ## + nodeSelector: {} + + ## Tolerations for PostgreSQL read only pods assignment + ## ref: https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + ## + tolerations: [] + labels: {} + annotations: {} + podLabels: {} + podAnnotations: {} + priorityClassName: '' + + ## Extra init containers + ## Example + ## + ## extraInitContainers: + ## - name: do-something + ## image: busybox + ## command: ['do', 'something'] + ## + extraInitContainers: [] + + ## Additional PostgreSQL read replicas Volume mounts + ## + extraVolumeMounts: [] + + ## Additional PostgreSQL read replicas Volumes + ## + extraVolumes: [] + + ## Add sidecars to the pod + ## + ## For example: + ## sidecars: + ## - name: your-image-name + ## image: your-image + ## imagePullPolicy: Always + ## ports: + ## - name: portname + ## containerPort: 1234 + ## + sidecars: [] + + ## Override the service configuration for read + ## + service: {} + # type: + # nodePort: + # clusterIP: + + ## Whether to enable PostgreSQL read replicas data Persistent + ## + persistence: + enabled: true + + # Override the resource configuration for read replicas + resources: {} + # requests: + # memory: 256Mi + # cpu: 250m + +## Configure resource requests and limits +## ref: http://kubernetes.io/docs/user-guide/compute-resources/ +## +resources: + requests: + memory: 256Mi + cpu: 250m + +## Add annotations to all the deployed resources +## +commonAnnotations: {} + +networkPolicy: + ## Enable creation of NetworkPolicy resources. Only Ingress traffic is filtered for now. + ## + enabled: false + + ## The Policy model to apply. When set to false, only pods with the correct + ## client label will have network access to the port PostgreSQL is listening + ## on. When true, PostgreSQL will accept connections from any source + ## (with the correct destination port). + ## + allowExternal: true + + ## 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 DB. + ## But sometimes, we want the DB 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: {} + +## Configure extra options for startup, liveness and readiness probes +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes/#configure-probes +## +startupProbe: + enabled: false + initialDelaySeconds: 30 + periodSeconds: 15 + timeoutSeconds: 5 + failureThreshold: 10 + successThreshold: 1 + +livenessProbe: + enabled: true + initialDelaySeconds: 30 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + +readinessProbe: + enabled: true + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + +## Custom Startup probe +## +customStartupProbe: {} + +## Custom Liveness probe +## +customLivenessProbe: {} + +## Custom Rediness probe +## +customReadinessProbe: {} + +## +## TLS configuration +## +tls: + # Enable TLS traffic + enabled: false + # + # Whether to use the server's TLS cipher preferences rather than the client's. + preferServerCiphers: true + # + # Name of the Secret that contains the certificates + certificatesSecret: '' + # + # Certificate filename + certFilename: '' + # + # Certificate Key filename + certKeyFilename: '' + # + # CA Certificate filename + # If provided, PostgreSQL will authenticate TLS/SSL clients by requesting them a certificate + # ref: https://www.postgresql.org/docs/9.6/auth-methods.html + certCAFilename: + # + # File containing a Certificate Revocation List + crlFilename: + +## Configure metrics exporter +## +metrics: + enabled: false + # resources: {} + service: + type: ClusterIP + annotations: + prometheus.io/scrape: 'true' + prometheus.io/port: '9187' + loadBalancerIP: + serviceMonitor: + enabled: false + additionalLabels: {} + # namespace: monitoring + # interval: 30s + # scrapeTimeout: 10s + ## Custom PrometheusRule to be defined + ## The value is evaluated as a template, so, for example, the value can depend on .Release or .Chart + ## ref: https://github.com/coreos/prometheus-operator#customresourcedefinitions + ## + prometheusRule: + enabled: false + additionalLabels: {} + namespace: '' + ## These are just examples rules, please adapt them to your needs. + ## Make sure to constraint the rules to the current postgresql service. + ## rules: + ## - alert: HugeReplicationLag + ## expr: pg_replication_lag{service="{{ template "common.names.fullname" . }}-metrics"} / 3600 > 1 + ## for: 1m + ## labels: + ## severity: critical + ## annotations: + ## description: replication for {{ template "common.names.fullname" . }} PostgreSQL is lagging by {{ "{{ $value }}" }} hour(s). + ## summary: PostgreSQL replication is lagging by {{ "{{ $value }}" }} hour(s). + ## + rules: [] + + image: + registry: docker.io + repository: bitnami/postgres-exporter + tag: 0.9.0-debian-10-r43 + 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 + ## Define additional custom metrics + ## ref: https://github.com/wrouesnel/postgres_exporter#adding-new-metrics-via-a-config-file + # customMetrics: + # pg_database: + # query: "SELECT d.datname AS name, CASE WHEN pg_catalog.has_database_privilege(d.datname, 'CONNECT') THEN pg_catalog.pg_database_size(d.datname) ELSE 0 END AS size_bytes FROM pg_catalog.pg_database d where datname not in ('template0', 'template1', 'postgres')" + # metrics: + # - name: + # usage: "LABEL" + # description: "Name of the database" + # - size_bytes: + # usage: "GAUGE" + # description: "Size of the database in bytes" + # + ## An array to add extra env vars to configure postgres-exporter + ## see: https://github.com/wrouesnel/postgres_exporter#environment-variables + ## For example: + # extraEnvVars: + # - name: PG_EXPORTER_DISABLE_DEFAULT_METRICS + # value: "true" + extraEnvVars: {} + + ## Pod Security Context + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ + ## + securityContext: + enabled: false + runAsUser: 1001 + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-probes/#configure-probes) + ## Configure extra options for liveness and readiness probes + ## + livenessProbe: + enabled: true + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + + readinessProbe: + enabled: true + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + failureThreshold: 6 + successThreshold: 1 + +## Array with extra yaml to deploy with the chart. Evaluated as a template +## +extraDeploy: [] diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/access-tls-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/access-tls-values.yaml new file mode 100644 index 0000000000..1a8c4698d2 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/access-tls-values.yaml @@ -0,0 +1,24 @@ +databaseUpgradeReady: true +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" +access: + accessConfig: + security: + tls: true + resetAccessCAKeys: true diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/default-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/default-values.yaml new file mode 100644 index 0000000000..fc34693998 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/default-values.yaml @@ -0,0 +1,21 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/derby-test-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/derby-test-values.yaml new file mode 100644 index 0000000000..82ff485457 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/derby-test-values.yaml @@ -0,0 +1,19 @@ +databaseUpgradeReady: true + +postgresql: + enabled: false +artifactory: + podSecurityContext: + fsGroupChangePolicy: "OnRootMismatch" + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/global-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/global-values.yaml new file mode 100644 index 0000000000..33bbf04a20 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/global-values.yaml @@ -0,0 +1,247 @@ +databaseUpgradeReady: true +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + customInitContainersBegin: | + - name: "custom-init-begin-local" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in local" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: artifactory-volume + customInitContainers: | + - name: "custom-init-local" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in local" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: artifactory-volume + # Add custom volumes + customVolumes: | + - name: custom-script-local + emptyDir: + sizeLimit: 100Mi + # Add custom volumesMounts + customVolumeMounts: | + - name: custom-script-local + mountPath: "/scriptslocal" + # Add custom sidecar containers + customSidecarContainers: | + - name: "sidecar-list-local" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - NET_RAW + command: ["sh","-c","echo 'Sidecar is running in local' >> /scriptslocal/sidecarlocal.txt; cat /scriptslocal/sidecarlocal.txt; while true; do sleep 30; done"] + volumeMounts: + - mountPath: "/scriptslocal" + name: custom-script-local + resources: + requests: + memory: "32Mi" + cpu: "50m" + limits: + memory: "128Mi" + cpu: "100m" + +global: + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + customInitContainersBegin: | + - name: "custom-init-begin-global" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in global" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: artifactory-volume + customInitContainers: | + - name: "custom-init-global" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in global" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: artifactory-volume + # Add custom volumes + customVolumes: | + - name: custom-script-global + emptyDir: + sizeLimit: 100Mi + # Add custom volumesMounts + customVolumeMounts: | + - name: custom-script-global + mountPath: "/scripts" + # Add custom sidecar containers + customSidecarContainers: | + - name: "sidecar-list-global" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - NET_RAW + command: ["sh","-c","echo 'Sidecar is running in global' >> /scripts/sidecarglobal.txt; cat /scripts/sidecarglobal.txt; while true; do sleep 30; done"] + volumeMounts: + - mountPath: "/scripts" + name: custom-script-global + resources: + requests: + memory: "32Mi" + cpu: "50m" + limits: + memory: "128Mi" + cpu: "100m" + +nginx: + customInitContainers: | + - name: "custom-init-begin-nginx" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + command: + - 'sh' + - '-c' + - echo "running in nginx" + volumeMounts: + - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + name: custom-script-local + customSidecarContainers: | + - name: "sidecar-list-nginx" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + securityContext: + allowPrivilegeEscalation: false + capabilities: + drop: + - NET_RAW + command: ["sh","-c","echo 'Sidecar is running in local' >> /scriptslocal/sidecarlocal.txt; cat /scriptslocal/sidecarlocal.txt; while true; do sleep 30; done"] + volumeMounts: + - mountPath: "/scriptslocal" + name: custom-script-local + resources: + requests: + memory: "32Mi" + cpu: "50m" + limits: + memory: "128Mi" + cpu: "100m" + # Add custom volumes + customVolumes: | + - name: custom-script-local + emptyDir: + sizeLimit: 100Mi + + artifactoryConf: | + {{- if .Values.nginx.https.enabled }} + ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3; + ssl_certificate {{ .Values.nginx.persistence.mountPath }}/ssl/tls.crt; + ssl_certificate_key {{ .Values.nginx.persistence.mountPath }}/ssl/tls.key; + ssl_session_cache shared:SSL:1m; + ssl_prefer_server_ciphers on; + {{- end }} + ## server configuration + server { + listen 8088; + {{- if .Values.nginx.internalPortHttps }} + listen {{ .Values.nginx.internalPortHttps }} ssl; + {{- else -}} + {{- if .Values.nginx.https.enabled }} + listen {{ .Values.nginx.https.internalPort }} ssl; + {{- end }} + {{- end }} + {{- if .Values.nginx.internalPortHttp }} + listen {{ .Values.nginx.internalPortHttp }}; + {{- else -}} + {{- if .Values.nginx.http.enabled }} + listen {{ .Values.nginx.http.internalPort }}; + {{- end }} + {{- end }} + server_name ~(?.+)\.{{ include "artifactory.fullname" . }} {{ include "artifactory.fullname" . }} + {{- range .Values.ingress.hosts -}} + {{- if contains "." . -}} + {{ "" | indent 0 }} ~(?.+)\.{{ . }} + {{- end -}} + {{- end -}}; + + if ($http_x_forwarded_proto = '') { + set $http_x_forwarded_proto $scheme; + } + ## Application specific logs + ## access_log /var/log/nginx/artifactory-access.log timing; + ## error_log /var/log/nginx/artifactory-error.log; + rewrite ^/artifactory/?$ / redirect; + if ( $repo != "" ) { + rewrite ^/(v1|v2)/(.*) /artifactory/api/docker/$repo/$1/$2 break; + } + chunked_transfer_encoding on; + client_max_body_size 0; + + location / { + proxy_read_timeout 900; + proxy_pass_header Server; + proxy_cookie_path ~*^/.* /; + proxy_pass {{ include "artifactory.scheme" . }}://{{ include "artifactory.fullname" . }}:{{ .Values.artifactory.externalPort }}/; + {{- if .Values.nginx.service.ssloffload}} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host; + {{- else }} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host:$server_port; + proxy_set_header X-Forwarded-Port $server_port; + {{- end }} + proxy_set_header X-Forwarded-Proto $http_x_forwarded_proto; + proxy_set_header Host $http_host; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always; + + location /artifactory/ { + if ( $request_uri ~ ^/artifactory/(.*)$ ) { + proxy_pass {{ include "artifactory.scheme" . }}://{{ include "artifactory.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/$1; + } + proxy_pass {{ include "artifactory.scheme" . }}://{{ include "artifactory.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/; + } + } + } + + ## A list of custom ports to expose on the NGINX pod. Follows the conventional Kubernetes yaml syntax for container ports. + customPorts: + - containerPort: 8088 + name: http2 + service: + ## A list of custom ports to expose through the Ingress controller service. Follows the conventional Kubernetes yaml syntax for service ports. + customPorts: + - port: 8088 + targetPort: 8088 + protocol: TCP + name: http2 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/large-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/large-values.yaml new file mode 100644 index 0000000000..94a485d6f4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/large-values.yaml @@ -0,0 +1,82 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + database: + maxOpenConnections: 150 + tomcat: + connector: + maxThreads: 300 + resources: + requests: + memory: "6Gi" + cpu: "2" + limits: + memory: "10Gi" + cpu: "8" + javaOpts: + xms: "8g" + xmx: "10g" +access: + database: + maxOpenConnections: 150 + tomcat: + connector: + maxThreads: 100 +router: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + database: + maxOpenConnections: 150 + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +jfconnect: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/loggers-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/loggers-values.yaml new file mode 100644 index 0000000000..03c94be953 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/loggers-values.yaml @@ -0,0 +1,43 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + + loggers: + - access-audit.log + - access-request.log + - access-security-audit.log + - access-service.log + - artifactory-access.log + - artifactory-event.log + - artifactory-import-export.log + - artifactory-request.log + - artifactory-service.log + - frontend-request.log + - frontend-service.log + - metadata-request.log + - metadata-service.log + - router-request.log + - router-service.log + - router-traefik.log + + catalinaLoggers: + - tomcat-catalina.log + - tomcat-localhost.log diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/medium-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/medium-values.yaml new file mode 100644 index 0000000000..35044dc36d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/medium-values.yaml @@ -0,0 +1,82 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + database: + maxOpenConnections: 100 + tomcat: + connector: + maxThreads: 200 + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "8Gi" + cpu: "6" + javaOpts: + xms: "6g" + xmx: "8g" +access: + database: + maxOpenConnections: 100 + tomcat: + connector: + maxThreads: 50 +router: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + database: + maxOpenConnections: 100 + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +jfconnect: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "200Mi" + cpu: "200m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/migration-disabled-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/migration-disabled-values.yaml new file mode 100644 index 0000000000..092756fb65 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/migration-disabled-values.yaml @@ -0,0 +1,21 @@ +databaseUpgradeReady: true +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + migration: + enabled: false + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/nginx-autoreload-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/nginx-autoreload-values.yaml new file mode 100644 index 0000000000..09616c5bf4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/nginx-autoreload-values.yaml @@ -0,0 +1,42 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + +nginx: + customVolumes: | + - name: scripts + configMap: + name: {{ template "artifactory.fullname" . }}-nginx-scripts + defaultMode: 0550 + customVolumeMounts: | + - name: scripts + mountPath: /var/opt/jfrog/nginx/scripts/ + customCommand: + - /bin/sh + - -c + - | + # watch for configmap changes + /sbin/inotifyd /var/opt/jfrog/nginx/scripts/configreloader.sh {{ .Values.nginx.persistence.mountPath -}}/conf.d:n & + {{ if .Values.nginx.https.enabled -}} + # watch for tls secret changes + /sbin/inotifyd /var/opt/jfrog/nginx/scripts/configreloader.sh {{ .Values.nginx.persistence.mountPath -}}/ssl:n & + {{ end -}} + nginx -g 'daemon off;' diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values-access-tls-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values-access-tls-values.yaml new file mode 100644 index 0000000000..a38969a8ff --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values-access-tls-values.yaml @@ -0,0 +1,96 @@ +databaseUpgradeReady: true +artifactory: + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + +access: + accessConfig: + security: + tls: true + resetAccessCAKeys: true + +postgresql: + postgresqlPassword: password + postgresqlExtendedConf: + maxConnections: 102 + persistence: + enabled: false + +rbac: + create: true +serviceAccount: + create: true + automountServiceAccountToken: true + +ingress: + enabled: true + className: "testclass" + hosts: + - demonow.xyz +nginx: + enabled: false +jfconnect: + enabled: true + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +mc: + enabled: true +splitServicesToContainers: true + +router: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values.yaml new file mode 100644 index 0000000000..057ae9bf36 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/rtsplit-values.yaml @@ -0,0 +1,151 @@ +databaseUpgradeReady: true +artifactory: + replicaCount: 1 + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + + # Add lifecycle hooks for artifactory container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the artifactory postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the artifactory postStart handler >> /tmp/message"] + +postgresql: + postgresqlPassword: password + postgresqlExtendedConf: + maxConnections: 100 + persistence: + enabled: false + +rbac: + create: true +serviceAccount: + create: true + automountServiceAccountToken: true + +ingress: + enabled: true + className: "testclass" + hosts: + - demonow.xyz +nginx: + enabled: false +jfconnect: + enabled: true + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + # Add lifecycle hooks for jfconect container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the jfconnect postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the jfconnect postStart handler >> /tmp/message"] + + +mc: + enabled: true +splitServicesToContainers: true + +router: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + # Add lifecycle hooks for router container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the router postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the router postStart handler >> /tmp/message"] + +frontend: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + # Add lifecycle hooks for frontend container + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the frontend postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the frontend postStart handler >> /tmp/message"] + +metadata: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the metadata postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the metadata postStart handler >> /tmp/message"] + +event: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the event postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the event postStart handler >> /tmp/message"] + +observability: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" + lifecycle: + postStart: + exec: + command: ["/bin/sh", "-c", "echo Hello from the observability postStart handler >> /tmp/message"] + preStop: + exec: + command: ["/bin/sh", "-c", "echo Hello from the observability postStart handler >> /tmp/message"] diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/small-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/small-values.yaml new file mode 100644 index 0000000000..70d77790ab --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/small-values.yaml @@ -0,0 +1,82 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +databaseUpgradeReady: true + +# To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release +postgresql: + postgresqlPassword: password + persistence: + enabled: false +artifactory: + persistence: + enabled: false + database: + maxOpenConnections: 80 + tomcat: + connector: + maxThreads: 200 + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "6g" +access: + database: + maxOpenConnections: 80 + tomcat: + connector: + maxThreads: 50 +router: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +frontend: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +metadata: + database: + maxOpenConnections: 80 + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +event: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +jfconnect: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" +observability: + resources: + requests: + memory: "100Mi" + cpu: "100m" + limits: + memory: "1Gi" + cpu: "1" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/test-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/test-values.yaml new file mode 100644 index 0000000000..d2beb0eff6 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/ci/test-values.yaml @@ -0,0 +1,84 @@ +databaseUpgradeReady: true +artifactory: + replicaCount: 3 + joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + unifiedSecretInstallation: false + metrics: + enabled: true + persistence: + enabled: false + resources: + requests: + memory: "4Gi" + cpu: "2" + limits: + memory: "6Gi" + cpu: "4" + javaOpts: + xms: "4g" + xmx: "4g" + statefulset: + annotations: + artifactory: test + +postgresql: + postgresqlPassword: password + postgresqlExtendedConf: + maxConnections: 100 + persistence: + enabled: false + +rbac: + create: true +serviceAccount: + create: true + automountServiceAccountToken: true + +ingress: + enabled: true + className: "testclass" + hosts: + - demonow.xyz +nginx: + enabled: false + +jfconnect: + enabled: false + +autoscaling: + enabled: false + minReplicas: 1 + maxReplicas: 3 + targetCPUUtilizationPercentage: 70 + +## filebeat sidecar +filebeat: + enabled: true + filebeatYml: | + logging.level: info + path.data: {{ .Values.artifactory.persistence.mountPath }}/log/filebeat + name: artifactory-filebeat + queue.spool: + file: + permissions: 0760 + filebeat.inputs: + - type: log + enabled: true + close_eof: ${CLOSE:false} + paths: + - {{ .Values.artifactory.persistence.mountPath }}/log/*.log + fields: + service: "jfrt" + log_type: "artifactory" + output.file: + path: "/tmp/filebeat" + filename: filebeat + readinessProbe: + exec: + command: + - sh + - -c + - | + #!/usr/bin/env bash -e + curl --fail 127.0.0.1:5066 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/binarystore.xml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/binarystore.xml new file mode 100644 index 0000000000..e396e0a418 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/binarystore.xml @@ -0,0 +1,426 @@ +{{- if eq .Values.artifactory.persistence.type "nfs" -}} + + {{- if (.Values.artifactory.persistence.maxCacheSize) }} + + + + + + {{- else }} + + + + {{- end }} + + {{- if .Values.artifactory.persistence.maxCacheSize }} + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + {{- end }} + + + {{ .Values.artifactory.persistence.nfs.dataDir }}/filestore + + +{{- end }} +{{- if eq .Values.artifactory.persistence.type "file-system" -}} + + + + {{- if .Values.artifactory.persistence.fileSystem.cache.enabled }} + + {{- end }} + + {{- if .Values.artifactory.persistence.fileSystem.cache.enabled }} + + {{- end }} + + + {{- if .Values.artifactory.persistence.fileSystem.cache.enabled }} + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + {{- end }} + +{{- end }} +{{- if eq .Values.artifactory.persistence.type "cluster-file-system" -}} + + + + + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + {{ .Values.artifactory.persistence.lenientLimit }} + 2 + + + + + + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + + + shard-fs-1 + local + + + + + 30 + tester-remote1 + 10000 + remote + + + +{{- end }} +{{- if or (eq .Values.artifactory.persistence.type "google-storage") (eq .Values.artifactory.persistence.type "google-storage-v2") (eq .Values.artifactory.persistence.type "cluster-google-storage-v2") (eq .Values.artifactory.persistence.type "google-storage-v2-direct") }} + + + {{- if or (eq .Values.artifactory.persistence.type "google-storage") (eq .Values.artifactory.persistence.type "google-storage-v2") }} + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "cluster-google-storage-v2" }} + + + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + {{ .Values.artifactory.persistence.lenientLimit }} + 2 + + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "google-storage-v2-direct" }} + + + + + + {{- end }} + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + {{- if eq .Values.artifactory.persistence.type "cluster-google-storage-v2" }} + + local + + + + 30 + 10000 + remote + + {{- end }} + + + {{- if .Values.artifactory.persistence.googleStorage.useInstanceCredentials }} + true + {{- else }} + false + {{- end }} + {{ .Values.artifactory.persistence.googleStorage.enableSignedUrlRedirect }} + google-cloud-storage + {{ .Values.artifactory.persistence.googleStorage.endpoint }} + {{ .Values.artifactory.persistence.googleStorage.httpsOnly }} + {{ .Values.artifactory.persistence.googleStorage.bucketName }} + {{ .Values.artifactory.persistence.googleStorage.path }} + {{ .Values.artifactory.persistence.googleStorage.bucketExists }} + + +{{- end }} +{{- if or (eq .Values.artifactory.persistence.type "aws-s3-v3") (eq .Values.artifactory.persistence.type "s3-storage-v3-direct") (eq .Values.artifactory.persistence.type "cluster-s3-storage-v3") (eq .Values.artifactory.persistence.type "s3-storage-v3-archive") }} + + + {{- if eq .Values.artifactory.persistence.type "aws-s3-v3" }} + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "s3-storage-v3-direct" }} + + + + + + {{- else if eq .Values.artifactory.persistence.type "cluster-s3-storage-v3" }} + + + + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "s3-storage-v3-archive" }} + + + + + + + {{- end }} + + {{- if or (eq .Values.artifactory.persistence.type "aws-s3-v3") (eq .Values.artifactory.persistence.type "s3-storage-v3-direct") (eq .Values.artifactory.persistence.type "cluster-s3-storage-v3") }} + + + {{ .Values.artifactory.persistence.maxCacheSize | int64}} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + {{- end }} + + {{- if eq .Values.artifactory.persistence.type "cluster-s3-storage-v3" }} + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + {{ .Values.artifactory.persistence.lenientLimit }} + + + + + remote + + + + local + + {{- end }} + + {{- with .Values.artifactory.persistence.awsS3V3 }} + + {{ .testConnection }} + {{- if .identity }} + {{ .identity }} + {{- end }} + {{- if .credential }} + {{ .credential }} + {{- end }} + {{ .region }} + {{ .bucketName }} + {{ .path }} + {{ .endpoint }} + {{- with .port }} + {{ . }} + {{- end }} + {{- with .useHttp }} + {{ . }} + {{- end }} + {{- with .maxConnections }} + {{ . }} + {{- end }} + {{- with .connectionTimeout }} + {{ . }} + {{- end }} + {{- with .socketTimeout }} + {{ . }} + {{- end }} + {{- with .kmsServerSideEncryptionKeyId }} + {{ . }} + {{- end }} + {{- with .kmsKeyRegion }} + {{ . }} + {{- end }} + {{- with .kmsCryptoMode }} + {{ . }} + {{- end }} + {{- if .useInstanceCredentials }} + true + {{- else }} + false + {{- end }} + {{ .usePresigning }} + {{ .signatureExpirySeconds }} + {{ .signedUrlExpirySeconds }} + {{- with .cloudFrontDomainName }} + {{ . }} + {{- end }} + {{- with .cloudFrontKeyPairId }} + {{ . }} + {{- end }} + {{- with .cloudFrontPrivateKey }} + {{ . }} + {{- end }} + {{- with .enableSignedUrlRedirect }} + {{ . }} + {{- end }} + {{- with .enablePathStyleAccess }} + {{ . }} + {{- end }} + {{- with .multiPartLimit }} + {{ . | int64 }} + {{- end }} + {{- with .multipartElementSize }} + {{ . | int64 }} + {{- end }} + + {{- end }} + +{{- end }} + +{{- if or (eq .Values.artifactory.persistence.type "azure-blob") (eq .Values.artifactory.persistence.type "azure-blob-storage-direct") (eq .Values.artifactory.persistence.type "cluster-azure-blob-storage") }} + + + {{- if or (eq .Values.artifactory.persistence.type "azure-blob") }} + + + + + + + + + + {{- else if eq .Values.artifactory.persistence.type "azure-blob-storage-direct" }} + + + + + + {{- else if eq .Values.artifactory.persistence.type "cluster-azure-blob-storage" }} + + + + + + + + + + + + + {{- end }} + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + {{- if eq .Values.artifactory.persistence.type "cluster-azure-blob-storage" }} + + + crossNetworkStrategy + crossNetworkStrategy + {{ .Values.artifactory.persistence.redundancy }} + {{ .Values.artifactory.persistence.lenientLimit }} + + + + remote + + + local + + {{- end }} + + + {{ .Values.artifactory.persistence.azureBlob.accountName }} + {{ .Values.artifactory.persistence.azureBlob.accountKey }} + {{ .Values.artifactory.persistence.azureBlob.endpoint }} + {{ .Values.artifactory.persistence.azureBlob.containerName }} + {{ .Values.artifactory.persistence.azureBlob.multiPartLimit | int64 }} + {{ .Values.artifactory.persistence.azureBlob.multipartElementSize | int64 }} + {{ .Values.artifactory.persistence.azureBlob.testConnection }} + + +{{- end }} +{{- if eq .Values.artifactory.persistence.type "azure-blob-storage-v2-direct" -}} + + + + {{ .Values.artifactory.persistence.maxCacheSize | int64 }} + {{ .Values.artifactory.persistence.cacheProviderDir }} + {{- if .Values.artifactory.persistence.maxFileSizeLimit }} + {{.Values.artifactory.persistence.maxFileSizeLimit | int64}} + {{- end }} + {{- if .Values.artifactory.persistence.skipDuringUpload }} + {{.Values.artifactory.persistence.skipDuringUpload}} + {{- end }} + + + {{ .Values.artifactory.persistence.azureBlob.accountName }} + {{ .Values.artifactory.persistence.azureBlob.accountKey }} + {{ .Values.artifactory.persistence.azureBlob.endpoint }} + {{ .Values.artifactory.persistence.azureBlob.containerName }} + {{ .Values.artifactory.persistence.azureBlob.multiPartLimit | int64 }} + {{ .Values.artifactory.persistence.azureBlob.multipartElementSize | int64 }} + {{ .Values.artifactory.persistence.azureBlob.testConnection }} + + +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/installer-info.json b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/installer-info.json new file mode 100644 index 0000000000..79f42ed16d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/installer-info.json @@ -0,0 +1,32 @@ +{ + "productId": "Helm_artifactory/{{ .Chart.Version }}", + "features": [ + { + "featureId": "Platform/{{ printf "%s-%s" "kubernetes" .Capabilities.KubeVersion.Version }}" + }, + { + "featureId": "Database/{{ .Values.database.type }}" + }, + { + "featureId": "PostgreSQL_Enabled/{{ .Values.postgresql.enabled }}" + }, + { + "featureId": "Nginx_Enabled/{{ .Values.nginx.enabled }}" + }, + { + "featureId": "ArtifactoryPersistence_Type/{{ .Values.artifactory.persistence.type }}" + }, + { + "featureId": "SplitServicesToContainers_Enabled/{{ .Values.splitServicesToContainers }}" + }, + { + "featureId": "UnifiedSecretInstallation_Enabled/{{ .Values.artifactory.unifiedSecretInstallation }}" + }, + { + "featureId": "Filebeat_Enabled/{{ .Values.filebeat.enabled }}" + }, + { + "featureId": "ReplicaCount/{{ .Values.artifactory.replicaCount }}" + } + ] +} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/migrate.sh b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/migrate.sh new file mode 100644 index 0000000000..ba44160f47 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/migrate.sh @@ -0,0 +1,4311 @@ +#!/bin/bash + +# Flags +FLAG_Y="y" +FLAG_N="n" +FLAGS_Y_N="$FLAG_Y $FLAG_N" +FLAG_NOT_APPLICABLE="_NA_" + +CURRENT_VERSION=$1 + +WRAPPER_SCRIPT_TYPE_RPMDEB="RPMDEB" +WRAPPER_SCRIPT_TYPE_DOCKER_COMPOSE="DOCKERCOMPOSE" + +SENSITIVE_KEY_VALUE="__sensitive_key_hidden___" + +# Shared system keys +SYS_KEY_SHARED_JFROGURL="shared.jfrogUrl" +SYS_KEY_SHARED_SECURITY_JOINKEY="shared.security.joinKey" +SYS_KEY_SHARED_SECURITY_MASTERKEY="shared.security.masterKey" + +SYS_KEY_SHARED_NODE_ID="shared.node.id" +SYS_KEY_SHARED_JAVAHOME="shared.javaHome" + +SYS_KEY_SHARED_DATABASE_TYPE="shared.database.type" +SYS_KEY_SHARED_DATABASE_TYPE_VALUE_POSTGRES="postgresql" +SYS_KEY_SHARED_DATABASE_DRIVER="shared.database.driver" +SYS_KEY_SHARED_DATABASE_URL="shared.database.url" +SYS_KEY_SHARED_DATABASE_USERNAME="shared.database.username" +SYS_KEY_SHARED_DATABASE_PASSWORD="shared.database.password" + +SYS_KEY_SHARED_ELASTICSEARCH_URL="shared.elasticsearch.url" +SYS_KEY_SHARED_ELASTICSEARCH_USERNAME="shared.elasticsearch.username" +SYS_KEY_SHARED_ELASTICSEARCH_PASSWORD="shared.elasticsearch.password" +SYS_KEY_SHARED_ELASTICSEARCH_CLUSTERSETUP="shared.elasticsearch.clusterSetup" +SYS_KEY_SHARED_ELASTICSEARCH_UNICASTFILE="shared.elasticsearch.unicastFile" +SYS_KEY_SHARED_ELASTICSEARCH_CLUSTERSETUP_VALUE="YES" + +# Define this in product specific script. Should contain the path to unitcast file +# File used by insight server to write cluster active nodes info. This will be read by elasticsearch +#SYS_KEY_SHARED_ELASTICSEARCH_UNICASTFILE_VALUE="" + +SYS_KEY_RABBITMQ_ACTIVE_NODE_NAME="shared.rabbitMq.active.node.name" +SYS_KEY_RABBITMQ_ACTIVE_NODE_IP="shared.rabbitMq.active.node.ip" + +# Filenames +FILE_NAME_SYSTEM_YAML="system.yaml" +FILE_NAME_JOIN_KEY="join.key" +FILE_NAME_MASTER_KEY="master.key" +FILE_NAME_INSTALLER_YAML="installer.yaml" + +# Global constants used in business logic +NODE_TYPE_STANDALONE="standalone" +NODE_TYPE_CLUSTER_NODE="node" +NODE_TYPE_DATABASE="database" + +# External(isable) databases +DATABASE_POSTGRES="POSTGRES" +DATABASE_ELASTICSEARCH="ELASTICSEARCH" +DATABASE_RABBITMQ="RABBITMQ" + +POSTGRES_LABEL="PostgreSQL" +ELASTICSEARCH_LABEL="Elasticsearch" +RABBITMQ_LABEL="Rabbitmq" + +ARTIFACTORY_LABEL="Artifactory" +JFMC_LABEL="Mission Control" +DISTRIBUTION_LABEL="Distribution" +XRAY_LABEL="Xray" + +POSTGRES_CONTAINER="postgres" +ELASTICSEARCH_CONTAINER="elasticsearch" +RABBITMQ_CONTAINER="rabbitmq" +REDIS_CONTAINER="redis" + +#Adding a small timeout before a read ensures it is positioned correctly in the screen +read_timeout=0.5 + +# Options related to data directory location +PROMPT_DATA_DIR_LOCATION="Installation Directory" +KEY_DATA_DIR_LOCATION="installer.data_dir" + +SYS_KEY_SHARED_NODE_HAENABLED="shared.node.haEnabled" +PROMPT_ADD_TO_CLUSTER="Are you adding an additional node to an existing product cluster?" +KEY_ADD_TO_CLUSTER="installer.ha" +VALID_VALUES_ADD_TO_CLUSTER="$FLAGS_Y_N" + +MESSAGE_POSTGRES_INSTALL="The installer can install a $POSTGRES_LABEL database, or you can connect to an existing compatible $POSTGRES_LABEL database\n(compatible databases: https://www.jfrog.com/confluence/display/JFROG/System+Requirements#SystemRequirements-RequirementsMatrix)" +PROMPT_POSTGRES_INSTALL="Do you want to install $POSTGRES_LABEL?" +KEY_POSTGRES_INSTALL="installer.install_postgresql" +VALID_VALUES_POSTGRES_INSTALL="$FLAGS_Y_N" + +# Postgres connection details +RPM_DEB_POSTGRES_HOME_DEFAULT="/var/opt/jfrog/postgres" +RPM_DEB_MESSAGE_STANDALONE_POSTGRES_DATA="$POSTGRES_LABEL home will have data and its configuration" +RPM_DEB_PROMPT_STANDALONE_POSTGRES_DATA="Type desired $POSTGRES_LABEL home location" +RPM_DEB_KEY_STANDALONE_POSTGRES_DATA="installer.postgresql.home" + +MESSAGE_DATABASE_URL="Provide the database connection details" +PROMPT_DATABASE_URL(){ + local databaseURlExample= + case "$PRODUCT_NAME" in + $ARTIFACTORY_LABEL) + databaseURlExample="jdbc:postgresql://:/artifactory" + ;; + $JFMC_LABEL) + databaseURlExample="postgresql://:/mission_control?sslmode=disable" + ;; + $DISTRIBUTION_LABEL) + databaseURlExample="jdbc:postgresql://:/distribution?sslmode=disable" + ;; + $XRAY_LABEL) + databaseURlExample="postgres://:/xraydb?sslmode=disable" + ;; + esac + if [ -z "$databaseURlExample" ]; then + echo -n "$POSTGRES_LABEL URL" # For consistency with username and password + return + fi + echo -n "$POSTGRES_LABEL url. Example: [$databaseURlExample]" +} +REGEX_DATABASE_URL(){ + local databaseURlExample= + case "$PRODUCT_NAME" in + $ARTIFACTORY_LABEL) + databaseURlExample="jdbc:postgresql://.*/artifactory.*" + ;; + $JFMC_LABEL) + databaseURlExample="postgresql://.*/mission_control.*" + ;; + $DISTRIBUTION_LABEL) + databaseURlExample="jdbc:postgresql://.*/distribution.*" + ;; + $XRAY_LABEL) + databaseURlExample="postgres://.*/xraydb.*" + ;; + esac + echo -n "^$databaseURlExample\$" +} +ERROR_MESSAGE_DATABASE_URL="Invalid $POSTGRES_LABEL URL" +KEY_DATABASE_URL="$SYS_KEY_SHARED_DATABASE_URL" +#NOTE: It is important to display the label. Since the message may be hidden if URL is known +PROMPT_DATABASE_USERNAME="$POSTGRES_LABEL username" +KEY_DATABASE_USERNAME="$SYS_KEY_SHARED_DATABASE_USERNAME" +#NOTE: It is important to display the label. Since the message may be hidden if URL is known +PROMPT_DATABASE_PASSWORD="$POSTGRES_LABEL password" +KEY_DATABASE_PASSWORD="$SYS_KEY_SHARED_DATABASE_PASSWORD" +IS_SENSITIVE_DATABASE_PASSWORD="$FLAG_Y" + +MESSAGE_STANDALONE_ELASTICSEARCH_INSTALL="The installer can install a $ELASTICSEARCH_LABEL database or you can connect to an existing compatible $ELASTICSEARCH_LABEL database" +PROMPT_STANDALONE_ELASTICSEARCH_INSTALL="Do you want to install $ELASTICSEARCH_LABEL?" +KEY_STANDALONE_ELASTICSEARCH_INSTALL="installer.install_elasticsearch" +VALID_VALUES_STANDALONE_ELASTICSEARCH_INSTALL="$FLAGS_Y_N" + +# Elasticsearch connection details +MESSAGE_ELASTICSEARCH_DETAILS="Provide the $ELASTICSEARCH_LABEL connection details" +PROMPT_ELASTICSEARCH_URL="$ELASTICSEARCH_LABEL URL" +KEY_ELASTICSEARCH_URL="$SYS_KEY_SHARED_ELASTICSEARCH_URL" + +PROMPT_ELASTICSEARCH_USERNAME="$ELASTICSEARCH_LABEL username" +KEY_ELASTICSEARCH_USERNAME="$SYS_KEY_SHARED_ELASTICSEARCH_USERNAME" + +PROMPT_ELASTICSEARCH_PASSWORD="$ELASTICSEARCH_LABEL password" +KEY_ELASTICSEARCH_PASSWORD="$SYS_KEY_SHARED_ELASTICSEARCH_PASSWORD" +IS_SENSITIVE_ELASTICSEARCH_PASSWORD="$FLAG_Y" + +# Cluster related questions +MESSAGE_CLUSTER_MASTER_KEY="Provide the cluster's master key. It can be found in the data directory of the first node under /etc/security/master.key" +PROMPT_CLUSTER_MASTER_KEY="Master Key" +KEY_CLUSTER_MASTER_KEY="$SYS_KEY_SHARED_SECURITY_MASTERKEY" +IS_SENSITIVE_CLUSTER_MASTER_KEY="$FLAG_Y" + +MESSAGE_JOIN_KEY="The Join key is the secret key used to establish trust between services in the JFrog Platform.\n(You can copy the Join Key from Admin > User Management > Settings)" +PROMPT_JOIN_KEY="Join Key" +KEY_JOIN_KEY="$SYS_KEY_SHARED_SECURITY_JOINKEY" +IS_SENSITIVE_JOIN_KEY="$FLAG_Y" +REGEX_JOIN_KEY="^[a-zA-Z0-9]{16,}\$" +ERROR_MESSAGE_JOIN_KEY="Invalid Join Key" + +# Rabbitmq related cluster information +MESSAGE_RABBITMQ_ACTIVE_NODE_NAME="Provide an active ${RABBITMQ_LABEL} node name. Run the command [ hostname -s ] on any of the existing nodes in the product cluster to get this" +PROMPT_RABBITMQ_ACTIVE_NODE_NAME="${RABBITMQ_LABEL} active node name" +KEY_RABBITMQ_ACTIVE_NODE_NAME="$SYS_KEY_RABBITMQ_ACTIVE_NODE_NAME" + +# Rabbitmq related cluster information (necessary only for docker-compose) +PROMPT_RABBITMQ_ACTIVE_NODE_IP="${RABBITMQ_LABEL} active node ip" +KEY_RABBITMQ_ACTIVE_NODE_IP="$SYS_KEY_RABBITMQ_ACTIVE_NODE_IP" + +MESSAGE_JFROGURL(){ + echo -e "The JFrog URL allows ${PRODUCT_NAME} to connect to a JFrog Platform Instance.\n(You can copy the JFrog URL from Administration > User Management > Settings > Connection details)" +} +PROMPT_JFROGURL="JFrog URL" +KEY_JFROGURL="$SYS_KEY_SHARED_JFROGURL" +REGEX_JFROGURL="^https?://.*:{0,}[0-9]{0,4}\$" +ERROR_MESSAGE_JFROGURL="Invalid JFrog URL" + + +# Set this to FLAG_Y on upgrade +IS_UPGRADE="${FLAG_N}" + +# This belongs in JFMC but is the ONLY one that needs it so keeping it here for now. Can be made into a method and overridden if necessary +MESSAGE_MULTIPLE_PG_SCHEME="Please setup $POSTGRES_LABEL with schema as described in https://www.jfrog.com/confluence/display/JFROG/Installing+Mission+Control" + +_getMethodOutputOrVariableValue() { + unset EFFECTIVE_MESSAGE + local keyToSearch=$1 + local effectiveMessage= + local result="0" + # logSilly "Searching for method: [$keyToSearch]" + LC_ALL=C type "$keyToSearch" > /dev/null 2>&1 || result="$?" + if [[ "$result" == "0" ]]; then + # logSilly "Found method for [$keyToSearch]" + EFFECTIVE_MESSAGE="$($keyToSearch)" + return + fi + eval EFFECTIVE_MESSAGE=\${$keyToSearch} + if [ ! -z "$EFFECTIVE_MESSAGE" ]; then + return + fi + # logSilly "Didn't find method or variable for [$keyToSearch]" +} + + +# REF https://misc.flogisoft.com/bash/tip_colors_and_formatting +cClear="\e[0m" +cBlue="\e[38;5;69m" +cRedDull="\e[1;31m" +cYellow="\e[1;33m" +cRedBright="\e[38;5;197m" +cBold="\e[1m" + + +_loggerGetModeRaw() { + local MODE="$1" + case $MODE in + INFO) + printf "" + ;; + DEBUG) + printf "%s" "[${MODE}] " + ;; + WARN) + printf "${cRedDull}%s%s${cClear}" "[" "${MODE}" "] " + ;; + ERROR) + printf "${cRedBright}%s%s${cClear}" "[" "${MODE}" "] " + ;; + esac +} + + +_loggerGetMode() { + local MODE="$1" + case $MODE in + INFO) + printf "${cBlue}%s%-5s%s${cClear}" "[" "${MODE}" "]" + ;; + DEBUG) + printf "%-7s" "[${MODE}]" + ;; + WARN) + printf "${cRedDull}%s%-5s%s${cClear}" "[" "${MODE}" "]" + ;; + ERROR) + printf "${cRedBright}%s%-5s%s${cClear}" "[" "${MODE}" "]" + ;; + esac +} + +# Capitalises the first letter of the message +_loggerGetMessage() { + local originalMessage="$*" + local firstChar=$(echo "${originalMessage:0:1}" | awk '{ print toupper($0) }') + local resetOfMessage="${originalMessage:1}" + echo "$firstChar$resetOfMessage" +} + +# The spec also says content should be left-trimmed but this is not necessary in our case. We don't reach the limit. +_loggerGetStackTrace() { + printf "%s%-30s%s" "[" "$1:$2" "]" +} + +_loggerGetThread() { + printf "%s" "[main]" +} + +_loggerGetServiceType() { + printf "%s%-5s%s" "[" "shell" "]" +} + +#Trace ID is not applicable to scripts +_loggerGetTraceID() { + printf "%s" "[]" +} + +logRaw() { + echo "" + printf "$1" + echo "" +} + +logBold(){ + echo "" + printf "${cBold}$1${cClear}" + echo "" +} + +# The date binary works differently based on whether it is GNU/BSD +is_date_supported=0 +date --version > /dev/null 2>&1 || is_date_supported=1 +IS_GNU=$(echo $is_date_supported) + +_loggerGetTimestamp() { + if [ "${IS_GNU}" == "0" ]; then + echo -n $(date -u +%FT%T.%3NZ) + else + echo -n $(date -u +%FT%T.000Z) + fi +} + +# https://www.shellscript.sh/tips/spinner/ +_spin() +{ + spinner="/|\\-/|\\-" + while : + do + for i in `seq 0 7` + do + echo -n "${spinner:$i:1}" + echo -en "\010" + sleep 1 + done + done +} + +showSpinner() { + # Start the Spinner: + _spin & + # Make a note of its Process ID (PID): + SPIN_PID=$! + # Kill the spinner on any signal, including our own exit. + trap "kill -9 $SPIN_PID" `seq 0 15` &> /dev/null || return 0 +} + +stopSpinner() { + local occurrences=$(ps -ef | grep -wc "${SPIN_PID}") + let "occurrences+=0" + # validate that it is present (2 since this search itself will show up in the results) + if [ $occurrences -gt 1 ]; then + kill -9 $SPIN_PID &>/dev/null || return 0 + wait $SPIN_ID &>/dev/null + fi +} + +_getEffectiveMessage(){ + local MESSAGE="$1" + local MODE=${2-"INFO"} + + if [ -z "$CONTEXT" ]; then + CONTEXT=$(caller) + fi + + _EFFECTIVE_MESSAGE= + if [ -z "$LOG_BEHAVIOR_ADD_META" ]; then + _EFFECTIVE_MESSAGE="$(_loggerGetModeRaw $MODE)$(_loggerGetMessage $MESSAGE)" + else + local SERVICE_TYPE="script" + local TRACE_ID="" + local THREAD="main" + + local CONTEXT_LINE=$(echo "$CONTEXT" | awk '{print $1}') + local CONTEXT_FILE=$(echo "$CONTEXT" | awk -F"/" '{print $NF}') + + _EFFECTIVE_MESSAGE="$(_loggerGetTimestamp) $(_loggerGetServiceType) $(_loggerGetMode $MODE) $(_loggerGetTraceID) $(_loggerGetStackTrace $CONTEXT_FILE $CONTEXT_LINE) $(_loggerGetThread) - $(_loggerGetMessage $MESSAGE)" + fi + CONTEXT= +} + +# Important - don't call any log method from this method. Will become an infinite loop. Use echo to debug +_logToFile() { + local MODE=${1-"INFO"} + local targetFile="$LOG_BEHAVIOR_ADD_REDIRECTION" + # IF the file isn't passed, abort + if [ -z "$targetFile" ]; then + return + fi + # IF this is not being run in verbose mode and mode is debug or lower, abort + if [ "${VERBOSE_MODE}" != "$FLAG_Y" ] && [ "${VERBOSE_MODE}" != "true" ] && [ "${VERBOSE_MODE}" != "debug" ]; then + if [ "$MODE" == "DEBUG" ] || [ "$MODE" == "SILLY" ]; then + return + fi + fi + + # Create the file if it doesn't exist + if [ ! -f "${targetFile}" ]; then + return + # touch $targetFile > /dev/null 2>&1 || true + fi + # # Make it readable + # chmod 640 $targetFile > /dev/null 2>&1 || true + + # Log contents + printf "%s\n" "$_EFFECTIVE_MESSAGE" >> "$targetFile" || true +} + +logger() { + if [ "$LOG_BEHAVIOR_ADD_NEW_LINE" == "$FLAG_Y" ]; then + echo "" + fi + _getEffectiveMessage "$@" + local MODE=${2-"INFO"} + printf "%s\n" "$_EFFECTIVE_MESSAGE" + _logToFile "$MODE" +} + +logDebug(){ + VERBOSE_MODE=${VERBOSE_MODE-"false"} + CONTEXT=$(caller) + if [ "${VERBOSE_MODE}" == "$FLAG_Y" ] || [ "${VERBOSE_MODE}" == "true" ] || [ "${VERBOSE_MODE}" == "debug" ];then + logger "$1" "DEBUG" + else + logger "$1" "DEBUG" >&6 + fi + CONTEXT= +} + +logSilly(){ + VERBOSE_MODE=${VERBOSE_MODE-"false"} + CONTEXT=$(caller) + if [ "${VERBOSE_MODE}" == "silly" ];then + logger "$1" "DEBUG" + else + logger "$1" "DEBUG" >&6 + fi + CONTEXT= +} + +logError() { + CONTEXT=$(caller) + logger "$1" "ERROR" + CONTEXT= +} + +errorExit () { + CONTEXT=$(caller) + logger "$1" "ERROR" + CONTEXT= + exit 1 +} + +warn () { + CONTEXT=$(caller) + logger "$1" "WARN" + CONTEXT= +} + +note () { + CONTEXT=$(caller) + logger "$1" "NOTE" + CONTEXT= +} + +bannerStart() { + title=$1 + echo + echo -e "\033[1m${title}\033[0m" + echo +} + +bannerSection() { + title=$1 + echo + echo -e "******************************** ${title} ********************************" + echo +} + +bannerSubSection() { + title=$1 + echo + echo -e "************** ${title} *******************" + echo +} + +bannerMessge() { + title=$1 + echo + echo -e "********************************" + echo -e "${title}" + echo -e "********************************" + echo +} + +setRed () { + local input="$1" + echo -e \\033[31m${input}\\033[0m +} +setGreen () { + local input="$1" + echo -e \\033[32m${input}\\033[0m +} +setYellow () { + local input="$1" + echo -e \\033[33m${input}\\033[0m +} + +logger_addLinebreak () { + echo -e "---\n" +} + +bannerImportant() { + title=$1 + local bold="\033[1m" + local noColour="\033[0m" + echo + echo -e "${bold}######################################## IMPORTANT ########################################${noColour}" + echo -e "${bold}${title}${noColour}" + echo -e "${bold}###########################################################################################${noColour}" + echo +} + +bannerEnd() { + #TODO pass a title and calculate length dynamically so that start and end look alike + echo + echo "*****************************************************************************" + echo +} + +banner() { + title=$1 + content=$2 + bannerStart "${title}" + echo -e "$content" +} + +# The logic below helps us redirect content we'd normally hide to the log file. + # + # We have several commands which clutter the console with output and so use + # `cmd > /dev/null` - this redirects the command's output to null. + # + # However, the information we just hid maybe useful for support. Using the code pattern + # `cmd >&6` (instead of `cmd> >/dev/null` ), the command's output is hidden from the console + # but redirected to the installation log file + # + +#Default value of 6 is just null +exec 6>>/dev/null +redirectLogsToFile() { + echo "" + # local file=$1 + + # [ ! -z "${file}" ] || return 0 + + # local logDir=$(dirname "$file") + + # if [ ! -f "${file}" ]; then + # [ -d "${logDir}" ] || mkdir -p ${logDir} || \ + # ( echo "WARNING : Could not create parent directory (${logDir}) to redirect console log : ${file}" ; return 0 ) + # fi + + # #6 now points to the log file + # exec 6>>${file} + # #reference https://unix.stackexchange.com/questions/145651/using-exec-and-tee-to-redirect-logs-to-stdout-and-a-log-file-in-the-same-time + # exec 2>&1 > >(tee -a "${file}") +} + +# Check if a give key contains any sensitive string as part of it +# Based on the result, the caller can decide its value can be displayed or not +# Sample usage : isKeySensitive "${key}" && displayValue="******" || displayValue=${value} +isKeySensitive(){ + local key=$1 + local sensitiveKeys="password|secret|key|token" + + if [ -z "${key}" ]; then + return 1 + else + local lowercaseKey=$(echo "${key}" | tr '[:upper:]' '[:lower:]' 2>/dev/null) + [[ "${lowercaseKey}" =~ ${sensitiveKeys} ]] && return 0 || return 1 + fi +} + +getPrintableValueOfKey(){ + local displayValue= + local key="$1" + if [ -z "$key" ]; then + # This is actually an incorrect usage of this method but any logging will cause unexpected content in the caller + echo -n "" + return + fi + + local value="$2" + isKeySensitive "${key}" && displayValue="$SENSITIVE_KEY_VALUE" || displayValue="${value}" + echo -n $displayValue +} + +_createConsoleLog(){ + if [ -z "${JF_PRODUCT_HOME}" ]; then + return + fi + local targetFile="${JF_PRODUCT_HOME}/var/log/console.log" + mkdir -p "${JF_PRODUCT_HOME}/var/log" || true + if [ ! -f ${targetFile} ]; then + touch $targetFile > /dev/null 2>&1 || true + fi + chmod 640 $targetFile > /dev/null 2>&1 || true +} + +# Output from application's logs are piped to this method. It checks a configuration variable to determine if content should be logged to +# the common console.log file +redirectServiceLogsToFile() { + + local result="0" + # check if the function getSystemValue exists + LC_ALL=C type getSystemValue > /dev/null 2>&1 || result="$?" + if [[ "$result" != "0" ]]; then + warn "Couldn't find the systemYamlHelper. Skipping log redirection" + return 0 + fi + + getSystemValue "shared.consoleLog" "NOT_SET" + if [[ "${YAML_VALUE}" == "false" ]]; then + logger "Redirection is set to false. Skipping log redirection" + return 0; + fi + + if [ -z "${JF_PRODUCT_HOME}" ] || [ "${JF_PRODUCT_HOME}" == "" ]; then + warn "JF_PRODUCT_HOME is unavailable. Skipping log redirection" + return 0 + fi + + local targetFile="${JF_PRODUCT_HOME}/var/log/console.log" + + _createConsoleLog + + while read -r line; do + printf '%s\n' "${line}" >> $targetFile || return 0 # Don't want to log anything - might clutter the screen + done +} + +## Display environment variables starting with JF_ along with its value +## Value of sensitive keys will be displayed as "******" +## +## Sample Display : +## +## ======================== +## JF Environment variables +## ======================== +## +## JF_SHARED_NODE_ID : locahost +## JF_SHARED_JOINKEY : ****** +## +## +displayEnv() { + local JFEnv=$(printenv | grep ^JF_ 2>/dev/null) + local key= + local value= + + if [ -z "${JFEnv}" ]; then + return + fi + + cat << ENV_START_MESSAGE + +======================== +JF Environment variables +======================== +ENV_START_MESSAGE + + for entry in ${JFEnv}; do + key=$(echo "${entry}" | awk -F'=' '{print $1}') + value=$(echo "${entry}" | awk -F'=' '{print $2}') + + isKeySensitive "${key}" && value="******" || value=${value} + + printf "\n%-35s%s" "${key}" " : ${value}" + done + echo; +} + +_addLogRotateConfiguration() { + logDebug "Method ${FUNCNAME[0]}" + # mandatory inputs + local confFile="$1" + local logFile="$2" + + # Method available in _ioOperations.sh + LC_ALL=C type io_setYQPath > /dev/null 2>&1 || return 1 + + io_setYQPath + + # Method available in _systemYamlHelper.sh + LC_ALL=C type getSystemValue > /dev/null 2>&1 || return 1 + + local frequency="daily" + local archiveFolder="archived" + + local compressLogFiles= + getSystemValue "shared.logging.rotation.compress" "true" + if [[ "${YAML_VALUE}" == "true" ]]; then + compressLogFiles="compress" + fi + + getSystemValue "shared.logging.rotation.maxFiles" "10" + local noOfBackupFiles="${YAML_VALUE}" + + getSystemValue "shared.logging.rotation.maxSizeMb" "25" + local sizeOfFile="${YAML_VALUE}M" + + logDebug "Adding logrotate configuration for [$logFile] to [$confFile]" + + # Add configuration to file + local confContent=$(cat << LOGROTATECONF +$logFile { + $frequency + missingok + rotate $noOfBackupFiles + $compressLogFiles + notifempty + olddir $archiveFolder + dateext + extension .log + dateformat -%Y-%m-%d + size ${sizeOfFile} +} +LOGROTATECONF +) + echo "${confContent}" > ${confFile} || return 1 +} + +_operationIsBySameUser() { + local targetUser="$1" + local currentUserID=$(id -u) + local currentUserName=$(id -un) + + if [ $currentUserID == $targetUser ] || [ $currentUserName == $targetUser ]; then + echo -n "yes" + else + echo -n "no" + fi +} + +_addCronJobForLogrotate() { + logDebug "Method ${FUNCNAME[0]}" + + # Abort if logrotate is not available + [ "$(io_commandExists 'crontab')" != "yes" ] && warn "cron is not available" && return 1 + + # mandatory inputs + local productHome="$1" + local confFile="$2" + local cronJobOwner="$3" + + # We want to use our binary if possible. It may be more recent than the one in the OS + local logrotateBinary="$productHome/app/third-party/logrotate/logrotate" + + if [ ! -f "$logrotateBinary" ]; then + logrotateBinary="logrotate" + [ "$(io_commandExists 'logrotate')" != "yes" ] && warn "logrotate is not available" && return 1 + fi + local cmd="$logrotateBinary ${confFile} --state $productHome/var/etc/logrotate/logrotate-state" #--verbose + + id -u $cronJobOwner > /dev/null 2>&1 || { warn "User $cronJobOwner does not exist. Aborting logrotate configuration" && return 1; } + + # Remove the existing line + removeLogRotation "$productHome" "$cronJobOwner" || true + + # Run logrotate daily at 23:55 hours + local cronInterval="55 23 * * * $cmd" + + local standaloneMode=$(_operationIsBySameUser "$cronJobOwner") + + # If this is standalone mode, we cannot use -u - the user running this process may not have the necessary privileges + if [ "$standaloneMode" == "no" ]; then + (crontab -l -u $cronJobOwner 2>/dev/null; echo "$cronInterval") | crontab -u $cronJobOwner - + else + (crontab -l 2>/dev/null; echo "$cronInterval") | crontab - + fi +} + +## Configure logrotate for a product +## Failure conditions: +## If logrotation could not be setup for some reason +## Parameters: +## $1: The product name +## $2: The product home +## Depends on global: none +## Updates global: none +## Returns: NA + +configureLogRotation() { + logDebug "Method ${FUNCNAME[0]}" + + # mandatory inputs + local productName="$1" + if [ -z $productName ]; then + warn "Incorrect usage. A product name is necessary for configuring log rotation" && return 1 + fi + + local productHome="$2" + if [ -z $productHome ]; then + warn "Incorrect usage. A product home folder is necessary for configuring log rotation" && return 1 + fi + + local logFile="${productHome}/var/log/console.log" + if [[ $(uname) == "Darwin" ]]; then + logger "Log rotation for [$logFile] has not been configured. Please setup manually" + return 0 + fi + + local userID="$3" + if [ -z $userID ]; then + warn "Incorrect usage. A userID is necessary for configuring log rotation" && return 1 + fi + + local groupID=${4:-$userID} + local logConfigOwner=${5:-$userID} + + logDebug "Configuring log rotation as user [$userID], group [$groupID], effective cron User [$logConfigOwner]" + + local errorMessage="Could not configure logrotate. Please configure log rotation of the file: [$logFile] manually" + + local confFile="${productHome}/var/etc/logrotate/logrotate.conf" + + # TODO move to recursive method + createDir "${productHome}" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var/log" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var/log/archived" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + + # TODO move to recursive method + createDir "${productHome}/var/etc" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + createDir "${productHome}/var/etc/logrotate" "$logConfigOwner" || { warn "${errorMessage}" && return 1; } + + # conf file should be owned by the user running the script + createFile "${confFile}" "${logConfigOwner}" || { warn "Could not create configuration file [$confFile]" return 1; } + + _addLogRotateConfiguration "${confFile}" "${logFile}" "$userID" "$groupID" || { warn "${errorMessage}" && return 1; } + _addCronJobForLogrotate "${productHome}" "${confFile}" "${logConfigOwner}" || { warn "${errorMessage}" && return 1; } +} + +_pauseExecution() { + if [ "${VERBOSE_MODE}" == "debug" ]; then + + local breakPoint="$1" + if [ ! -z "$breakPoint" ]; then + printf "${cBlue}Breakpoint${cClear} [$breakPoint] " + echo "" + fi + printf "${cBlue}Press enter once you are ready to continue${cClear}" + read -s choice + echo "" + fi +} + +# removeLogRotation "$productHome" "$cronJobOwner" || true +removeLogRotation() { + logDebug "Method ${FUNCNAME[0]}" + if [[ $(uname) == "Darwin" ]]; then + logDebug "Not implemented for Darwin." + return 0 + fi + local productHome="$1" + local cronJobOwner="$2" + local standaloneMode=$(_operationIsBySameUser "$cronJobOwner") + + local confFile="${productHome}/var/etc/logrotate/logrotate.conf" + + if [ "$standaloneMode" == "no" ]; then + crontab -l -u $cronJobOwner 2>/dev/null | grep -v "$confFile" | crontab -u $cronJobOwner - + else + crontab -l 2>/dev/null | grep -v "$confFile" | crontab - + fi +} + +# NOTE: This method does not check the configuration to see if redirection is necessary. +# This is intentional. If we don't redirect, tomcat logs might get redirected to a folder/file +# that does not exist, causing the service itself to not start +setupTomcatRedirection() { + logDebug "Method ${FUNCNAME[0]}" + local consoleLog="${JF_PRODUCT_HOME}/var/log/console.log" + _createConsoleLog + export CATALINA_OUT="${consoleLog}" +} + +setupScriptLogsRedirection() { + logDebug "Method ${FUNCNAME[0]}" + if [ -z "${JF_PRODUCT_HOME}" ]; then + logDebug "No JF_PRODUCT_HOME. Returning" + return + fi + # Create the console.log file if it is not already present + # _createConsoleLog || true + # # Ensure any logs (logger/logError/warn) also get redirected to the console.log + # # Using installer.log as a temparory fix. Please change this to console.log once INST-291 is fixed + export LOG_BEHAVIOR_ADD_REDIRECTION="${JF_PRODUCT_HOME}/var/log/console.log" + export LOG_BEHAVIOR_ADD_META="$FLAG_Y" +} + +# Returns Y if this method is run inside a container +isRunningInsideAContainer() { + local check1=$(grep -sq 'docker\|kubepods' /proc/1/cgroup; echo $?) + local check2=$(grep -sq 'containers' /proc/self/mountinfo; echo $?) + if [[ $check1 == 0 || $check2 == 0 || -f "/.dockerenv" ]]; then + echo -n "$FLAG_Y" + else + echo -n "$FLAG_N" + fi +} + +POSTGRES_USER=999 +NGINX_USER=104 +NGINX_GROUP=107 +ES_USER=1000 +REDIS_USER=999 +MONGO_USER=999 +RABBITMQ_USER=999 +LOG_FILE_PERMISSION=640 +PID_FILE_PERMISSION=644 + +# Copy file +copyFile(){ + local source=$1 + local target=$2 + local mode=${3:-overwrite} + local enableVerbose=${4:-"${FLAG_N}"} + local verboseFlag="" + + if [ ! -z "${enableVerbose}" ] && [ "${enableVerbose}" == "${FLAG_Y}" ]; then + verboseFlag="-v" + fi + + if [[ ! ( $source && $target ) ]]; then + warn "Source and target is mandatory to copy file" + return 1 + fi + + if [[ -f "${target}" ]]; then + [[ "$mode" = "overwrite" ]] && ( cp ${verboseFlag} -f "$source" "$target" || errorExit "Unable to copy file, command : cp -f ${source} ${target}") || true + else + cp ${verboseFlag} -f "$source" "$target" || errorExit "Unable to copy file, command : cp -f ${source} ${target}" + fi +} + +# Copy files recursively from given source directory to destination directory +# This method wil copy but will NOT overwrite +# Destination will be created if its not available +copyFilesNoOverwrite(){ + local src=$1 + local dest=$2 + local enableVerboseCopy="${3:-${FLAG_Y}}" + + if [[ -z "${src}" || -z "${dest}" ]]; then + return + fi + + if [ -d "${src}" ] && [ "$(ls -A ${src})" ]; then + local relativeFilePath="" + local targetFilePath="" + + for file in $(find ${src} -type f 2>/dev/null) ; do + # Derive relative path and attach it to destination + # Example : + # src=/extra_config + # dest=/var/opt/jfrog/artifactory/etc + # file=/extra_config/config.xml + # relativeFilePath=config.xml + # targetFilePath=/var/opt/jfrog/artifactory/etc/config.xml + relativeFilePath=${file/${src}/} + targetFilePath=${dest}${relativeFilePath} + + createDir "$(dirname "$targetFilePath")" + copyFile "${file}" "${targetFilePath}" "no_overwrite" "${enableVerboseCopy}" + done + fi +} + +# TODO : WINDOWS ? +# Check the max open files and open processes set on the system +checkULimits () { + local minMaxOpenFiles=${1:-32000} + local minMaxOpenProcesses=${2:-1024} + local setValue=${3:-true} + local warningMsgForFiles=${4} + local warningMsgForProcesses=${5} + + logger "Checking open files and processes limits" + + local currentMaxOpenFiles=$(ulimit -n) + logger "Current max open files is $currentMaxOpenFiles" + if [ ${currentMaxOpenFiles} != "unlimited" ] && [ "$currentMaxOpenFiles" -lt "$minMaxOpenFiles" ]; then + if [ "${setValue}" ]; then + ulimit -n "${minMaxOpenFiles}" >/dev/null 2>&1 || warn "Max number of open files $currentMaxOpenFiles is low!" + [ -z "${warningMsgForFiles}" ] || warn "${warningMsgForFiles}" + else + errorExit "Max number of open files $currentMaxOpenFiles, is too low. Cannot run the application!" + fi + fi + + local currentMaxOpenProcesses=$(ulimit -u) + logger "Current max open processes is $currentMaxOpenProcesses" + if [ "$currentMaxOpenProcesses" != "unlimited" ] && [ "$currentMaxOpenProcesses" -lt "$minMaxOpenProcesses" ]; then + if [ "${setValue}" ]; then + ulimit -u "${minMaxOpenProcesses}" >/dev/null 2>&1 || warn "Max number of open files $currentMaxOpenFiles is low!" + [ -z "${warningMsgForProcesses}" ] || warn "${warningMsgForProcesses}" + else + errorExit "Max number of open files $currentMaxOpenProcesses, is too low. Cannot run the application!" + fi + fi +} + +createDirs() { + local appDataDir=$1 + local serviceName=$2 + local folders="backup bootstrap data etc logs work" + + [ -z "${appDataDir}" ] && errorExit "An application directory is mandatory to create its data structure" || true + [ -z "${serviceName}" ] && errorExit "A service name is mandatory to create service data structure" || true + + for folder in ${folders} + do + folder=${appDataDir}/${folder}/${serviceName} + if [ ! -d "${folder}" ]; then + logger "Creating folder : ${folder}" + mkdir -p "${folder}" || errorExit "Failed to create ${folder}" + fi + done +} + + +testReadWritePermissions () { + local dir_to_check=$1 + local error=false + + [ -d ${dir_to_check} ] || errorExit "'${dir_to_check}' is not a directory" + + local test_file=${dir_to_check}/test-permissions + + # Write file + if echo test > ${test_file} 1> /dev/null 2>&1; then + # Write succeeded. Testing read... + if cat ${test_file} > /dev/null; then + rm -f ${test_file} + else + error=true + fi + else + error=true + fi + + if [ ${error} == true ]; then + return 1 + else + return 0 + fi +} + +# Test directory has read/write permissions for current user +testDirectoryPermissions () { + local dir_to_check=$1 + local error=false + + [ -d ${dir_to_check} ] || errorExit "'${dir_to_check}' is not a directory" + + local u_id=$(id -u) + local id_str="id ${u_id}" + + logger "Testing directory ${dir_to_check} has read/write permissions for user ${id_str}" + + if ! testReadWritePermissions ${dir_to_check}; then + error=true + fi + + if [ "${error}" == true ]; then + local stat_data=$(stat -Lc "Directory: %n, permissions: %a, owner: %U, group: %G" ${dir_to_check}) + logger "###########################################################" + logger "${dir_to_check} DOES NOT have proper permissions for user ${id_str}" + logger "${stat_data}" + logger "Mounted directory must have read/write permissions for user ${id_str}" + logger "###########################################################" + errorExit "Directory ${dir_to_check} has bad permissions for user ${id_str}" + fi + logger "Permissions for ${dir_to_check} are good" +} + +# Utility method to create a directory path recursively with chown feature as +# Failure conditions: +## Exits if unable to create a directory +# Parameters: +## $1: Root directory from where the path can be created +## $2: List of recursive child directories separated by space +## $3: user who should own the directory. Optional +## $4: group who should own the directory. Optional +# Depends on global: none +# Updates global: none +# Returns: NA +# +# Usage: +# createRecursiveDir "/opt/jfrog/product/var" "bootstrap tomcat lib" "user_name" "group_name" +createRecursiveDir(){ + local rootDir=$1 + local pathDirs=$2 + local user=$3 + local group=${4:-${user}} + local fullPath= + + [ ! -z "${rootDir}" ] || return 0 + + createDir "${rootDir}" "${user}" "${group}" + + [ ! -z "${pathDirs}" ] || return 0 + + fullPath=${rootDir} + + for dir in ${pathDirs}; do + fullPath=${fullPath}/${dir} + createDir "${fullPath}" "${user}" "${group}" + done +} + +# Utility method to create a directory +# Failure conditions: +## Exits if unable to create a directory +# Parameters: +## $1: directory to create +## $2: user who should own the directory. Optional +## $3: group who should own the directory. Optional +# Depends on global: none +# Updates global: none +# Returns: NA + +createDir(){ + local dirName="$1" + local printMessage=no + logSilly "Method ${FUNCNAME[0]} invoked with [$dirName]" + [ -z "${dirName}" ] && return + + logDebug "Attempting to create ${dirName}" + mkdir -p "${dirName}" || errorExit "Unable to create directory: [${dirName}]" + local userID="$2" + local groupID=${3:-$userID} + + # If UID/GID is passed, chown the folder + if [ ! -z "$userID" ] && [ ! -z "$groupID" ]; then + # Earlier, this line would have returned 1 if it failed. Now it just warns. + # This is intentional. Earlier, this line would NOT be reached if the folder already existed. + # Since it will always come to this line and the script may be running as a non-root user, this method will just warn if + # setting permissions fails (so as to not affect any existing flows) + io_setOwnershipNonRecursive "$dirName" "$userID" "$groupID" || warn "Could not set owner of [$dirName] to [$userID:$groupID]" + fi + # logging message to print created dir with user and group + local logMessage=${4:-$printMessage} + if [[ "${logMessage}" == "yes" ]]; then + logger "Successfully created directory [${dirName}]. Owner: [${userID}:${groupID}]" + fi +} + +removeSoftLinkAndCreateDir () { + local dirName="$1" + local userID="$2" + local groupID="$3" + local logMessage="$4" + removeSoftLink "${dirName}" + createDir "${dirName}" "${userID}" "${groupID}" "${logMessage}" +} + +# Utility method to remove a soft link +removeSoftLink () { + local dirName="$1" + if [[ -L "${dirName}" ]]; then + targetLink=$(readlink -f "${dirName}") + logger "Removing the symlink [${dirName}] pointing to [${targetLink}]" + rm -f "${dirName}" + fi +} + +# Check Directory exist in the path +checkDirExists () { + local directoryPath="$1" + + [[ -d "${directoryPath}" ]] && echo -n "true" || echo -n "false" +} + + +# Utility method to create a file +# Failure conditions: +# Parameters: +## $1: file to create +# Depends on global: none +# Updates global: none +# Returns: NA + +createFile(){ + local fileName="$1" + logSilly "Method ${FUNCNAME[0]} [$fileName]" + [ -f "${fileName}" ] && return 0 + touch "${fileName}" || return 1 + + local userID="$2" + local groupID=${3:-$userID} + + # If UID/GID is passed, chown the folder + if [ ! -z "$userID" ] && [ ! -z "$groupID" ]; then + io_setOwnership "$fileName" "$userID" "$groupID" || return 1 + fi +} + +# Check File exist in the filePath +# IMPORTANT- DON'T ADD LOGGING to this method +checkFileExists () { + local filePath="$1" + + [[ -f "${filePath}" ]] && echo -n "true" || echo -n "false" +} + +# Check for directories contains any (files or sub directories) +# IMPORTANT- DON'T ADD LOGGING to this method +checkDirContents () { + local directoryPath="$1" + if [[ "$(ls -1 "${directoryPath}" | wc -l)" -gt 0 ]]; then + echo -n "true" + else + echo -n "false" + fi +} + +# Check contents exist in directory +# IMPORTANT- DON'T ADD LOGGING to this method +checkContentExists () { + local source="$1" + + if [[ "$(checkDirContents "${source}")" != "true" ]]; then + echo -n "false" + else + echo -n "true" + fi +} + +# Resolve the variable +# IMPORTANT- DON'T ADD LOGGING to this method +evalVariable () { + local output="$1" + local input="$2" + + eval "${output}"=\${"${input}"} + eval echo \${"${output}"} +} + +# Usage: if [ "$(io_commandExists 'curl')" == "yes" ] +# IMPORTANT- DON'T ADD LOGGING to this method +io_commandExists() { + local commandToExecute="$1" + hash "${commandToExecute}" 2>/dev/null + local rt=$? + if [ "$rt" == 0 ]; then echo -n "yes"; else echo -n "no"; fi +} + +# Usage: if [ "$(io_curlExists)" != "yes" ] +# IMPORTANT- DON'T ADD LOGGING to this method +io_curlExists() { + io_commandExists "curl" +} + + +io_hasMatch() { + logSilly "Method ${FUNCNAME[0]}" + local result=0 + logDebug "Executing [echo \"$1\" | grep \"$2\" >/dev/null 2>&1]" + echo "$1" | grep "$2" >/dev/null 2>&1 || result=1 + return $result +} + +# Utility method to check if the string passed (usually a connection url) corresponds to this machine itself +# Failure conditions: None +# Parameters: +## $1: string to check against +# Depends on global: none +# Updates global: IS_LOCALHOST with value "yes/no" +# Returns: NA + +io_getIsLocalhost() { + logSilly "Method ${FUNCNAME[0]}" + IS_LOCALHOST="$FLAG_N" + local inputString="$1" + logDebug "Parsing [$inputString] to check if we are dealing with this machine itself" + + io_hasMatch "$inputString" "localhost" && { + logDebug "Found localhost. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for localhost" + + local hostIP=$(io_getPublicHostIP) + io_hasMatch "$inputString" "$hostIP" && { + logDebug "Found $hostIP. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for $hostIP" + + local hostID=$(io_getPublicHostID) + io_hasMatch "$inputString" "$hostID" && { + logDebug "Found $hostID. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for $hostID" + + local hostName=$(io_getPublicHostName) + io_hasMatch "$inputString" "$hostName" && { + logDebug "Found $hostName. Returning [$FLAG_Y]" + IS_LOCALHOST="$FLAG_Y" && return; + } || logDebug "Did not find match for $hostName" + +} + +# Usage: if [ "$(io_tarExists)" != "yes" ] +# IMPORTANT- DON'T ADD LOGGING to this method +io_tarExists() { + io_commandExists "tar" +} + +# IMPORTANT- DON'T ADD LOGGING to this method +io_getPublicHostIP() { + local OS_TYPE=$(uname) + local publicHostIP= + if [ "${OS_TYPE}" == "Darwin" ]; then + ipStatus=$(ifconfig en0 | grep "status" | awk '{print$2}') + if [ "${ipStatus}" == "active" ]; then + publicHostIP=$(ifconfig en0 | grep inet | grep -v inet6 | awk '{print $2}') + else + errorExit "Host IP could not be resolved!" + fi + elif [ "${OS_TYPE}" == "Linux" ]; then + publicHostIP=$(hostname -i 2>/dev/null || echo "127.0.0.1") + fi + publicHostIP=$(echo "${publicHostIP}" | awk '{print $1}') + echo -n "${publicHostIP}" +} + +# Will return the short host name (up to the first dot) +# IMPORTANT- DON'T ADD LOGGING to this method +io_getPublicHostName() { + echo -n "$(hostname -s)" +} + +# Will return the full host name (use this as much as possible) +# IMPORTANT- DON'T ADD LOGGING to this method +io_getPublicHostID() { + echo -n "$(hostname)" +} + +# Utility method to backup a file +# Failure conditions: NA +# Parameters: filePath +# Depends on global: none, +# Updates global: none +# Returns: NA +io_backupFile() { + logSilly "Method ${FUNCNAME[0]}" + fileName="$1" + if [ ! -f "${filePath}" ]; then + logDebug "No file: [${filePath}] to backup" + return + fi + dateTime=$(date +"%Y-%m-%d-%H-%M-%S") + targetFileName="${fileName}.backup.${dateTime}" + yes | \cp -f "$fileName" "${targetFileName}" + logger "File [${fileName}] backedup as [${targetFileName}]" +} + +# Reference https://stackoverflow.com/questions/4023830/how-to-compare-two-strings-in-dot-separated-version-format-in-bash/4025065#4025065 +is_number() { + case "$BASH_VERSION" in + 3.1.*) + PATTERN='\^\[0-9\]+\$' + ;; + *) + PATTERN='^[0-9]+$' + ;; + esac + + [[ "$1" =~ $PATTERN ]] +} + +io_compareVersions() { + if [[ $# != 2 ]] + then + echo "Usage: min_version current minimum" + return + fi + + A="${1%%.*}" + B="${2%%.*}" + + if [[ "$A" != "$1" && "$B" != "$2" && "$A" == "$B" ]] + then + io_compareVersions "${1#*.}" "${2#*.}" + else + if is_number "$A" && is_number "$B" + then + if [[ "$A" -eq "$B" ]]; then + echo "0" + elif [[ "$A" -gt "$B" ]]; then + echo "1" + elif [[ "$A" -lt "$B" ]]; then + echo "-1" + fi + fi + fi +} + +# Reference https://stackoverflow.com/questions/369758/how-to-trim-whitespace-from-a-bash-variable +# Strip all leading and trailing spaces +# IMPORTANT- DON'T ADD LOGGING to this method +io_trim() { + local var="$1" + # remove leading whitespace characters + var="${var#"${var%%[![:space:]]*}"}" + # remove trailing whitespace characters + var="${var%"${var##*[![:space:]]}"}" + echo -n "$var" +} + +# temporary function will be removing it ASAP +# search for string and replace text in file +replaceText_migration_hook () { + local regexString="$1" + local replaceText="$2" + local file="$3" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e "s/${regexString}/${replaceText}/" "${file}" || warn "Failed to replace the text in ${file}" + else + sed -i -e "s/${regexString}/${replaceText}/" "${file}" || warn "Failed to replace the text in ${file}" + fi +} + +# search for string and replace text in file +replaceText () { + local regexString="$1" + local replaceText="$2" + local file="$3" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e "s#${regexString}#${replaceText}#" "${file}" || warn "Failed to replace the text in ${file}" + else + sed -i -e "s#${regexString}#${replaceText}#" "${file}" || warn "Failed to replace the text in ${file}" + logDebug "Replaced [$regexString] with [$replaceText] in [$file]" + fi +} + +# search for string and prepend text in file +prependText () { + local regexString="$1" + local text="$2" + local file="$3" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e '/'"${regexString}"'/i\'$'\n\\'"${text}"''$'\n' "${file}" || warn "Failed to prepend the text in ${file}" + else + sed -i -e '/'"${regexString}"'/i\'$'\n\\'"${text}"''$'\n' "${file}" || warn "Failed to prepend the text in ${file}" + fi +} + +# add text to beginning of the file +addText () { + local text="$1" + local file="$2" + + if [[ "$(checkFileExists "${file}")" != "true" ]]; then + return + fi + if [[ $(uname) == "Darwin" ]]; then + sed -i '' -e '1s/^/'"${text}"'\'$'\n/' "${file}" || warn "Failed to add the text in ${file}" + else + sed -i -e '1s/^/'"${text}"'\'$'\n/' "${file}" || warn "Failed to add the text in ${file}" + fi +} + +io_replaceString () { + local value="$1" + local firstString="$2" + local secondString="$3" + local separator=${4:-"/"} + local updateValue= + if [[ $(uname) == "Darwin" ]]; then + updateValue=$(echo "${value}" | sed "s${separator}${firstString}${separator}${secondString}${separator}") + else + updateValue=$(echo "${value}" | sed "s${separator}${firstString}${separator}${secondString}${separator}") + fi + echo -n "${updateValue}" +} + +_findYQ() { + # logSilly "Method ${FUNCNAME[0]}" (Intentionally not logging. Does not add value) + local parentDir="$1" + if [ -z "$parentDir" ]; then + return + fi + logDebug "Executing command [find "${parentDir}" -name third-party -type d]" + local yq=$(find "${parentDir}" -name third-party -type d) + if [ -d "${yq}/yq" ]; then + export YQ_PATH="${yq}/yq" + fi +} + + +io_setYQPath() { + # logSilly "Method ${FUNCNAME[0]}" (Intentionally not logging. Does not add value) + if [ "$(io_commandExists 'yq')" == "yes" ]; then + return + fi + + if [ ! -z "${JF_PRODUCT_HOME}" ] && [ -d "${JF_PRODUCT_HOME}" ]; then + _findYQ "${JF_PRODUCT_HOME}" + fi + + if [ -z "${YQ_PATH}" ] && [ ! -z "${COMPOSE_HOME}" ] && [ -d "${COMPOSE_HOME}" ]; then + _findYQ "${COMPOSE_HOME}" + fi + # TODO We can remove this block after all the code is restructured. + if [ -z "${YQ_PATH}" ] && [ ! -z "${SCRIPT_HOME}" ] && [ -d "${SCRIPT_HOME}" ]; then + _findYQ "${SCRIPT_HOME}" + fi + +} + +io_getLinuxDistribution() { + LINUX_DISTRIBUTION= + + # Make sure running on Linux + [ $(uname -s) != "Linux" ] && return + + # Find out what Linux distribution we are on + + cat /etc/*-release | grep -i Red >/dev/null 2>&1 && LINUX_DISTRIBUTION=RedHat || true + + # OS 6.x + cat /etc/issue.net | grep Red >/dev/null 2>&1 && LINUX_DISTRIBUTION=RedHat || true + + # OS 7.x + cat /etc/*-release | grep -i centos >/dev/null 2>&1 && LINUX_DISTRIBUTION=CentOS && LINUX_DISTRIBUTION_VER="7" || true + + # OS 8.x + grep -q -i "release 8" /etc/redhat-release >/dev/null 2>&1 && LINUX_DISTRIBUTION_VER="8" || true + + # OS 7.x + grep -q -i "release 7" /etc/redhat-release >/dev/null 2>&1 && LINUX_DISTRIBUTION_VER="7" || true + + # OS 6.x + grep -q -i "release 6" /etc/redhat-release >/dev/null 2>&1 && LINUX_DISTRIBUTION_VER="6" || true + + cat /etc/*-release | grep -i Red | grep -i 'VERSION=7' >/dev/null 2>&1 && LINUX_DISTRIBUTION=RedHat && LINUX_DISTRIBUTION_VER="7" || true + + cat /etc/*-release | grep -i debian >/dev/null 2>&1 && LINUX_DISTRIBUTION=Debian || true + + cat /etc/*-release | grep -i ubuntu >/dev/null 2>&1 && LINUX_DISTRIBUTION=Ubuntu || true +} + +## Utility method to check ownership of folders/files +## Failure conditions: + ## If invoked with incorrect inputs - FATAL + ## If file is not owned by the user & group +## Parameters: + ## user + ## group + ## folder to chown +## Globals: none +## Returns: none +## NOTE: The method does NOTHING if the OS is Mac +io_checkOwner () { + logSilly "Method ${FUNCNAME[0]}" + local osType=$(uname) + + if [ "${osType}" != "Linux" ]; then + logDebug "Unsupported OS. Skipping check" + return 0 + fi + + local file_to_check=$1 + local user_id_to_check=$2 + + + if [ -z "$user_id_to_check" ] || [ -z "$file_to_check" ]; then + errorExit "Invalid invocation of method. Missing mandatory inputs" + fi + + local group_id_to_check=${3:-$user_id_to_check} + local check_user_name=${4:-"no"} + + logDebug "Checking permissions on [$file_to_check] for user [$user_id_to_check] & group [$group_id_to_check]" + + local stat= + + if [ "${check_user_name}" == "yes" ]; then + stat=( $(stat -Lc "%U %G" ${file_to_check}) ) + else + stat=( $(stat -Lc "%u %g" ${file_to_check}) ) + fi + + local user_id=${stat[0]} + local group_id=${stat[1]} + + if [[ "${user_id}" != "${user_id_to_check}" ]] || [[ "${group_id}" != "${group_id_to_check}" ]] ; then + logDebug "Ownership mismatch. [${file_to_check}] is not owned by [${user_id_to_check}:${group_id_to_check}]" + return 1 + else + return 0 + fi +} + +## Utility method to change ownership of a file/folder - NON recursive +## Failure conditions: + ## If invoked with incorrect inputs - FATAL + ## If chown operation fails - returns 1 +## Parameters: + ## user + ## group + ## file to chown +## Globals: none +## Returns: none +## NOTE: The method does NOTHING if the OS is Mac + +io_setOwnershipNonRecursive() { + + local osType=$(uname) + if [ "${osType}" != "Linux" ]; then + return + fi + + local targetFile=$1 + local user=$2 + + if [ -z "$user" ] || [ -z "$targetFile" ]; then + errorExit "Invalid invocation of method. Missing mandatory inputs" + fi + + local group=${3:-$user} + logDebug "Method ${FUNCNAME[0]}. Executing [chown ${user}:${group} ${targetFile}]" + chown ${user}:${group} ${targetFile} || return 1 +} + +## Utility method to change ownership of a file. +## IMPORTANT +## If being called on a folder, should ONLY be called for fresh folders or may cause performance issues +## Failure conditions: + ## If invoked with incorrect inputs - FATAL + ## If chown operation fails - returns 1 +## Parameters: + ## user + ## group + ## file to chown +## Globals: none +## Returns: none +## NOTE: The method does NOTHING if the OS is Mac + +io_setOwnership() { + + local osType=$(uname) + if [ "${osType}" != "Linux" ]; then + return + fi + + local targetFile=$1 + local user=$2 + + if [ -z "$user" ] || [ -z "$targetFile" ]; then + errorExit "Invalid invocation of method. Missing mandatory inputs" + fi + + local group=${3:-$user} + logDebug "Method ${FUNCNAME[0]}. Executing [chown -R ${user}:${group} ${targetFile}]" + chown -R ${user}:${group} ${targetFile} || return 1 +} + +## Utility method to create third party folder structure necessary for Postgres +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## POSTGRESQL_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createPostgresDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${POSTGRESQL_DATA_ROOT}" ] && return 0 + + logDebug "Property [${POSTGRESQL_DATA_ROOT}] exists. Proceeding" + + createDir "${POSTGRESQL_DATA_ROOT}/data" + io_setOwnership "${POSTGRESQL_DATA_ROOT}" "${POSTGRES_USER}" "${POSTGRES_USER}" || errorExit "Setting ownership of [${POSTGRESQL_DATA_ROOT}] to [${POSTGRES_USER}:${POSTGRES_USER}] failed" +} + +## Utility method to create third party folder structure necessary for Nginx +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## NGINX_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createNginxDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${NGINX_DATA_ROOT}" ] && return 0 + + logDebug "Property [${NGINX_DATA_ROOT}] exists. Proceeding" + + createDir "${NGINX_DATA_ROOT}" + io_setOwnership "${NGINX_DATA_ROOT}" "${NGINX_USER}" "${NGINX_GROUP}" || errorExit "Setting ownership of [${NGINX_DATA_ROOT}] to [${NGINX_USER}:${NGINX_GROUP}] failed" +} + +## Utility method to create third party folder structure necessary for ElasticSearch +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## ELASTIC_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createElasticSearchDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${ELASTIC_DATA_ROOT}" ] && return 0 + + logDebug "Property [${ELASTIC_DATA_ROOT}] exists. Proceeding" + + createDir "${ELASTIC_DATA_ROOT}/data" + io_setOwnership "${ELASTIC_DATA_ROOT}" "${ES_USER}" "${ES_USER}" || errorExit "Setting ownership of [${ELASTIC_DATA_ROOT}] to [${ES_USER}:${ES_USER}] failed" +} + +## Utility method to create third party folder structure necessary for Redis +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## REDIS_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createRedisDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${REDIS_DATA_ROOT}" ] && return 0 + + logDebug "Property [${REDIS_DATA_ROOT}] exists. Proceeding" + + createDir "${REDIS_DATA_ROOT}" + io_setOwnership "${REDIS_DATA_ROOT}" "${REDIS_USER}" "${REDIS_USER}" || errorExit "Setting ownership of [${REDIS_DATA_ROOT}] to [${REDIS_USER}:${REDIS_USER}] failed" +} + +## Utility method to create third party folder structure necessary for Mongo +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## MONGODB_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createMongoDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${MONGODB_DATA_ROOT}" ] && return 0 + + logDebug "Property [${MONGODB_DATA_ROOT}] exists. Proceeding" + + createDir "${MONGODB_DATA_ROOT}/logs" + createDir "${MONGODB_DATA_ROOT}/configdb" + createDir "${MONGODB_DATA_ROOT}/db" + io_setOwnership "${MONGODB_DATA_ROOT}" "${MONGO_USER}" "${MONGO_USER}" || errorExit "Setting ownership of [${MONGODB_DATA_ROOT}] to [${MONGO_USER}:${MONGO_USER}] failed" +} + +## Utility method to create third party folder structure necessary for RabbitMQ +## Failure conditions: +## If creation of directory or assigning permissions fails +## Parameters: none +## Globals: +## RABBITMQ_DATA_ROOT +## Returns: none +## NOTE: The method does NOTHING if the folder already exists +io_createRabbitMQDir() { + logDebug "Method ${FUNCNAME[0]}" + [ -z "${RABBITMQ_DATA_ROOT}" ] && return 0 + + logDebug "Property [${RABBITMQ_DATA_ROOT}] exists. Proceeding" + + createDir "${RABBITMQ_DATA_ROOT}" + io_setOwnership "${RABBITMQ_DATA_ROOT}" "${RABBITMQ_USER}" "${RABBITMQ_USER}" || errorExit "Setting ownership of [${RABBITMQ_DATA_ROOT}] to [${RABBITMQ_USER}:${RABBITMQ_USER}] failed" +} + +# Add or replace a property in provided properties file +addOrReplaceProperty() { + local propertyName=$1 + local propertyValue=$2 + local propertiesPath=$3 + local delimiter=${4:-"="} + + # Return if any of the inputs are empty + [[ -z "$propertyName" || "$propertyName" == "" ]] && return + [[ -z "$propertyValue" || "$propertyValue" == "" ]] && return + [[ -z "$propertiesPath" || "$propertiesPath" == "" ]] && return + + grep "^${propertyName}\s*${delimiter}.*$" ${propertiesPath} > /dev/null 2>&1 + [ $? -ne 0 ] && echo -e "\n${propertyName}${delimiter}${propertyValue}" >> ${propertiesPath} + sed -i -e "s|^${propertyName}\s*${delimiter}.*$|${propertyName}${delimiter}${propertyValue}|g;" ${propertiesPath} +} + +# Set property only if its not set +io_setPropertyNoOverride(){ + local propertyName=$1 + local propertyValue=$2 + local propertiesPath=$3 + + # Return if any of the inputs are empty + [[ -z "$propertyName" || "$propertyName" == "" ]] && return + [[ -z "$propertyValue" || "$propertyValue" == "" ]] && return + [[ -z "$propertiesPath" || "$propertiesPath" == "" ]] && return + + grep "^${propertyName}:" ${propertiesPath} > /dev/null 2>&1 + if [ $? -ne 0 ]; then + echo -e "${propertyName}: ${propertyValue}" >> ${propertiesPath} || warn "Setting property ${propertyName}: ${propertyValue} in [ ${propertiesPath} ] failed" + else + logger "Skipping update of property : ${propertyName}" >&6 + fi +} + +# Add a line to a file if it doesn't already exist +addLine() { + local line_to_add=$1 + local target_file=$2 + logger "Trying to add line $1 to $2" >&6 2>&1 + cat "$target_file" | grep -F "$line_to_add" -wq >&6 2>&1 + if [ $? != 0 ]; then + logger "Line does not exist and will be added" >&6 2>&1 + echo $line_to_add >> $target_file || errorExit "Could not update $target_file" + fi +} + +# Utility method to check if a value (first parameter) exists in an array (2nd parameter) +# 1st parameter "value to find" +# 2nd parameter "The array to search in. Please pass a string with each value separated by space" +# Example: containsElement "y" "y Y n N" +containsElement () { + local searchElement=$1 + local searchArray=($2) + local found=1 + for elementInIndex in "${searchArray[@]}";do + if [[ $elementInIndex == $searchElement ]]; then + found=0 + fi + done + return $found +} + +# Utility method to get user's choice +# 1st parameter "what to ask the user" +# 2nd parameter "what choices to accept, separated by spaces" +# 3rd parameter "what is the default choice (to use if the user simply presses Enter)" +# Example 'getUserChoice "Are you feeling lucky? Punk!" "y n Y N" "y"' +getUserChoice(){ + configureLogOutput + read_timeout=${read_timeout:-0.5} + local choice="na" + local text_to_display=$1 + local choices=$2 + local default_choice=$3 + users_choice= + + until containsElement "$choice" "$choices"; do + echo "";echo ""; + sleep $read_timeout #This ensures correct placement of the question. + read -p "$text_to_display :" choice + : ${choice:=$default_choice} + done + users_choice=$choice + echo -e "\n$text_to_display: $users_choice" >&6 + sleep $read_timeout #This ensures correct logging +} + +setFilePermission () { + local permission=$1 + local file=$2 + chmod "${permission}" "${file}" || warn "Setting permission ${permission} to file [ ${file} ] failed" +} + + +#setting required paths +setAppDir (){ + SCRIPT_DIR=$(dirname $0) + SCRIPT_HOME="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" + APP_DIR="`cd "${SCRIPT_HOME}";pwd`" +} + +ZIP_TYPE="zip" +COMPOSE_TYPE="compose" +HELM_TYPE="helm" +RPM_TYPE="rpm" +DEB_TYPE="debian" + +sourceScript () { + local file="$1" + + [ ! -z "${file}" ] || errorExit "target file is not passed to source a file" + + if [ ! -f "${file}" ]; then + errorExit "${file} file is not found" + else + source "${file}" || errorExit "Unable to source ${file}, please check if the user ${USER} has permissions to perform this action" + fi +} +# Source required helpers +initHelpers () { + local systemYamlHelper="${APP_DIR}/systemYamlHelper.sh" + local thirdPartyDir=$(find ${APP_DIR}/.. -name third-party -type d) + export YQ_PATH="${thirdPartyDir}/yq" + LIBXML2_PATH="${thirdPartyDir}/libxml2/bin/xmllint" + export LD_LIBRARY_PATH="${thirdPartyDir}/libxml2/lib" + sourceScript "${systemYamlHelper}" +} +# Check migration info yaml file available in the path +checkMigrationInfoYaml () { + + if [[ -f "${APP_DIR}/migrationHelmInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationHelmInfo.yaml" + INSTALLER="${HELM_TYPE}" + elif [[ -f "${APP_DIR}/migrationZipInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationZipInfo.yaml" + INSTALLER="${ZIP_TYPE}" + elif [[ -f "${APP_DIR}/migrationRpmInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationRpmInfo.yaml" + INSTALLER="${RPM_TYPE}" + elif [[ -f "${APP_DIR}/migrationDebInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationDebInfo.yaml" + INSTALLER="${DEB_TYPE}" + elif [[ -f "${APP_DIR}/migrationComposeInfo.yaml" ]]; then + MIGRATION_SYSTEM_YAML_INFO="${APP_DIR}/migrationComposeInfo.yaml" + INSTALLER="${COMPOSE_TYPE}" + else + errorExit "File migration Info yaml does not exist in [${APP_DIR}]" + fi +} + +retrieveYamlValue () { + local yamlPath="$1" + local value="$2" + local output="$3" + local message="$4" + + [[ -z "${yamlPath}" ]] && errorExit "yamlPath is mandatory to get value from ${MIGRATION_SYSTEM_YAML_INFO}" + + getYamlValue "${yamlPath}" "${MIGRATION_SYSTEM_YAML_INFO}" "false" + value="${YAML_VALUE}" + if [[ -z "${value}" ]]; then + if [[ "${output}" == "Warning" ]]; then + warn "Empty value for ${yamlPath} in [${MIGRATION_SYSTEM_YAML_INFO}]" + elif [[ "${output}" == "Skip" ]]; then + return + else + errorExit "${message}" + fi + fi +} + +checkEnv () { + + if [[ "${INSTALLER}" == "${ZIP_TYPE}" ]]; then + # check Environment JF_PRODUCT_HOME is set before migration + NEW_DATA_DIR="$(evalVariable "NEW_DATA_DIR" "JF_PRODUCT_HOME")" + if [[ -z "${NEW_DATA_DIR}" ]]; then + errorExit "Environment variable JF_PRODUCT_HOME is not set, this is required to perform Migration" + fi + # appending var directory to $JF_PRODUCT_HOME + NEW_DATA_DIR="${NEW_DATA_DIR}/var" + elif [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + getCustomDataDir_hook + NEW_DATA_DIR="${OLD_DATA_DIR}" + if [[ -z "${NEW_DATA_DIR}" ]] && [[ -z "${OLD_DATA_DIR}" ]]; then + errorExit "Could not find ${PROMPT_DATA_DIR_LOCATION} to perform Migration" + fi + else + # check Environment JF_ROOT_DATA_DIR is set before migration + OLD_DATA_DIR="$(evalVariable "OLD_DATA_DIR" "JF_ROOT_DATA_DIR")" + # check Environment JF_ROOT_DATA_DIR is set before migration + NEW_DATA_DIR="$(evalVariable "NEW_DATA_DIR" "JF_ROOT_DATA_DIR")" + if [[ -z "${NEW_DATA_DIR}" ]] && [[ -z "${OLD_DATA_DIR}" ]]; then + errorExit "Could not find ${PROMPT_DATA_DIR_LOCATION} to perform Migration" + fi + # appending var directory to $JF_PRODUCT_HOME + NEW_DATA_DIR="${NEW_DATA_DIR}/var" + fi + +} + +getDataDir () { + + if [[ "${INSTALLER}" == "${ZIP_TYPE}" || "${INSTALLER}" == "${COMPOSE_TYPE}"|| "${INSTALLER}" == "${HELM_TYPE}" ]]; then + checkEnv + else + getCustomDataDir_hook + NEW_DATA_DIR="`cd "${APP_DIR}"/../../;pwd`" + NEW_DATA_DIR="${NEW_DATA_DIR}/var" + fi +} + +# Retrieve Product name from MIGRATION_SYSTEM_YAML_INFO +getProduct () { + retrieveYamlValue "migration.product" "${YAML_VALUE}" "Fail" "Empty value under ${yamlPath} in [${MIGRATION_SYSTEM_YAML_INFO}]" + PRODUCT="${YAML_VALUE}" + PRODUCT=$(echo "${PRODUCT}" | tr '[:upper:]' '[:lower:]' 2>/dev/null) + if [[ "${PRODUCT}" != "artifactory" && "${PRODUCT}" != "distribution" && "${PRODUCT}" != "xray" ]]; then + errorExit "migration.product in [${MIGRATION_SYSTEM_YAML_INFO}] is not correct, please set based on product as ARTIFACTORY or DISTRIBUTION" + fi + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + JF_USER="${PRODUCT}" + fi +} +# Compare product version with minProductVersion and maxProductVersion +migrateCheckVersion () { + local productVersion="$1" + local minProductVersion="$2" + local maxProductVersion="$3" + local productVersion618="6.18.0" + local unSupportedProductVersions7=("7.2.0 7.2.1") + + if [[ "$(io_compareVersions "${productVersion}" "${maxProductVersion}")" -eq 0 || "$(io_compareVersions "${productVersion}" "${maxProductVersion}")" -eq 1 ]]; then + logger "Migration not necessary. ${PRODUCT} is already ${productVersion}" + exit 11 + elif [[ "$(io_compareVersions "${productVersion}" "${minProductVersion}")" -eq 0 || "$(io_compareVersions "${productVersion}" "${minProductVersion}")" -eq 1 ]]; then + if [[ ("$(io_compareVersions "${productVersion}" "${productVersion618}")" -eq 0 || "$(io_compareVersions "${productVersion}" "${productVersion618}")" -eq 1) && " ${unSupportedProductVersions7[@]} " =~ " ${CURRENT_VERSION} " ]]; then + touch /tmp/error; + errorExit "Current ${PRODUCT} version (${productVersion}) does not support migration to ${CURRENT_VERSION}" + else + bannerStart "Detected ${PRODUCT} ${productVersion}, initiating migration" + fi + else + logger "Current ${PRODUCT} ${productVersion} version is not supported for migration" + exit 1 + fi +} + +getProductVersion () { + local minProductVersion="$1" + local maxProductVersion="$2" + local newfilePath="$3" + local oldfilePath="$4" + local propertyInDocker="$5" + local property="$6" + local productVersion= + local status= + + if [[ "$INSTALLER" == "${COMPOSE_TYPE}" ]]; then + if [[ -f "${oldfilePath}" ]]; then + if [[ "${PRODUCT}" == "artifactory" ]]; then + productVersion="$(readKey "${property}" "${oldfilePath}")" + else + productVersion="$(cat "${oldfilePath}")" + fi + status="success" + elif [[ -f "${newfilePath}" ]]; then + productVersion="$(readKey "${propertyInDocker}" "${newfilePath}")" + status="fail" + else + logger "File [${oldfilePath}] or [${newfilePath}] not found to get current version." + exit 0 + fi + elif [[ "$INSTALLER" == "${HELM_TYPE}" ]]; then + if [[ -f "${oldfilePath}" ]]; then + if [[ "${PRODUCT}" == "artifactory" ]]; then + productVersion="$(readKey "${property}" "${oldfilePath}")" + else + productVersion="$(cat "${oldfilePath}")" + fi + status="success" + else + productVersion="${CURRENT_VERSION}" + [[ -z "${productVersion}" || "${productVersion}" == "" ]] && logger "${PRODUCT} CURRENT_VERSION is not set" && exit 0 + fi + else + if [[ -f "${newfilePath}" ]]; then + productVersion="$(readKey "${property}" "${newfilePath}")" + status="fail" + elif [[ -f "${oldfilePath}" ]]; then + productVersion="$(readKey "${property}" "${oldfilePath}")" + status="success" + else + if [[ "${INSTALLER}" == "${ZIP_TYPE}" ]]; then + logger "File [${newfilePath}] not found to get current version." + else + logger "File [${oldfilePath}] or [${newfilePath}] not found to get current version." + fi + exit 0 + fi + fi + if [[ -z "${productVersion}" || "${productVersion}" == "" ]]; then + [[ "${status}" == "success" ]] && logger "No version found in file [${oldfilePath}]." + [[ "${status}" == "fail" ]] && logger "No version found in file [${newfilePath}]." + exit 0 + fi + + migrateCheckVersion "${productVersion}" "${minProductVersion}" "${maxProductVersion}" +} + +readKey () { + local property="$1" + local file="$2" + local version= + + while IFS='=' read -r key value || [ -n "${key}" ]; + do + [[ ! "${key}" =~ \#.* && ! -z "${key}" && ! -z "${value}" ]] + key="$(io_trim "${key}")" + if [[ "${key}" == "${property}" ]]; then + version="${value}" && check=true && break + else + check=false + fi + done < "${file}" + if [[ "${check}" == "false" ]]; then + return + fi + echo "${version}" +} + +# create Log directory +createLogDir () { + if [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + getUserAndGroupFromFile + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/log" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" + fi +} + +# Creating migration log file +creationMigrateLog () { + local LOG_FILE_NAME="migration.log" + createLogDir + local MIGRATION_LOG_FILE="${NEW_DATA_DIR}/log/${LOG_FILE_NAME}" + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + MIGRATION_LOG_FILE="${SCRIPT_HOME}/${LOG_FILE_NAME}" + fi + touch "${MIGRATION_LOG_FILE}" + setFilePermission "${LOG_FILE_PERMISSION}" "${MIGRATION_LOG_FILE}" + exec &> >(tee -a "${MIGRATION_LOG_FILE}") +} +# Set path where system.yaml should create +setSystemYamlPath () { + SYSTEM_YAML_PATH="${NEW_DATA_DIR}/etc/system.yaml" + if [[ "${INSTALLER}" != "${HELM_TYPE}" ]]; then + logger "system.yaml will be created in path [${SYSTEM_YAML_PATH}]" + fi +} +# Create directory +createDirectory () { + local directory="$1" + local output="$2" + local check=false + local message="Could not create directory ${directory}, please check if the user ${USER} has permissions to perform this action" + removeSoftLink "${directory}" + mkdir -p "${directory}" && check=true || check=false + if [[ "${check}" == "false" ]]; then + if [[ "${output}" == "Warning" ]]; then + warn "${message}" + else + errorExit "${message}" + fi + fi + setOwnershipBasedOnInstaller "${directory}" +} + +setOwnershipBasedOnInstaller () { + local directory="$1" + if [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + getUserAndGroupFromFile + chown -R ${USER_TO_CHECK}:${GROUP_TO_CHECK} "${directory}" || warn "Setting ownership on $directory failed" + elif [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + io_setOwnership "${directory}" "${JF_USER}" "${JF_USER}" + fi +} + +getUserAndGroup () { + local file="$1" + read uid gid <<<$(stat -c '%U %G' ${file}) + USER_TO_CHECK="${uid}" + GROUP_TO_CHECK="${gid}" +} + +# set ownership +getUserAndGroupFromFile () { + case $PRODUCT in + artifactory) + getUserAndGroup "/etc/opt/jfrog/artifactory/artifactory.properties" + ;; + distribution) + getUserAndGroup "${OLD_DATA_DIR}/etc/versions.properties" + ;; + xray) + getUserAndGroup "${OLD_DATA_DIR}/security/master.key" + ;; + esac +} + +# creating required directories +createRequiredDirs () { + bannerSubSection "CREATING REQUIRED DIRECTORIES" + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/etc/security" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/data" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/log/archived" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/work" "${JF_USER}" "${JF_USER}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/backup" "${JF_USER}" "${JF_USER}" "yes" + io_setOwnership "${NEW_DATA_DIR}" "${JF_USER}" "${JF_USER}" + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" ]]; then + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/data/postgres" "${POSTGRES_USER}" "${POSTGRES_USER}" "yes" + fi + elif [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + getUserAndGroupFromFile + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/etc" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/etc/security" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/data" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/log/archived" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/work" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + removeSoftLinkAndCreateDir "${NEW_DATA_DIR}/backup" "${USER_TO_CHECK}" "${GROUP_TO_CHECK}" "yes" + fi +} + +# Check entry in map is format +checkMapEntry () { + local entry="$1" + + [[ "${entry}" != *"="* ]] && echo -n "false" || echo -n "true" +} +# Check value Empty and warn +warnIfEmpty () { + local filePath="$1" + local yamlPath="$2" + local check= + + if [[ -z "${filePath}" ]]; then + warn "Empty value in yamlpath [${yamlPath} in [${MIGRATION_SYSTEM_YAML_INFO}]" + check=false + else + check=true + fi + echo "${check}" +} + +logCopyStatus () { + local status="$1" + local logMessage="$2" + local warnMessage="$3" + + [[ "${status}" == "success" ]] && logger "${logMessage}" + [[ "${status}" == "fail" ]] && warn "${warnMessage}" +} +# copy contents from source to destination +copyCmd () { + local source="$1" + local target="$2" + local mode="$3" + local status= + + case $mode in + unique) + cp -up "${source}"/* "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied directory contents from [${source}] to [${target}]" "Failed to copy directory contents from [${source}] to [${target}]" + ;; + specific) + cp -pf "${source}" "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied file [${source}] to [${target}]" "Failed to copy file [${source}] to [${target}]" + ;; + patternFiles) + cp -pf "${source}"* "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied files matching [${source}*] to [${target}]" "Failed to copy files matching [${source}*] to [${target}]" + ;; + full) + cp -prf "${source}"/* "${target}"/ && status="success" || status="fail" + logCopyStatus "${status}" "Successfully copied directory contents from [${source}] to [${target}]" "Failed to copy directory contents from [${source}] to [${target}]" + ;; + esac +} +# Check contents exist in source before copying +copyOnContentExist () { + local source="$1" + local target="$2" + local mode="$3" + + if [[ "$(checkContentExists "${source}")" == "true" ]]; then + copyCmd "${source}" "${target}" "${mode}" + else + logger "No contents to copy from [${source}]" + fi +} + +# move source to destination +moveCmd () { + local source="$1" + local target="$2" + local status= + + mv -f "${source}" "${target}" && status="success" || status="fail" + [[ "${status}" == "success" ]] && logger "Successfully moved directory [${source}] to [${target}]" + [[ "${status}" == "fail" ]] && warn "Failed to move directory [${source}] to [${target}]" +} + +# symlink target to source +symlinkCmd () { + local source="$1" + local target="$2" + local symlinkSubDir="$3" + local check=false + + if [[ "${symlinkSubDir}" == "subDir" ]]; then + ln -sf "${source}"/* "${target}" && check=true || check=false + else + ln -sf "${source}" "${target}" && check=true || check=false + fi + + [[ "${check}" == "true" ]] && logger "Successfully symlinked directory [${target}] to old [${source}]" + [[ "${check}" == "false" ]] && warn "Symlink operation failed" +} +# Check contents exist in source before symlinking +symlinkOnExist () { + local source="$1" + local target="$2" + local symlinkSubDir="$3" + + if [[ "$(checkContentExists "${source}")" == "true" ]]; then + if [[ "${symlinkSubDir}" == "subDir" ]]; then + symlinkCmd "${source}" "${target}" "subDir" + else + symlinkCmd "${source}" "${target}" + fi + else + logger "No contents to symlink from [${source}]" + fi +} + +prependDir () { + local absolutePath="$1" + local fullPath="$2" + local sourcePath= + + if [[ "${absolutePath}" = \/* ]]; then + sourcePath="${absolutePath}" + else + sourcePath="${fullPath}" + fi + echo "${sourcePath}" +} + +getFirstEntry (){ + local entry="$1" + + [[ -z "${entry}" ]] && return + echo "${entry}" | awk -F"=" '{print $1}' +} + +getSecondEntry () { + local entry="$1" + + [[ -z "${entry}" ]] && return + echo "${entry}" | awk -F"=" '{print $2}' +} +# To get absolutePath +pathResolver () { + local directoryPath="$1" + local dataDir= + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + retrieveYamlValue "migration.oldDataDir" "oldDataDir" "Warning" + dataDir="${YAML_VALUE}" + cd "${dataDir}" + else + cd "${OLD_DATA_DIR}" + fi + absoluteDir="`cd "${directoryPath}";pwd`" + echo "${absoluteDir}" +} + +checkPathResolver () { + local value="$1" + + if [[ "${value}" == \/* ]]; then + value="${value}" + else + value="$(pathResolver "${value}")" + fi + echo "${value}" +} + +propertyMigrate () { + local entry="$1" + local filePath="$2" + local fileName="$3" + local check=false + + local yamlPath="$(getFirstEntry "${entry}")" + local property="$(getSecondEntry "${entry}")" + if [[ -z "${property}" ]]; then + warn "Property is empty in map [${entry}] in the file [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + if [[ -z "${yamlPath}" ]]; then + warn "yamlPath is empty for [${property}] in [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + local keyValues=$(cat "${NEW_DATA_DIR}/${filePath}/${fileName}" | grep "^[^#]" | grep "[*=*]") + for i in ${keyValues}; do + key=$(echo "${i}" | awk -F"=" '{print $1}') + value=$(echo "${i}" | cut -f 2- -d '=') + [ -z "${key}" ] && continue + [ -z "${value}" ] && continue + if [[ "${key}" == "${property}" ]]; then + if [[ "${PRODUCT}" == "artifactory" ]]; then + value="$(migrateResolveDerbyPath "${key}" "${value}")" + value="$(migrateResolveHaDirPath "${key}" "${value}")" + if [[ "${INSTALLER}" != "${DOCKER_TYPE}" ]]; then + value="$(updatePostgresUrlString_Hook "${yamlPath}" "${value}")" + fi + fi + if [[ "${key}" == "context.url" ]]; then + local ip=$(echo "${value}" | awk -F/ '{print $3}' | sed 's/:.*//') + setSystemValue "shared.node.ip" "${ip}" "${SYSTEM_YAML_PATH}" + logger "Setting [shared.node.ip] with [${ip}] in system.yaml" + fi + setSystemValue "${yamlPath}" "${value}" "${SYSTEM_YAML_PATH}" && logger "Setting [${yamlPath}] with value of the property [${property}] in system.yaml" && check=true && break || check=false + fi + done + [[ "${check}" == "false" ]] && logger "Property [${property}] not found in file [${fileName}]" +} + +setHaEnabled_hook () { + echo "" +} + +migratePropertiesFiles () { + local fileList= + local filePath= + local fileName= + local map= + + retrieveYamlValue "migration.propertyFiles.files" "fileList" "Skip" + fileList="${YAML_VALUE}" + if [[ -z "${fileList}" ]]; then + return + fi + bannerSection "PROCESSING MIGRATION OF PROPERTY FILES" + for file in ${fileList}; + do + bannerSubSection "Processing Migration of $file" + retrieveYamlValue "migration.propertyFiles.$file.filePath" "filePath" "Warning" + filePath="${YAML_VALUE}" + retrieveYamlValue "migration.propertyFiles.$file.fileName" "fileName" "Warning" + fileName="${YAML_VALUE}" + [[ -z "${filePath}" && -z "${fileName}" ]] && continue + if [[ "$(checkFileExists "${NEW_DATA_DIR}/${filePath}/${fileName}")" == "true" ]]; then + logger "File [${fileName}] found in path [${NEW_DATA_DIR}/${filePath}]" + # setting haEnabled with true only if ha-node.properties is present + setHaEnabled_hook "${filePath}" + retrieveYamlValue "migration.propertyFiles.$file.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + propertyMigrate "${entry}" "${filePath}" "${fileName}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e yamlPath=property" + fi + done + else + logger "File [${fileName}] was not found in path [${NEW_DATA_DIR}/${filePath}] to migrate" + fi + done +} + +createTargetDir () { + local mountDir="$1" + local target="$2" + + logger "Target directory not found [${mountDir}/${target}], creating it" + createDirectoryRecursive "${mountDir}" "${target}" "Warning" +} + +createDirectoryRecursive () { + local mountDir="$1" + local target="$2" + local output="$3" + local check=false + local message="Could not create directory ${directory}, please check if the user ${USER} has permissions to perform this action" + removeSoftLink "${mountDir}/${target}" + local directory=$(echo "${target}" | tr '/' ' ' ) + local targetDir="${mountDir}" + for dir in ${directory}; + do + targetDir="${targetDir}/${dir}" + mkdir -p "${targetDir}" && check=true || check=false + setOwnershipBasedOnInstaller "${targetDir}" + done + if [[ "${check}" == "false" ]]; then + if [[ "${output}" == "Warning" ]]; then + warn "${message}" + else + errorExit "${message}" + fi + fi +} + +copyOperation () { + local source="$1" + local target="$2" + local mode="$3" + local check=false + local targetDataDir= + local targetLink= + local date= + + # prepend OLD_DATA_DIR only if source is relative path + source="$(prependDir "${source}" "${OLD_DATA_DIR}/${source}")" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + copyLogMessage "${mode}" + #remove source if it is a symlink + if [[ -L "${source}" ]]; then + targetLink=$(readlink -f "${source}") + logger "Removing the symlink [${source}] pointing to [${targetLink}]" + rm -f "${source}" + source=${targetLink} + fi + if [[ "$(checkDirExists "${source}")" != "true" ]]; then + logger "Source [${source}] directory not found in path" + return + fi + if [[ "$(checkDirContents "${source}")" != "true" ]]; then + logger "No contents to copy from [${source}]" + return + fi + if [[ "$(checkDirExists "${targetDataDir}/${target}")" != "true" ]]; then + createTargetDir "${targetDataDir}" "${target}" + fi + copyOnContentExist "${source}" "${targetDataDir}/${target}" "${mode}" +} + +copySpecificFiles () { + local source="$1" + local target="$2" + local mode="$3" + + # prepend OLD_DATA_DIR only if source is relative path + source="$(prependDir "${source}" "${OLD_DATA_DIR}/${source}")" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + copyLogMessage "${mode}" + if [[ "$(checkFileExists "${source}")" != "true" ]]; then + logger "Source file [${source}] does not exist in path" + return + fi + if [[ "$(checkDirExists "${targetDataDir}/${target}")" != "true" ]]; then + createTargetDir "${targetDataDir}" "${target}" + fi + copyCmd "${source}" "${targetDataDir}/${target}" "${mode}" +} + +copyPatternMatchingFiles () { + local source="$1" + local target="$2" + local mode="$3" + local sourcePath="${4}" + + # prepend OLD_DATA_DIR only if source is relative path + sourcePath="$(prependDir "${sourcePath}" "${OLD_DATA_DIR}/${sourcePath}")" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + copyLogMessage "${mode}" + if [[ "$(checkDirExists "${sourcePath}")" != "true" ]]; then + logger "Source [${sourcePath}] directory not found in path" + return + fi + if ls "${sourcePath}/${source}"* 1> /dev/null 2>&1; then + if [[ "$(checkDirExists "${targetDataDir}/${target}")" != "true" ]]; then + createTargetDir "${targetDataDir}" "${target}" + fi + copyCmd "${sourcePath}/${source}" "${targetDataDir}/${target}" "${mode}" + else + logger "Source file [${sourcePath}/${source}*] does not exist in path" + fi +} + +copyLogMessage () { + local mode="$1" + case $mode in + specific) + logger "Copy file [${source}] to target [${targetDataDir}/${target}]" + ;; + patternFiles) + logger "Copy files matching [${sourcePath}/${source}*] to target [${targetDataDir}/${target}]" + ;; + full) + logger "Copy directory contents from source [${source}] to target [${targetDataDir}/${target}]" + ;; + unique) + logger "Copy directory contents from source [${source}] to target [${targetDataDir}/${target}]" + ;; + esac +} + +copyBannerMessages () { + local mode="$1" + local textMode="$2" + case $mode in + specific) + bannerSection "COPY ${textMode} FILES" + ;; + patternFiles) + bannerSection "COPY MATCHING ${textMode}" + ;; + full) + bannerSection "COPY ${textMode} DIRECTORIES CONTENTS" + ;; + unique) + bannerSection "COPY ${textMode} DIRECTORIES CONTENTS" + ;; + esac +} + +invokeCopyFunctions () { + local mode="$1" + local source="$2" + local target="$3" + + case $mode in + specific) + copySpecificFiles "${source}" "${target}" "${mode}" + ;; + patternFiles) + retrieveYamlValue "migration.${copyFormat}.sourcePath" "map" "Warning" + local sourcePath="${YAML_VALUE}" + copyPatternMatchingFiles "${source}" "${target}" "${mode}" "${sourcePath}" + ;; + full) + copyOperation "${source}" "${target}" "${mode}" + ;; + unique) + copyOperation "${source}" "${target}" "${mode}" + ;; + esac +} +# Copies contents from source directory and target directory +copyDataDirectories () { + local copyFormat="$1" + local mode="$2" + local map= + local source= + local target= + local textMode= + local targetDataDir= + local copyFormatValue= + + retrieveYamlValue "migration.${copyFormat}" "${copyFormat}" "Skip" + copyFormatValue="${YAML_VALUE}" + if [[ -z "${copyFormatValue}" ]]; then + return + fi + textMode=$(echo "${mode}" | tr '[:lower:]' '[:upper:]' 2>/dev/null) + copyBannerMessages "${mode}" "${textMode}" + retrieveYamlValue "migration.${copyFormat}.map" "map" "Warning" + map="${YAML_VALUE}" + if [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + targetDataDir="${NEW_DATA_DIR}" + else + targetDataDir="`cd "${NEW_DATA_DIR}"/../;pwd`" + fi + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + source="$(getSecondEntry "${entry}")" + target="$(getFirstEntry "${entry}")" + [[ -z "${source}" ]] && warn "source value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${target}" ]] && warn "target value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + invokeCopyFunctions "${mode}" "${source}" "${target}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e target=source" + fi + echo ""; + done +} + +invokeMoveFunctions () { + local source="$1" + local target="$2" + local sourceDataDir= + local targetBasename= + # prepend OLD_DATA_DIR only if source is relative path + sourceDataDir=$(prependDir "${source}" "${OLD_DATA_DIR}/${source}") + targetBasename=$(dirname "${target}") + logger "Moving directory source [${sourceDataDir}] to target [${NEW_DATA_DIR}/${target}]" + if [[ "$(checkDirExists "${sourceDataDir}")" != "true" ]]; then + logger "Directory [${sourceDataDir}] not found in path to move" + return + fi + if [[ "$(checkDirExists "${NEW_DATA_DIR}/${targetBasename}")" != "true" ]]; then + createTargetDir "${NEW_DATA_DIR}" "${targetBasename}" + moveCmd "${sourceDataDir}" "${NEW_DATA_DIR}/${target}" + else + moveCmd "${sourceDataDir}" "${NEW_DATA_DIR}/tempDir" + moveCmd "${NEW_DATA_DIR}/tempDir" "${NEW_DATA_DIR}/${target}" + fi +} + +# Move source directory and target directory +moveDirectories () { + local moveDataDirectories= + local map= + local source= + local target= + + retrieveYamlValue "migration.moveDirectories" "moveDirectories" "Skip" + moveDirectories="${YAML_VALUE}" + if [[ -z "${moveDirectories}" ]]; then + return + fi + bannerSection "MOVE DIRECTORIES" + retrieveYamlValue "migration.moveDirectories.map" "map" "Warning" + map="${YAML_VALUE}" + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + source="$(getSecondEntry "${entry}")" + target="$(getFirstEntry "${entry}")" + [[ -z "${source}" ]] && warn "source value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${target}" ]] && warn "target value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + invokeMoveFunctions "${source}" "${target}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e target=source" + fi + echo ""; + done +} + +# Trim masterKey if its generated using hex 32 +trimMasterKey () { + local masterKeyDir=/opt/jfrog/artifactory/var/etc/security + local oldMasterKey=$(<${masterKeyDir}/master.key) + local oldMasterKey_Length=$(echo ${#oldMasterKey}) + local newMasterKey= + if [[ ${oldMasterKey_Length} -gt 32 ]]; then + bannerSection "TRIM MASTERKEY" + newMasterKey=$(echo ${oldMasterKey:0:32}) + cp ${masterKeyDir}/master.key ${masterKeyDir}/backup_master.key + logger "Original masterKey is backed up : ${masterKeyDir}/backup_master.key" + rm -rf ${masterKeyDir}/master.key + echo ${newMasterKey} > ${masterKeyDir}/master.key + logger "masterKey is trimmed : ${masterKeyDir}/master.key" + fi +} + +copyDirectories () { + + copyDataDirectories "copyFiles" "full" + copyDataDirectories "copyUniqueFiles" "unique" + copyDataDirectories "copySpecificFiles" "specific" + copyDataDirectories "copyPatternMatchingFiles" "patternFiles" +} + +symlinkDir () { + local source="$1" + local target="$2" + local targetDir= + local basename= + local targetParentDir= + + targetDir="$(dirname "${target}")" + if [[ "${targetDir}" == "${source}" ]]; then + # symlink the sub directories + createDirectory "${NEW_DATA_DIR}/${target}" "Warning" + if [[ "$(checkDirExists "${NEW_DATA_DIR}/${target}")" == "true" ]]; then + symlinkOnExist "${OLD_DATA_DIR}/${source}" "${NEW_DATA_DIR}/${target}" "subDir" + basename="$(basename "${target}")" + cd "${NEW_DATA_DIR}/${target}" && rm -f "${basename}" + fi + else + targetParentDir="$(dirname "${NEW_DATA_DIR}/${target}")" + createDirectory "${targetParentDir}" "Warning" + if [[ "$(checkDirExists "${targetParentDir}")" == "true" ]]; then + symlinkOnExist "${OLD_DATA_DIR}/${source}" "${NEW_DATA_DIR}/${target}" + fi + fi +} + +symlinkOperation () { + local source="$1" + local target="$2" + local check=false + local targetLink= + local date= + + # Check if source is a link and do symlink + if [[ -L "${OLD_DATA_DIR}/${source}" ]]; then + targetLink=$(readlink -f "${OLD_DATA_DIR}/${source}") + symlinkOnExist "${targetLink}" "${NEW_DATA_DIR}/${target}" + else + # check if source is directory and do symlink + if [[ "$(checkDirExists "${OLD_DATA_DIR}/${source}")" != "true" ]]; then + logger "Source [${source}] directory not found in path to symlink" + return + fi + if [[ "$(checkDirContents "${OLD_DATA_DIR}/${source}")" != "true" ]]; then + logger "No contents found in [${OLD_DATA_DIR}/${source}] to symlink" + return + fi + if [[ "$(checkDirExists "${NEW_DATA_DIR}/${target}")" != "true" ]]; then + logger "Target directory [${NEW_DATA_DIR}/${target}] does not exist to create symlink, creating it" + symlinkDir "${source}" "${target}" + else + rm -rf "${NEW_DATA_DIR}/${target}" && check=true || check=false + [[ "${check}" == "false" ]] && warn "Failed to remove contents in [${NEW_DATA_DIR}/${target}/]" + symlinkDir "${source}" "${target}" + fi + fi +} +# Creates a symlink path - Source directory to which the symbolic link should point. +symlinkDirectories () { + local linkFiles= + local map= + local source= + local target= + + retrieveYamlValue "migration.linkFiles" "linkFiles" "Skip" + linkFiles="${YAML_VALUE}" + if [[ -z "${linkFiles}" ]]; then + return + fi + bannerSection "SYMLINK DIRECTORIES" + retrieveYamlValue "migration.linkFiles.map" "map" "Warning" + map="${YAML_VALUE}" + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + source="$(getSecondEntry "${entry}")" + target="$(getFirstEntry "${entry}")" + logger "Symlink directory [${NEW_DATA_DIR}/${target}] to old [${OLD_DATA_DIR}/${source}]" + [[ -z "${source}" ]] && warn "source value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${target}" ]] && warn "target value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + symlinkOperation "${source}" "${target}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e target=source" + fi + echo ""; + done +} + +updateConnectionString () { + local yamlPath="$1" + local value="$2" + local mongoPath="shared.mongo.url" + local rabbitmqPath="shared.rabbitMq.url" + local postgresPath="shared.database.url" + local redisPath="shared.redis.connectionString" + local mongoConnectionString="mongo.connectionString" + local sourceKey= + local hostIp=$(io_getPublicHostIP) + local hostKey= + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + # Replace @postgres:,@mongodb:,@rabbitmq:,@redis: to @{hostIp}: (Compose Installer) + hostKey="@${hostIp}:" + case $yamlPath in + ${postgresPath}) + sourceKey="@postgres:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${mongoPath}) + sourceKey="@mongodb:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${rabbitmqPath}) + sourceKey="@rabbitmq:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${redisPath}) + sourceKey="@redis:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + ${mongoConnectionString}) + sourceKey="@mongodb:" + value=$(io_replaceString "${value}" "${sourceKey}" "${hostKey}") + ;; + esac + fi + echo -n "${value}" +} + +yamlMigrate () { + local entry="$1" + local sourceFile="$2" + local value= + local yamlPath= + local key= + yamlPath="$(getFirstEntry "${entry}")" + key="$(getSecondEntry "${entry}")" + if [[ -z "${key}" ]]; then + warn "key is empty in map [${entry}] in the file [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + if [[ -z "${yamlPath}" ]]; then + warn "yamlPath is empty for [${key}] in [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + getYamlValue "${key}" "${sourceFile}" "false" + value="${YAML_VALUE}" + if [[ ! -z "${value}" ]]; then + value=$(updateConnectionString "${yamlPath}" "${value}") + fi + if [[ -z "${value}" ]]; then + logger "No value for [${key}] in [${sourceFile}]" + else + setSystemValue "${yamlPath}" "${value}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value of the key [${key}] in system.yaml" + fi +} + +migrateYamlFile () { + local files= + local filePath= + local fileName= + local sourceFile= + local map= + retrieveYamlValue "migration.yaml.files" "files" "Skip" + files="${YAML_VALUE}" + if [[ -z "${files}" ]]; then + return + fi + bannerSection "MIGRATION OF YAML FILES" + for file in $files; + do + bannerSubSection "Processing Migration of $file" + retrieveYamlValue "migration.yaml.$file.filePath" "filePath" "Warning" + filePath="${YAML_VALUE}" + retrieveYamlValue "migration.yaml.$file.fileName" "fileName" "Warning" + fileName="${YAML_VALUE}" + [[ -z "${filePath}" && -z "${fileName}" ]] && continue + sourceFile="${NEW_DATA_DIR}/${filePath}/${fileName}" + if [[ "$(checkFileExists "${sourceFile}")" == "true" ]]; then + logger "File [${fileName}] found in path [${NEW_DATA_DIR}/${filePath}]" + retrieveYamlValue "migration.yaml.$file.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + yamlMigrate "${entry}" "${sourceFile}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e yamlPath=key" + fi + done + else + logger "File [${fileName}] is not found in path [${NEW_DATA_DIR}/${filePath}] to migrate" + fi + done +} +# updates the key and value in system.yaml +updateYamlKeyValue () { + local entry="$1" + local value= + local yamlPath= + local key= + + yamlPath="$(getFirstEntry "${entry}")" + value="$(getSecondEntry "${entry}")" + if [[ -z "${value}" ]]; then + warn "value is empty in map [${entry}] in the file [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + if [[ -z "${yamlPath}" ]]; then + warn "yamlPath is empty for [${key}] in [${MIGRATION_SYSTEM_YAML_INFO}]" + return + fi + setSystemValue "${yamlPath}" "${value}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value [${value}] in system.yaml" +} + +updateSystemYamlFile () { + local updateYaml= + local map= + + retrieveYamlValue "migration.updateSystemYaml" "updateYaml" "Skip" + updateSystemYaml="${YAML_VALUE}" + if [[ -z "${updateSystemYaml}" ]]; then + return + fi + bannerSection "UPDATE SYSTEM YAML FILE WITH KEY AND VALUES" + retrieveYamlValue "migration.updateSystemYaml.map" "map" "Warning" + map="${YAML_VALUE}" + if [[ -z "${map}" ]]; then + return + fi + for entry in $map; + do + if [[ "$(checkMapEntry "${entry}")" == "true" ]]; then + updateYamlKeyValue "${entry}" + else + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e yamlPath=key" + fi + done +} + +backupFiles_hook () { + logSilly "Method ${FUNCNAME[0]}" +} + +backupDirectory () { + local backupDir="$1" + local dir="$2" + local targetDir="$3" + local effectiveUser= + local effectiveGroup= + + if [[ "${dir}" = \/* ]]; then + dir=$(echo "${dir/\//}") + fi + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + effectiveUser="${JF_USER}" + effectiveGroup="${JF_USER}" + elif [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + effectiveUser="${USER_TO_CHECK}" + effectiveGroup="${GROUP_TO_CHECK}" + fi + + removeSoftLinkAndCreateDir "${backupDir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local backupDirectory="${backupDir}/${PRODUCT}" + removeSoftLinkAndCreateDir "${backupDirectory}" "${effectiveUser}" "${effectiveGroup}" "yes" + removeSoftLinkAndCreateDir "${backupDirectory}/${dir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local outputCheckDirExists="$(checkDirExists "${backupDirectory}/${dir}")" + if [[ "${outputCheckDirExists}" == "true" ]]; then + copyOnContentExist "${targetDir}" "${backupDirectory}/${dir}" "full" + fi +} + +removeOldDirectory () { + local backupDir="$1" + local entry="$2" + local check=false + + # prepend OLD_DATA_DIR only if entry is relative path + local targetDir="$(prependDir "${entry}" "${OLD_DATA_DIR}/${entry}")" + local outputCheckDirExists="$(checkDirExists "${targetDir}")" + if [[ "${outputCheckDirExists}" != "true" ]]; then + logger "No [${targetDir}] directory found to delete" + echo ""; + return + fi + backupDirectory "${backupDir}" "${entry}" "${targetDir}" + rm -rf "${targetDir}" && check=true || check=false + [[ "${check}" == "true" ]] && logger "Successfully removed directory [${targetDir}]" + [[ "${check}" == "false" ]] && warn "Failed to remove directory [${targetDir}]" + echo ""; +} + +cleanUpOldDataDirectories () { + local cleanUpOldDataDir= + local map= + local entry= + + retrieveYamlValue "migration.cleanUpOldDataDir" "cleanUpOldDataDir" "Skip" + cleanUpOldDataDir="${YAML_VALUE}" + if [[ -z "${cleanUpOldDataDir}" ]]; then + return + fi + bannerSection "CLEAN UP OLD DATA DIRECTORIES" + retrieveYamlValue "migration.cleanUpOldDataDir.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + date="$(date +%Y%m%d%H%M)" + backupDir="${NEW_DATA_DIR}/backup/backup-${date}" + bannerImportant "****** Old data configurations are backedup in [${backupDir}] directory ******" + backupFiles_hook "${backupDir}/${PRODUCT}" + for entry in $map; + do + removeOldDirectory "${backupDir}" "${entry}" + done +} + +backupFiles () { + local backupDir="$1" + local dir="$2" + local targetDir="$3" + local fileName="$4" + local effectiveUser= + local effectiveGroup= + + if [[ "${dir}" = \/* ]]; then + dir=$(echo "${dir/\//}") + fi + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" ]]; then + effectiveUser="${JF_USER}" + effectiveGroup="${JF_USER}" + elif [[ "${INSTALLER}" == "${DEB_TYPE}" || "${INSTALLER}" == "${RPM_TYPE}" ]]; then + effectiveUser="${USER_TO_CHECK}" + effectiveGroup="${GROUP_TO_CHECK}" + fi + + removeSoftLinkAndCreateDir "${backupDir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local backupDirectory="${backupDir}/${PRODUCT}" + removeSoftLinkAndCreateDir "${backupDirectory}" "${effectiveUser}" "${effectiveGroup}" "yes" + removeSoftLinkAndCreateDir "${backupDirectory}/${dir}" "${effectiveUser}" "${effectiveGroup}" "yes" + local outputCheckDirExists="$(checkDirExists "${backupDirectory}/${dir}")" + if [[ "${outputCheckDirExists}" == "true" ]]; then + copyCmd "${targetDir}/${fileName}" "${backupDirectory}/${dir}" "specific" + fi +} + +removeOldFiles () { + local backupDir="$1" + local directoryName="$2" + local fileName="$3" + local check=false + + # prepend OLD_DATA_DIR only if entry is relative path + local targetDir="$(prependDir "${directoryName}" "${OLD_DATA_DIR}/${directoryName}")" + local outputCheckFileExists="$(checkFileExists "${targetDir}/${fileName}")" + if [[ "${outputCheckFileExists}" != "true" ]]; then + logger "No [${targetDir}/${fileName}] file found to delete" + return + fi + backupFiles "${backupDir}" "${directoryName}" "${targetDir}" "${fileName}" + rm -f "${targetDir}/${fileName}" && check=true || check=false + [[ "${check}" == "true" ]] && logger "Successfully removed file [${targetDir}/${fileName}]" + [[ "${check}" == "false" ]] && warn "Failed to remove file [${targetDir}/${fileName}]" + echo ""; +} + +cleanUpOldFiles () { + local cleanUpFiles= + local map= + local entry= + + retrieveYamlValue "migration.cleanUpOldFiles" "cleanUpOldFiles" "Skip" + cleanUpOldFiles="${YAML_VALUE}" + if [[ -z "${cleanUpOldFiles}" ]]; then + return + fi + bannerSection "CLEAN UP OLD FILES" + retrieveYamlValue "migration.cleanUpOldFiles.map" "map" "Warning" + map="${YAML_VALUE}" + [[ -z "${map}" ]] && continue + date="$(date +%Y%m%d%H%M)" + backupDir="${NEW_DATA_DIR}/backup/backup-${date}" + bannerImportant "****** Old files are backedup in [${backupDir}] directory ******" + for entry in $map; + do + local outputCheckMapEntry="$(checkMapEntry "${entry}")" + if [[ "${outputCheckMapEntry}" != "true" ]]; then + warn "map entry [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}] is not in correct format, correct format i.e directoryName=fileName" + fi + local fileName="$(getSecondEntry "${entry}")" + local directoryName="$(getFirstEntry "${entry}")" + [[ -z "${fileName}" ]] && warn "File name value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + [[ -z "${directoryName}" ]] && warn "Directory name value is empty for [${entry}] in [${MIGRATION_SYSTEM_YAML_INFO}]" && continue + removeOldFiles "${backupDir}" "${directoryName}" "${fileName}" + echo ""; + done +} + +startMigration () { + bannerSection "STARTING MIGRATION" +} + +endMigration () { + bannerSection "MIGRATION COMPLETED SUCCESSFULLY" +} + +initialize () { + setAppDir + _pauseExecution "setAppDir" + initHelpers + _pauseExecution "initHelpers" + checkMigrationInfoYaml + _pauseExecution "checkMigrationInfoYaml" + getProduct + _pauseExecution "getProduct" + getDataDir + _pauseExecution "getDataDir" +} + +main () { + case $PRODUCT in + artifactory) + migrateArtifactory + ;; + distribution) + migrateDistribution + ;; + xray) + migrationXray + ;; + esac + exit 0 +} + +# Ensures meta data is logged +LOG_BEHAVIOR_ADD_META="$FLAG_Y" + + +migrateResolveDerbyPath () { + local key="$1" + local value="$2" + + if [[ "${key}" == "url" && "${value}" == *"db.home"* ]]; then + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" ]]; then + derbyPath="/opt/jfrog/artifactory/var/data/artifactory/derby" + value=$(echo "${value}" | sed "s|{db.home}|$derbyPath|") + else + derbyPath="${NEW_DATA_DIR}/data/artifactory/derby" + value=$(echo "${value}" | sed "s|{db.home}|$derbyPath|") + fi + fi + echo "${value}" +} + +migrateResolveHaDirPath () { + local key="$1" + local value="$2" + + if [[ "${INSTALLER}" == "${RPM_TYPE}" || "${INSTALLER}" == "${COMPOSE_TYPE}" || "${INSTALLER}" == "${HELM_TYPE}" || "${INSTALLER}" == "${DEB_TYPE}" ]]; then + if [[ "${key}" == "artifactory.ha.data.dir" || "${key}" == "artifactory.ha.backup.dir" ]]; then + value=$(checkPathResolver "${value}") + fi + fi + echo "${value}" +} +updatePostgresUrlString_Hook () { + local yamlPath="$1" + local value="$2" + local hostIp=$(io_getPublicHostIP) + local sourceKey="//postgresql:" + if [[ "${yamlPath}" == "shared.database.url" ]]; then + value=$(io_replaceString "${value}" "${sourceKey}" "//${hostIp}:" "#") + fi + echo "${value}" +} +# Check Artifactory product version +checkArtifactoryVersion () { + local minProductVersion="6.0.0" + local maxProductVersion="7.0.0" + local propertyInDocker="ARTIFACTORY_VERSION" + local property="artifactory.version" + + if [[ "${INSTALLER}" == "${COMPOSE_TYPE}" ]]; then + local newfilePath="${APP_DIR}/../.env" + local oldfilePath="${OLD_DATA_DIR}/etc/artifactory.properties" + elif [[ "${INSTALLER}" == "${HELM_TYPE}" ]]; then + local oldfilePath="${OLD_DATA_DIR}/etc/artifactory.properties" + elif [[ "${INSTALLER}" == "${ZIP_TYPE}" ]]; then + local newfilePath="${NEW_DATA_DIR}/etc/artifactory/artifactory.properties" + local oldfilePath="${OLD_DATA_DIR}/etc/artifactory.properties" + else + local newfilePath="${NEW_DATA_DIR}/etc/artifactory/artifactory.properties" + local oldfilePath="/etc/opt/jfrog/artifactory/artifactory.properties" + fi + + getProductVersion "${minProductVersion}" "${maxProductVersion}" "${newfilePath}" "${oldfilePath}" "${propertyInDocker}" "${property}" +} + +getCustomDataDir_hook () { + retrieveYamlValue "migration.oldDataDir" "oldDataDir" "Fail" + OLD_DATA_DIR="${YAML_VALUE}" +} + +# Get protocol value of connector +getXmlConnectorProtocol () { + local i="$1" + local filePath="$2" + local fileName="$3" + local protocolValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@protocol' ${filePath}/${fileName} 2>/dev/null |awk -F"=" '{print $2}' | tr -d '"') + echo -e "${protocolValue}" +} + +# Get all attributes of connector +getXmlConnectorAttributes () { + local i="$1" + local filePath="$2" + local fileName="$3" + local connectorAttributes=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@*' ${filePath}/${fileName} 2>/dev/null) + # strip leading and trailing spaces + connectorAttributes=$(io_trim "${connectorAttributes}") + echo "${connectorAttributes}" +} + +# Get port value of connector +getXmlConnectorPort () { + local i="$1" + local filePath="$2" + local fileName="$3" + local portValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@port' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + echo -e "${portValue}" +} + +# Get maxThreads value of connector +getXmlConnectorMaxThreads () { + local i="$1" + local filePath="$2" + local fileName="$3" + local maxThreadValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@maxThreads' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + echo -e "${maxThreadValue}" +} +# Get sendReasonPhrase value of connector +getXmlConnectorSendReasonPhrase () { + local i="$1" + local filePath="$2" + local fileName="$3" + local sendReasonPhraseValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@sendReasonPhrase' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + echo -e "${sendReasonPhraseValue}" +} +# Get relaxedPathChars value of connector +getXmlConnectorRelaxedPathChars () { + local i="$1" + local filePath="$2" + local fileName="$3" + local relaxedPathCharsValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@relaxedPathChars' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + # strip leading and trailing spaces + relaxedPathCharsValue=$(io_trim "${relaxedPathCharsValue}") + echo -e "${relaxedPathCharsValue}" +} +# Get relaxedQueryChars value of connector +getXmlConnectorRelaxedQueryChars () { + local i="$1" + local filePath="$2" + local fileName="$3" + local relaxedQueryCharsValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@relaxedQueryChars' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + # strip leading and trailing spaces + relaxedQueryCharsValue=$(io_trim "${relaxedQueryCharsValue}") + echo -e "${relaxedQueryCharsValue}" +} + +# Updating system.yaml with Connector port +setConnectorPort () { + local yamlPath="$1" + local valuePort="$2" + local portYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${valuePort}" ]]; then + warn "port value is empty, could not migrate to system.yaml" + return + fi + ## Getting port yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" portYamlPath "Warning" + portYamlPath="${YAML_VALUE}" + if [[ -z "${portYamlPath}" ]]; then + return + fi + setSystemValue "${portYamlPath}" "${valuePort}" "${SYSTEM_YAML_PATH}" + logger "Setting [${portYamlPath}] with value [${valuePort}] in system.yaml" +} + +# Updating system.yaml with Connector maxThreads +setConnectorMaxThread () { + local yamlPath="$1" + local threadValue="$2" + local maxThreadYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${threadValue}" ]]; then + return + fi + ## Getting max Threads yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" maxThreadYamlPath "Warning" + maxThreadYamlPath="${YAML_VALUE}" + if [[ -z "${maxThreadYamlPath}" ]]; then + return + fi + setSystemValue "${maxThreadYamlPath}" "${threadValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${maxThreadYamlPath}] with value [${threadValue}] in system.yaml" +} + +# Updating system.yaml with Connector sendReasonPhrase +setConnectorSendReasonPhrase () { + local yamlPath="$1" + local sendReasonPhraseValue="$2" + local sendReasonPhraseYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${sendReasonPhraseValue}" ]]; then + return + fi + ## Getting sendReasonPhrase yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" sendReasonPhraseYamlPath "Warning" + sendReasonPhraseYamlPath="${YAML_VALUE}" + if [[ -z "${sendReasonPhraseYamlPath}" ]]; then + return + fi + setSystemValue "${sendReasonPhraseYamlPath}" "${sendReasonPhraseValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${sendReasonPhraseYamlPath}] with value [${sendReasonPhraseValue}] in system.yaml" +} + +# Updating system.yaml with Connector relaxedPathChars +setConnectorRelaxedPathChars () { + local yamlPath="$1" + local relaxedPathCharsValue="$2" + local relaxedPathCharsYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${relaxedPathCharsValue}" ]]; then + return + fi + ## Getting relaxedPathChars yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" relaxedPathCharsYamlPath "Warning" + relaxedPathCharsYamlPath="${YAML_VALUE}" + if [[ -z "${relaxedPathCharsYamlPath}" ]]; then + return + fi + setSystemValue "${relaxedPathCharsYamlPath}" "${relaxedPathCharsValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${relaxedPathCharsYamlPath}] with value [${relaxedPathCharsValue}] in system.yaml" +} + +# Updating system.yaml with Connector relaxedQueryChars +setConnectorRelaxedQueryChars () { + local yamlPath="$1" + local relaxedQueryCharsValue="$2" + local relaxedQueryCharsYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${relaxedQueryCharsValue}" ]]; then + return + fi + ## Getting relaxedQueryChars yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" relaxedQueryCharsYamlPath "Warning" + relaxedQueryCharsYamlPath="${YAML_VALUE}" + if [[ -z "${relaxedQueryCharsYamlPath}" ]]; then + return + fi + setSystemValue "${relaxedQueryCharsYamlPath}" "${relaxedQueryCharsValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${relaxedQueryCharsYamlPath}] with value [${relaxedQueryCharsValue}] in system.yaml" +} + +# Updating system.yaml with Connectors configurations +setConnectorExtraConfig () { + local yamlPath="$1" + local connectorAttributes="$2" + local extraConfigPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${connectorAttributes}" ]]; then + return + fi + ## Getting extraConfig yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" extraConfig "Warning" + extraConfigPath="${YAML_VALUE}" + if [[ -z "${extraConfigPath}" ]]; then + return + fi + # strip leading and trailing spaces + connectorAttributes=$(io_trim "${connectorAttributes}") + setSystemValue "${extraConfigPath}" "${connectorAttributes}" "${SYSTEM_YAML_PATH}" + logger "Setting [${extraConfigPath}] with connector attributes in system.yaml" +} + +# Updating system.yaml with extra Connectors +setExtraConnector () { + local yamlPath="$1" + local extraConnector="$2" + local extraConnectorYamlPath= + if [[ -z "${yamlPath}" ]]; then + return + fi + if [[ -z "${extraConnector}" ]]; then + return + fi + ## Getting extraConnecotr yaml path from migration info yaml + retrieveYamlValue "${yamlPath}" extraConnectorYamlPath "Warning" + extraConnectorYamlPath="${YAML_VALUE}" + if [[ -z "${extraConnectorYamlPath}" ]]; then + return + fi + getYamlValue "${extraConnectorYamlPath}" "${SYSTEM_YAML_PATH}" "false" + local connectorExtra="${YAML_VALUE}" + if [[ -z "${connectorExtra}" ]]; then + setSystemValue "${extraConnectorYamlPath}" "${extraConnector}" "${SYSTEM_YAML_PATH}" + logger "Setting [${extraConnectorYamlPath}] with extra connectors in system.yaml" + else + setSystemValue "${extraConnectorYamlPath}" "\"${connectorExtra} ${extraConnector}\"" "${SYSTEM_YAML_PATH}" + logger "Setting [${extraConnectorYamlPath}] with extra connectors in system.yaml" + fi +} + +# Migrate extra connectors to system.yaml +migrateExtraConnectors () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local excludeDefaultPort="$4" + local i="$5" + local extraConfig= + local extraConnector= + if [[ "${excludeDefaultPort}" == "yes" ]]; then + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + [[ "${portValue}" != "${DEFAULT_ACCESS_PORT}" && "${portValue}" != "${DEFAULT_RT_PORT}" ]] || continue + extraConnector=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']' ${filePath}/${fileName} 2>/dev/null) + setExtraConnector "${EXTRA_CONFIG_YAMLPATH}" "${extraConnector}" + done + else + extraConnector=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']' ${filePath}/${fileName} 2>/dev/null) + setExtraConnector "${EXTRA_CONFIG_YAMLPATH}" "${extraConnector}" + fi +} + +# Migrate connector configurations +migrateConnectorConfig () { + local i="$1" + local protocolType="$2" + local portValue="$3" + local connectorPortYamlPath="$4" + local connectorMaxThreadYamlPath="$5" + local connectorAttributesYamlPath="$6" + local filePath="$7" + local fileName="$8" + local connectorSendReasonPhraseYamlPath="$9" + local connectorRelaxedPathCharsYamlPath="${10}" + local connectorRelaxedQueryCharsYamlPath="${11}" + + # migrate port + setConnectorPort "${connectorPortYamlPath}" "${portValue}" + + # migrate maxThreads + local maxThreadValue=$(getXmlConnectorMaxThreads "$i" "${filePath}" "${fileName}") + setConnectorMaxThread "${connectorMaxThreadYamlPath}" "${maxThreadValue}" + + # migrate sendReasonPhrase + local sendReasonPhraseValue=$(getXmlConnectorSendReasonPhrase "$i" "${filePath}" "${fileName}") + setConnectorSendReasonPhrase "${connectorSendReasonPhraseYamlPath}" "${sendReasonPhraseValue}" + + # migrate relaxedPathChars + local relaxedPathCharsValue=$(getXmlConnectorRelaxedPathChars "$i" "${filePath}" "${fileName}") + setConnectorRelaxedPathChars "${connectorRelaxedPathCharsYamlPath}" "\"${relaxedPathCharsValue}\"" + # migrate relaxedQueryChars + local relaxedQueryCharsValue=$(getXmlConnectorRelaxedQueryChars "$i" "${filePath}" "${fileName}") + setConnectorRelaxedQueryChars "${connectorRelaxedQueryCharsYamlPath}" "\"${relaxedQueryCharsValue}\"" + + # migrate all attributes to extra config except port , maxThread , sendReasonPhrase ,relaxedPathChars and relaxedQueryChars + local connectorAttributes=$(getXmlConnectorAttributes "$i" "${filePath}" "${fileName}") + connectorAttributes=$(echo "${connectorAttributes}" | sed 's/port="'${portValue}'"//g' | sed 's/maxThreads="'${maxThreadValue}'"//g' | sed 's/sendReasonPhrase="'${sendReasonPhraseValue}'"//g' | sed 's/relaxedPathChars="\'${relaxedPathCharsValue}'\"//g' | sed 's/relaxedQueryChars="\'${relaxedQueryCharsValue}'\"//g') + # strip leading and trailing spaces + connectorAttributes=$(io_trim "${connectorAttributes}") + setConnectorExtraConfig "${connectorAttributesYamlPath}" "${connectorAttributes}" +} + +# Check for default port 8040 and 8081 in connectors and migrate +migrateConnectorPort () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local defaultPort="$4" + local connectorPortYamlPath="$5" + local connectorMaxThreadYamlPath="$6" + local connectorAttributesYamlPath="$7" + local connectorSendReasonPhraseYamlPath="$8" + local connectorRelaxedPathCharsYamlPath="$9" + local connectorRelaxedQueryCharsYamlPath="${10}" + local portYamlPath= + local maxThreadYamlPath= + local status= + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${protocolType}" == *AJP* ]] && continue + [[ "${portValue}" != "${defaultPort}" ]] && continue + if [[ "${portValue}" == "${DEFAULT_RT_PORT}" ]]; then + RT_DEFAULTPORT_STATUS=success + else + AC_DEFAULTPORT_STATUS=success + fi + migrateConnectorConfig "${i}" "${protocolType}" "${portValue}" "${connectorPortYamlPath}" "${connectorMaxThreadYamlPath}" "${connectorAttributesYamlPath}" "${filePath}" "${fileName}" "${connectorSendReasonPhraseYamlPath}" "${connectorRelaxedPathCharsYamlPath}" "${connectorRelaxedQueryCharsYamlPath}" + done +} + +# migrate to extra, connector having default port and protocol is AJP +migrateDefaultPortIfAjp () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local defaultPort="$4" + + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${protocolType}" != *AJP* ]] && continue + [[ "${portValue}" != "${defaultPort}" ]] && continue + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "no" "${i}" + done + +} + +# Comparing max threads in connectors +compareMaxThreads () { + local firstConnectorMaxThread="$1" + local firstConnectorNode="$2" + local secondConnectorMaxThread="$3" + local secondConnectorNode="$4" + local filePath="$5" + local fileName="$6" + + # choose higher maxThreads connector as Artifactory. + if [[ "${firstConnectorMaxThread}" -gt ${secondConnectorMaxThread} || "${firstConnectorMaxThread}" -eq ${secondConnectorMaxThread} ]]; then + # maxThread is higher in firstConnector, + # Taking firstConnector as Artifactory and SecondConnector as Access + # maxThread is equal in both connector,considering firstConnector as Artifactory and SecondConnector as Access + local rtPortValue=$(getXmlConnectorPort "${firstConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${firstConnectorNode}" "${protocolType}" "${rtPortValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + local acPortValue=$(getXmlConnectorPort "${secondConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${secondConnectorNode}" "${protocolType}" "${acPortValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + else + # maxThread is higher in SecondConnector, + # Taking SecondConnector as Artifactory and firstConnector as Access + local rtPortValue=$(getXmlConnectorPort "${secondConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${secondConnectorNode}" "${protocolType}" "${rtPortValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + local acPortValue=$(getXmlConnectorPort "${firstConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${firstConnectorNode}" "${protocolType}" "${acPortValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + fi +} + +# Check max threads exist to compare +maxThreadsExistToCompare () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local firstConnectorMaxThread= + local secondConnectorMaxThread= + local firstConnectorNode= + local secondConnectorNode= + local status=success + local firstnode=fail + + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + if [[ ${protocolType} == *AJP* ]]; then + # Migrate Connectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "no" "${i}" + continue + fi + # store maxthreads value of each connector + if [[ ${firstnode} == "fail" ]]; then + firstConnectorMaxThread=$(getXmlConnectorMaxThreads "${i}" "${filePath}" "${fileName}") + firstConnectorNode="${i}" + firstnode=success + else + secondConnectorMaxThread=$(getXmlConnectorMaxThreads "${i}" "${filePath}" "${fileName}") + secondConnectorNode="${i}" + fi + done + [[ -z "${firstConnectorMaxThread}" ]] && status=fail + [[ -z "${secondConnectorMaxThread}" ]] && status=fail + # maxThreads is set, now compare MaxThreads + if [[ "${status}" == "success" ]]; then + compareMaxThreads "${firstConnectorMaxThread}" "${firstConnectorNode}" "${secondConnectorMaxThread}" "${secondConnectorNode}" "${filePath}" "${fileName}" + else + # Assume first connector is RT, maxThreads is not set in both connectors + local rtPortValue=$(getXmlConnectorPort "${firstConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${firstConnectorNode}" "${protocolType}" "${rtPortValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + local acPortValue=$(getXmlConnectorPort "${secondConnectorNode}" "${filePath}" "${fileName}") + migrateConnectorConfig "${secondConnectorNode}" "${protocolType}" "${acPortValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + fi +} + +migrateExtraBasedOnNonAjpCount () { + local nonAjpCount="$1" + local filePath="$2" + local fileName="$3" + local connectorCount="$4" + local i="$5" + + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + if [[ "${protocolType}" == *AJP* ]]; then + if [[ "${nonAjpCount}" -eq 1 ]]; then + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "no" "${i}" + continue + else + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + continue + fi + fi +} + +# find RT and AC Connector +findRtAndAcConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local initialAjpCount=0 + local nonAjpCount=0 + + # get the count of non AJP + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${protocolType}" != *AJP* ]] || continue + nonAjpCount=$((initialAjpCount+1)) + initialAjpCount="${nonAjpCount}" + done + if [[ "${nonAjpCount}" -eq 1 ]]; then + # Add the connector found as access and artifactory connectors + # Mark port as 8040 for access + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + done + elif [[ "${nonAjpCount}" -eq 2 ]]; then + # compare maxThreads in both connectors + maxThreadsExistToCompare "${filePath}" "${fileName}" "${connectorCount}" + elif [[ "${nonAjpCount}" -gt 2 ]]; then + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + elif [[ "${nonAjpCount}" -eq 0 ]]; then + # setting with default port in system.yaml + setConnectorPort "${RT_PORT_YAMLPATH}" "${DEFAULT_RT_PORT}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + fi +} + +# get the count of non AJP +getCountOfNonAjp () { + local port="$1" + local connectorCount="$2" + local filePath=$3 + local fileName=$4 + local initialNonAjpCount=0 + + for ((i = 1 ; i <= "${connectorCount}" ; i++)); + do + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + local protocolType=$(getXmlConnectorProtocol "$i" "${filePath}" "${fileName}") + [[ "${portValue}" != "${port}" ]] || continue + [[ "${protocolType}" != *AJP* ]] || continue + local nonAjpCount=$((initialNonAjpCount+1)) + initialNonAjpCount="${nonAjpCount}" + done + echo -e "${nonAjpCount}" +} + +# Find for access connector +findAcConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + + # get the count of non AJP + local nonAjpCount=$(getCountOfNonAjp "${DEFAULT_RT_PORT}" "${connectorCount}" "${filePath}" "${fileName}") + if [[ "${nonAjpCount}" -eq 1 ]]; then + # Add the connector found as access connector and mark port as that of connector + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" != "${DEFAULT_RT_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + fi + done + elif [[ "${nonAjpCount}" -gt 1 ]]; then + # Take RT properties into access with 8040 + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" == "${DEFAULT_RT_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${AC_SENDREASONPHRASE_YAMLPATH}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + fi + done + elif [[ "${nonAjpCount}" -eq 0 ]]; then + # Add RT connector details as access connector and mark port as 8040 + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_RT_PORT}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${AC_SENDREASONPHRASE_YAMLPATH}" + setConnectorPort "${AC_PORT_YAMLPATH}" "${DEFAULT_ACCESS_PORT}" + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + fi +} + +# Find for artifactory connector +findRtConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + + # get the count of non AJP + local nonAjpCount=$(getCountOfNonAjp "${DEFAULT_ACCESS_PORT}" "${connectorCount}" "${filePath}" "${fileName}") + if [[ "${nonAjpCount}" -eq 1 ]]; then + # Add the connector found as RT connector + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" != "${DEFAULT_ACCESS_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + fi + done + elif [[ "${nonAjpCount}" -gt 1 ]]; then + # Take access properties into artifactory with 8081 + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + migrateExtraBasedOnNonAjpCount "${nonAjpCount}" "${filePath}" "${fileName}" "${connectorCount}" "$i" + local portValue=$(getXmlConnectorPort "$i" "${filePath}" "${fileName}") + if [[ "${portValue}" == "${DEFAULT_ACCESS_PORT}" ]]; then + migrateConnectorConfig "$i" "${protocolType}" "${portValue}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${filePath}" "${fileName}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + setConnectorPort "${RT_PORT_YAMLPATH}" "${DEFAULT_RT_PORT}" + fi + done + elif [[ "${nonAjpCount}" -eq 0 ]]; then + # Add access connector details as RT connector and mark as ${DEFAULT_RT_PORT} + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_ACCESS_PORT}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + setConnectorPort "${RT_PORT_YAMLPATH}" "${DEFAULT_RT_PORT}" + # migrateExtraConnectors + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + fi +} + +checkForTlsConnector () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + for ((i = 1 ; i <= "${connectorCount}" ; i++)) + do + local sslProtocolValue=$($LIBXML2_PATH --xpath '//Server/Service/Connector['$i']/@sslProtocol' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + if [[ "${sslProtocolValue}" == "TLS" ]]; then + bannerImportant "NOTE: Ignoring TLS connector during migration, modify the system yaml to enable TLS. Original server.xml is saved in path [${filePath}/${fileName}]" + TLS_CONNECTOR_EXISTS=${FLAG_Y} + continue + fi + done +} + +# set custom tomcat server Listeners to system.yaml +setListenerConnector () { + local filePath="$1" + local fileName="$2" + local listenerCount="$3" + for ((i = 1 ; i <= "${listenerCount}" ; i++)) + do + local listenerConnector=$($LIBXML2_PATH --xpath '//Server/Listener['$i']' ${filePath}/${fileName} 2>/dev/null) + local listenerClassName=$($LIBXML2_PATH --xpath '//Server/Listener['$i']/@className' ${filePath}/${fileName} 2>/dev/null | awk -F"=" '{print $2}' | tr -d '"') + if [[ "${listenerClassName}" == *Apr* ]]; then + setExtraConnector "${EXTRA_LISTENER_CONFIG_YAMLPATH}" "${listenerConnector}" + fi + done +} +# add custom tomcat server Listeners +addTomcatServerListeners () { + local filePath="$1" + local fileName="$2" + local listenerCount="$3" + if [[ "${listenerCount}" == "0" ]]; then + logger "No listener connectors found in the [${filePath}/${fileName}],skipping migration of listener connectors" + else + setListenerConnector "${filePath}" "${fileName}" "${listenerCount}" + setSystemValue "${RT_TOMCAT_HTTPSCONNECTOR_ENABLED}" "true" "${SYSTEM_YAML_PATH}" + logger "Setting [${RT_TOMCAT_HTTPSCONNECTOR_ENABLED}] with value [true] in system.yaml" + fi +} + +# server.xml migration operations +xmlMigrateOperation () { + local filePath="$1" + local fileName="$2" + local connectorCount="$3" + local listenerCount="$4" + RT_DEFAULTPORT_STATUS=fail + AC_DEFAULTPORT_STATUS=fail + TLS_CONNECTOR_EXISTS=${FLAG_N} + + # Check for connector with TLS , if found ignore migrating it + checkForTlsConnector "${filePath}" "${fileName}" "${connectorCount}" + if [[ "${TLS_CONNECTOR_EXISTS}" == "${FLAG_Y}" ]]; then + return + fi + addTomcatServerListeners "${filePath}" "${fileName}" "${listenerCount}" + # Migrate RT default port from connectors + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_RT_PORT}" "${RT_PORT_YAMLPATH}" "${RT_MAXTHREADS_YAMLPATH}" "${RT_EXTRACONFIG_YAMLPATH}" "${RT_SENDREASONPHRASE_YAMLPATH}" "${RT_RELAXEDPATHCHARS_YAMLPATH}" "${RT_RELAXEDQUERYCHARS_YAMLPATH}" + # Migrate to extra if RT default ports are AJP + migrateDefaultPortIfAjp "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_RT_PORT}" + # Migrate AC default port from connectors + migrateConnectorPort "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_ACCESS_PORT}" "${AC_PORT_YAMLPATH}" "${AC_MAXTHREADS_YAMLPATH}" "${AC_EXTRACONFIG_YAMLPATH}" "${AC_SENDREASONPHRASE_YAMLPATH}" + # Migrate to extra if access default ports are AJP + migrateDefaultPortIfAjp "${filePath}" "${fileName}" "${connectorCount}" "${DEFAULT_ACCESS_PORT}" + + if [[ "${AC_DEFAULTPORT_STATUS}" == "success" && "${RT_DEFAULTPORT_STATUS}" == "success" ]]; then + # RT and AC default port found + logger "Artifactory 8081 and Access 8040 default port are found" + migrateExtraConnectors "${filePath}" "${fileName}" "${connectorCount}" "yes" + elif [[ "${AC_DEFAULTPORT_STATUS}" == "success" && "${RT_DEFAULTPORT_STATUS}" == "fail" ]]; then + # Only AC default port found,find RT connector + logger "Found Access default 8040 port" + findRtConnector "${filePath}" "${fileName}" "${connectorCount}" + elif [[ "${AC_DEFAULTPORT_STATUS}" == "fail" && "${RT_DEFAULTPORT_STATUS}" == "success" ]]; then + # Only RT default port found,find AC connector + logger "Found Artifactory default 8081 port" + findAcConnector "${filePath}" "${fileName}" "${connectorCount}" + elif [[ "${AC_DEFAULTPORT_STATUS}" == "fail" && "${RT_DEFAULTPORT_STATUS}" == "fail" ]]; then + # RT and AC default port not found, find connector + logger "Artifactory 8081 and Access 8040 default port are not found" + findRtAndAcConnector "${filePath}" "${fileName}" "${connectorCount}" + fi +} + +# get count of connectors +getXmlConnectorCount () { + local filePath="$1" + local fileName="$2" + local count=$($LIBXML2_PATH --xpath 'count(/Server/Service/Connector)' ${filePath}/${fileName}) + echo -e "${count}" +} + +# get count of listener connectors +getTomcatServerListenersCount () { + local filePath="$1" + local fileName="$2" + local count=$($LIBXML2_PATH --xpath 'count(/Server/Listener)' ${filePath}/${fileName}) + echo -e "${count}" +} + +# Migrate server.xml configuration to system.yaml +migrateXmlFile () { + local xmlFiles= + local fileName= + local filePath= + local sourceFilePath= + DEFAULT_ACCESS_PORT="8040" + DEFAULT_RT_PORT="8081" + AC_PORT_YAMLPATH="migration.xmlFiles.serverXml.access.port" + AC_MAXTHREADS_YAMLPATH="migration.xmlFiles.serverXml.access.maxThreads" + AC_SENDREASONPHRASE_YAMLPATH="migration.xmlFiles.serverXml.access.sendReasonPhrase" + AC_EXTRACONFIG_YAMLPATH="migration.xmlFiles.serverXml.access.extraConfig" + RT_PORT_YAMLPATH="migration.xmlFiles.serverXml.artifactory.port" + RT_MAXTHREADS_YAMLPATH="migration.xmlFiles.serverXml.artifactory.maxThreads" + RT_SENDREASONPHRASE_YAMLPATH='migration.xmlFiles.serverXml.artifactory.sendReasonPhrase' + RT_RELAXEDPATHCHARS_YAMLPATH='migration.xmlFiles.serverXml.artifactory.relaxedPathChars' + RT_RELAXEDQUERYCHARS_YAMLPATH='migration.xmlFiles.serverXml.artifactory.relaxedQueryChars' + RT_EXTRACONFIG_YAMLPATH="migration.xmlFiles.serverXml.artifactory.extraConfig" + ROUTER_PORT_YAMLPATH="migration.xmlFiles.serverXml.router.port" + EXTRA_CONFIG_YAMLPATH="migration.xmlFiles.serverXml.extra.config" + EXTRA_LISTENER_CONFIG_YAMLPATH="migration.xmlFiles.serverXml.extra.listener" + RT_TOMCAT_HTTPSCONNECTOR_ENABLED="artifactory.tomcat.httpsConnector.enabled" + + retrieveYamlValue "migration.xmlFiles" "xmlFiles" "Skip" + xmlFiles="${YAML_VALUE}" + if [[ -z "${xmlFiles}" ]]; then + return + fi + bannerSection "PROCESSING MIGRATION OF XML FILES" + retrieveYamlValue "migration.xmlFiles.serverXml.fileName" "fileName" "Warning" + fileName="${YAML_VALUE}" + if [[ -z "${fileName}" ]]; then + return + fi + bannerSubSection "Processing Migration of $fileName" + retrieveYamlValue "migration.xmlFiles.serverXml.filePath" "filePath" "Warning" + filePath="${YAML_VALUE}" + if [[ -z "${filePath}" ]]; then + return + fi + # prepend NEW_DATA_DIR only if filePath is relative path + sourceFilePath=$(prependDir "${filePath}" "${NEW_DATA_DIR}/${filePath}") + if [[ "$(checkFileExists "${sourceFilePath}/${fileName}")" == "true" ]]; then + logger "File [${fileName}] is found in path [${sourceFilePath}]" + local connectorCount=$(getXmlConnectorCount "${sourceFilePath}" "${fileName}") + if [[ "${connectorCount}" == "0" ]]; then + logger "No connectors found in the [${filePath}/${fileName}],skipping migration of xml configuration" + return + fi + local listenerCount=$(getTomcatServerListenersCount "${sourceFilePath}" "${fileName}") + xmlMigrateOperation "${sourceFilePath}" "${fileName}" "${connectorCount}" "${listenerCount}" + else + logger "File [${fileName}] is not found in path [${sourceFilePath}] to migrate" + fi +} + +compareArtifactoryUser () { + local property="$1" + local oldPropertyValue="$2" + local newPropertyValue="$3" + local yamlPath="$4" + local sourceFile="$5" + + if [[ "${oldPropertyValue}" != "${newPropertyValue}" ]]; then + setSystemValue "${yamlPath}" "${oldPropertyValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value of the property [${property}] in system.yaml" + else + logger "No change in property [${property}] value in [${sourceFile}] to migrate" + fi +} + +migrateReplicator () { + local property="$1" + local oldPropertyValue="$2" + local yamlPath="$3" + + setSystemValue "${yamlPath}" "${oldPropertyValue}" "${SYSTEM_YAML_PATH}" + logger "Setting [${yamlPath}] with value of the property [${property}] in system.yaml" +} + +compareJavaOptions () { + local property="$1" + local oldPropertyValue="$2" + local newPropertyValue="$3" + local yamlPath="$4" + local sourceFile="$5" + local oldJavaOption= + local newJavaOption= + local extraJavaOption= + local check=false + local success=true + local status=true + + oldJavaOption=$(echo "${oldPropertyValue}" | awk 'BEGIN{FS=OFS="\""}{for(i=2;i.+)\.{{ include "artifactory.fullname" . }} {{ include "artifactory.fullname" . }} +{{ tpl (include "artifactory.nginx.hosts" .) . }}; + +if ($http_x_forwarded_proto = '') { + set $http_x_forwarded_proto $scheme; +} +set $host_port {{ .Values.nginx.https.externalPort }}; +if ( $scheme = "http" ) { + set $host_port {{ .Values.nginx.http.externalPort }}; +} +## Application specific logs +## access_log /var/log/nginx/artifactory-access.log timing; +## error_log /var/log/nginx/artifactory-error.log; +rewrite ^/artifactory/?$ / redirect; +if ( $repo != "" ) { + rewrite ^/(v1|v2)/(.*) /artifactory/api/docker/$repo/$1/$2 break; +} +chunked_transfer_encoding on; +client_max_body_size 0; + +location / { + proxy_read_timeout 900; + proxy_pass_header Server; + proxy_cookie_path ~*^/.* /; + proxy_pass {{ include "artifactory.scheme" . }}://{{ include "artifactory.fullname" . }}:{{ .Values.artifactory.externalPort }}/; + {{- if .Values.nginx.service.ssloffload}} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host; + {{- else }} + proxy_set_header X-JFrog-Override-Base-Url $http_x_forwarded_proto://$host:$host_port; + proxy_set_header X-Forwarded-Port $server_port; + {{- end }} + proxy_set_header X-Forwarded-Proto $http_x_forwarded_proto; + proxy_set_header Host $http_host; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + {{- if .Values.nginx.disableProxyBuffering}} + proxy_http_version 1.1; + proxy_request_buffering off; + proxy_buffering off; + {{- end }} + add_header Strict-Transport-Security "max-age=31536000; includeSubDomains" always; + location /artifactory/ { + if ( $request_uri ~ ^/artifactory/(.*)$ ) { + proxy_pass http://{{ include "artifactory.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/$1; + } + proxy_pass http://{{ include "artifactory.fullname" . }}:{{ .Values.artifactory.externalArtifactoryPort }}/artifactory/; + } + location /pipelines/ { + proxy_http_version 1.1; + proxy_set_header Upgrade $http_upgrade; + proxy_set_header Connection "upgrade"; + proxy_set_header Host $http_host; + {{- if .Values.router.tlsEnabled }} + proxy_pass https://{{ include "artifactory.fullname" . }}:{{ .Values.router.internalPort }}; + {{- else }} + proxy_pass http://{{ include "artifactory.fullname" . }}:{{ .Values.router.internalPort }}; + {{- end }} + } +} +} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/nginx-main-conf.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/nginx-main-conf.yaml new file mode 100644 index 0000000000..6ee7f98f9e --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/nginx-main-conf.yaml @@ -0,0 +1,83 @@ +# Main Nginx configuration file +worker_processes 4; + +{{- if .Values.nginx.logs.stderr }} +error_log stderr {{ .Values.nginx.logs.level }}; +{{- else -}} +error_log {{ .Values.nginx.persistence.mountPath }}/logs/error.log {{ .Values.nginx.logs.level }}; +{{- end }} +pid /var/run/nginx.pid; + +{{- if .Values.artifactory.ssh.enabled }} +## SSH Server Configuration +stream { + server { + {{- if .Values.nginx.singleStackIPv6Cluster }} + listen [::]:{{ .Values.nginx.ssh.internalPort }}; + {{- else -}} + listen {{ .Values.nginx.ssh.internalPort }}; + {{- end }} + proxy_pass {{ include "artifactory.fullname" . }}:{{ .Values.artifactory.ssh.externalPort }}; + } +} +{{- end }} + +events { + worker_connections 1024; +} + +http { + include /etc/nginx/mime.types; + default_type application/octet-stream; + + variables_hash_max_size 1024; + variables_hash_bucket_size 64; + server_names_hash_max_size 4096; + server_names_hash_bucket_size 128; + types_hash_max_size 2048; + types_hash_bucket_size 64; + proxy_read_timeout 2400s; + client_header_timeout 2400s; + client_body_timeout 2400s; + proxy_connect_timeout 75s; + proxy_send_timeout 2400s; + proxy_buffer_size 128k; + proxy_buffers 40 128k; + proxy_busy_buffers_size 128k; + proxy_temp_file_write_size 250m; + proxy_http_version 1.1; + client_body_buffer_size 128k; + + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + + log_format timing 'ip = $remote_addr ' + 'user = \"$remote_user\" ' + 'local_time = \"$time_local\" ' + 'host = $host ' + 'request = \"$request\" ' + 'status = $status ' + 'bytes = $body_bytes_sent ' + 'upstream = \"$upstream_addr\" ' + 'upstream_time = $upstream_response_time ' + 'request_time = $request_time ' + 'referer = \"$http_referer\" ' + 'UA = \"$http_user_agent\"'; + + {{- if .Values.nginx.logs.stdout }} + access_log /dev/stdout timing; + {{- else -}} + access_log {{ .Values.nginx.persistence.mountPath }}/logs/access.log timing; + {{- end }} + + sendfile on; + #tcp_nopush on; + + keepalive_timeout 65; + + #gzip on; + + include /etc/nginx/conf.d/*.conf; + +} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/system.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/system.yaml new file mode 100644 index 0000000000..053207fd01 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/files/system.yaml @@ -0,0 +1,156 @@ +router: + serviceRegistry: + insecure: {{ .Values.router.serviceRegistry.insecure }} +shared: +{{- if .Values.artifactory.coldStorage.enabled }} + jfrogColdStorage: + coldInstanceEnabled: true +{{- end }} +{{ tpl (include "artifactory.metrics" .) . }} + logging: + consoleLog: + enabled: {{ .Values.artifactory.consoleLog }} + extraJavaOpts: > + -Dartifactory.graceful.shutdown.max.request.duration.millis={{ mul .Values.artifactory.terminationGracePeriodSeconds 1000 }} + -Dartifactory.access.client.max.connections={{ .Values.access.tomcat.connector.maxThreads }} + {{- with .Values.artifactory.javaOpts }} + {{- if .corePoolSize }} + -Dartifactory.async.corePoolSize={{ .corePoolSize }} + {{- end }} + {{- if .xms }} + -Xms{{ .xms }} + {{- end }} + {{- if .xmx }} + -Xmx{{ .xmx }} + {{- end }} + {{- if .jmx.enabled }} + -Dcom.sun.management.jmxremote + -Dcom.sun.management.jmxremote.port={{ .jmx.port }} + -Dcom.sun.management.jmxremote.rmi.port={{ .jmx.port }} + -Dcom.sun.management.jmxremote.ssl={{ .jmx.ssl }} + {{- if .jmx.host }} + -Djava.rmi.server.hostname={{ tpl .jmx.host $ }} + {{- else }} + -Djava.rmi.server.hostname={{ template "artifactory.fullname" $ }} + {{- end }} + {{- if .jmx.authenticate }} + -Dcom.sun.management.jmxremote.authenticate=true + -Dcom.sun.management.jmxremote.access.file={{ .jmx.accessFile }} + -Dcom.sun.management.jmxremote.password.file={{ .jmx.passwordFile }} + {{- else }} + -Dcom.sun.management.jmxremote.authenticate=false + {{- end }} + {{- end }} + {{- if .other }} + {{ .other }} + {{- end }} + {{- end }} + {{- if or .Values.database.type .Values.postgresql.enabled }} + database: + allowNonPostgresql: {{ .Values.database.allowNonPostgresql }} + {{- if .Values.postgresql.enabled }} + type: postgresql + url: "jdbc:postgresql://{{ .Release.Name }}-postgresql:{{ .Values.postgresql.service.port }}/{{ .Values.postgresql.postgresqlDatabase }}" + driver: org.postgresql.Driver + username: "{{ .Values.postgresql.postgresqlUsername }}" + {{- else }} + type: "{{ .Values.database.type }}" + driver: "{{ .Values.database.driver }}" + {{- end }} + {{- end }} +artifactory: +{{- if or .Values.artifactory.haDataDir.enabled .Values.artifactory.haBackupDir.enabled }} + node: + {{- if .Values.artifactory.haDataDir.path }} + haDataDir: {{ .Values.artifactory.haDataDir.path }} + {{- end }} + {{- if .Values.artifactory.haBackupDir.path }} + haBackupDir: {{ .Values.artifactory.haBackupDir.path }} + {{- end }} +{{- end }} + database: + maxOpenConnections: {{ .Values.artifactory.database.maxOpenConnections }} + tomcat: + maintenanceConnector: + port: {{ .Values.artifactory.tomcat.maintenanceConnector.port }} + connector: + maxThreads: {{ .Values.artifactory.tomcat.connector.maxThreads }} + sendReasonPhrase: {{ .Values.artifactory.tomcat.connector.sendReasonPhrase }} + extraConfig: {{ .Values.artifactory.tomcat.connector.extraConfig }} +frontend: + session: + timeMinutes: {{ .Values.frontend.session.timeoutMinutes | quote }} +access: + runOnArtifactoryTomcat: {{ .Values.access.runOnArtifactoryTomcat | default false }} + database: + maxOpenConnections: {{ .Values.access.database.maxOpenConnections }} + {{- if not (.Values.access.runOnArtifactoryTomcat | default false) }} + extraJavaOpts: > + {{- if .Values.splitServicesToContainers }} + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=70 + {{- end }} + {{- with .Values.access.javaOpts }} + {{- if .other }} + {{ .other }} + {{- end }} + {{- end }} + {{- end }} + tomcat: + connector: + maxThreads: {{ .Values.access.tomcat.connector.maxThreads }} + sendReasonPhrase: {{ .Values.access.tomcat.connector.sendReasonPhrase }} + extraConfig: {{ .Values.access.tomcat.connector.extraConfig }} +{{- if .Values.mc.enabled }} +mc: + enabled: true + database: + maxOpenConnections: {{ .Values.mc.database.maxOpenConnections }} + idgenerator: + maxOpenConnections: {{ .Values.mc.idgenerator.maxOpenConnections }} + tomcat: + connector: + maxThreads: {{ .Values.mc.tomcat.connector.maxThreads }} + sendReasonPhrase: {{ .Values.mc.tomcat.connector.sendReasonPhrase }} + extraConfig: {{ .Values.mc.tomcat.connector.extraConfig }} +{{- end }} +metadata: + database: + maxOpenConnections: {{ .Values.metadata.database.maxOpenConnections }} +{{- if and .Values.jfconnect.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" .Values.artifactory.image.repository)) }} +jfconnect: + enabled: true +{{- else }} +jfconnect: + enabled: false +jfconnect_service: + enabled: false +{{- end }} +{{- if and .Values.federation.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" .Values.artifactory.image.repository)) }} +federation: + enabled: true + embedded: {{ .Values.federation.embedded }} + extraJavaOpts: {{ .Values.federation.extraJavaOpts }} + port: {{ .Values.federation.internalPort }} +rtfs: + database: + driver: org.postgresql.Driver + type: postgresql + username: {{ .Values.federation.database.username }} + password: {{ .Values.federation.database.password }} + url: jdbc:postgresql://{{ .Values.federation.database.host }}:{{ .Values.federation.database.port }}/{{ .Values.federation.database.name }} +{{- else }} +federation: + enabled: false +{{- end }} +{{- if .Values.event.webhooks }} +event: + webhooks: {{ toYaml .Values.event.webhooks | nindent 6 }} +{{- end }} +{{- if .Values.evidence.enabled }} +evidence: + enabled: true +{{- else }} +evidence: + enabled: false +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/logo/artifactory-logo.png b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/logo/artifactory-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..fe6c23c5a7f87edaf49f883ffa99d875073e2285 GIT binary patch literal 82419 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 z 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 GZv9`RA2Y%L literal 0 HcmV?d00001 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge-extra-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge-extra-config.yaml new file mode 100644 index 0000000000..7bccf330d9 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge-extra-config.yaml @@ -0,0 +1,41 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=200 + -Dartifactory.async.poolMaxQueueSize=100000 + -Dartifactory.http.client.max.total.connections=150 + -Dartifactory.http.client.max.connections.per.route=150 + -Dartifactory.access.client.max.connections=200 + -Dartifactory.metadata.event.operator.threads=5 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=1048576 + -XX:MaxDirectMemorySize=1024m + tomcat: + connector: + maxThreads: 800 + extraConfig: 'acceptCount="1200" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 200 + +access: + tomcat: + connector: + maxThreads: 200 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 200 + +metadata: + database: + maxOpenConnections: 200 + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge.yaml new file mode 100644 index 0000000000..be477939b4 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-2xlarge.yaml @@ -0,0 +1,126 @@ +############################################################## +# The 2xlarge sizing +# This size is intended for very large organizations. It can be increased with adding replicas +############################################################## +splitServicesToContainers: true +artifactory: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 6 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "4" + memory: 20Gi + limits: + # cpu: "20" + memory: 24Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "16" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 1 + memory: 2Gi + limits: + # cpu: 2 + memory: 4Gi + +router: + resources: + requests: + cpu: "1" + memory: 1Gi + limits: + # cpu: "6" + memory: 2Gi + +frontend: + resources: + requests: + cpu: "1" + memory: 500Mi + limits: + # cpu: "5" + memory: 1Gi + +metadata: + resources: + requests: + cpu: "1" + memory: 500Mi + limits: + # cpu: "5" + memory: 2Gi + +event: + resources: + requests: + cpu: 200m + memory: 100Mi + limits: + # cpu: "1" + memory: 500Mi + +observability: + resources: + requests: + cpu: 200m + memory: 100Mi + limits: + # cpu: "1" + memory: 500Mi + +jfconnect: + resources: + requests: + cpu: 100m + memory: 100Mi + limits: + # cpu: "1" + memory: 250Mi + +nginx: + replicaCount: 3 + disableProxyBuffering: true + resources: + requests: + cpu: "4" + memory: "6Gi" + limits: + # cpu: "14" + memory: "8Gi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "5000" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 256Gi + cpu: "64" + limits: + memory: 256Gi + # cpu: "128" \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large-extra-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large-extra-config.yaml new file mode 100644 index 0000000000..d97a85c9fd --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large-extra-config.yaml @@ -0,0 +1,41 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=65 + -Dartifactory.async.corePoolSize=80 + -Dartifactory.async.poolMaxQueueSize=20000 + -Dartifactory.http.client.max.total.connections=100 + -Dartifactory.http.client.max.connections.per.route=100 + -Dartifactory.access.client.max.connections=125 + -Dartifactory.metadata.event.operator.threads=4 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=524288 + -XX:MaxDirectMemorySize=512m + tomcat: + connector: + maxThreads: 500 + extraConfig: 'acceptCount="800" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 100 + +access: + tomcat: + connector: + maxThreads: 125 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 100 + +metadata: + database: + maxOpenConnections: 100 + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large.yaml new file mode 100644 index 0000000000..80326a8e43 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-large.yaml @@ -0,0 +1,126 @@ +############################################################## +# The large sizing +# This size is intended for large organizations. It can be increased with adding replicas or moving to the xlarge sizing +############################################################## +splitServicesToContainers: true +artifactory: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 3 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "2" + memory: 10Gi + limits: + # cpu: "14" + memory: 12Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "8" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 1 + memory: 1.5Gi + limits: + # cpu: 1 + memory: 2Gi + +router: + resources: + requests: + cpu: 200m + memory: 400Mi + limits: + # cpu: "4" + memory: 1Gi + +frontend: + resources: + requests: + cpu: 200m + memory: 300Mi + limits: + # cpu: "3" + memory: 1Gi + +metadata: + resources: + requests: + cpu: 200m + memory: 200Mi + limits: + # cpu: "4" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 2 + disableProxyBuffering: true + resources: + requests: + cpu: "1" + memory: "500Mi" + limits: + # cpu: "4" + memory: "1Gi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "600" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 64Gi + cpu: "16" + limits: + memory: 64Gi + # cpu: "32" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium-extra-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium-extra-config.yaml new file mode 100644 index 0000000000..1c294c0439 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium-extra-config.yaml @@ -0,0 +1,41 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=40 + -Dartifactory.async.poolMaxQueueSize=10000 + -Dartifactory.http.client.max.total.connections=50 + -Dartifactory.http.client.max.connections.per.route=50 + -Dartifactory.access.client.max.connections=75 + -Dartifactory.metadata.event.operator.threads=3 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=262144 + -XX:MaxDirectMemorySize=256m + tomcat: + connector: + maxThreads: 300 + extraConfig: 'acceptCount="600" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 50 + +access: + tomcat: + connector: + maxThreads: 75 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 50 + +metadata: + database: + maxOpenConnections: 50 + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium.yaml new file mode 100644 index 0000000000..8b72150419 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-medium.yaml @@ -0,0 +1,126 @@ +############################################################## +# The medium sizing +# This size is just 2 replicas of the small size. Vertical sizing of all services is not changed +############################################################## +splitServicesToContainers: true +artifactory: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 2 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "1" + memory: 4Gi + limits: + # cpu: "10" + memory: 5Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "2" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 500m + memory: 1.5Gi + limits: + # cpu: 1 + memory: 2Gi + +router: + resources: + requests: + cpu: 100m + memory: 250Mi + limits: + # cpu: "1" + memory: 500Mi + +frontend: + resources: + requests: + cpu: 100m + memory: 150Mi + limits: + # cpu: "2" + memory: 250Mi + +metadata: + resources: + requests: + cpu: 100m + memory: 200Mi + limits: + # cpu: "2" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 2 + disableProxyBuffering: true + resources: + requests: + cpu: "100m" + memory: "100Mi" + limits: + # cpu: "2" + memory: "500Mi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "200" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 32Gi + cpu: "8" + limits: + memory: 32Gi + # cpu: "16" \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small-extra-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small-extra-config.yaml new file mode 100644 index 0000000000..1c294c0439 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small-extra-config.yaml @@ -0,0 +1,41 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=40 + -Dartifactory.async.poolMaxQueueSize=10000 + -Dartifactory.http.client.max.total.connections=50 + -Dartifactory.http.client.max.connections.per.route=50 + -Dartifactory.access.client.max.connections=75 + -Dartifactory.metadata.event.operator.threads=3 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=262144 + -XX:MaxDirectMemorySize=256m + tomcat: + connector: + maxThreads: 300 + extraConfig: 'acceptCount="600" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 50 + +access: + tomcat: + connector: + maxThreads: 75 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 50 + +metadata: + database: + maxOpenConnections: 50 + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small.yaml new file mode 100644 index 0000000000..eb8d7239d8 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-small.yaml @@ -0,0 +1,124 @@ +############################################################## +# The small sizing +# This is the size recommended for running Artifactory for small teams +############################################################## +splitServicesToContainers: true +artifactory: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 1 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "1" + memory: 4Gi + limits: + # cpu: "10" + memory: 5Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "2" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 500m + memory: 1.5Gi + limits: + # cpu: 1 + memory: 2Gi + +router: + resources: + requests: + cpu: 100m + memory: 250Mi + limits: + # cpu: "1" + memory: 500Mi + +frontend: + resources: + requests: + cpu: 100m + memory: 150Mi + limits: + # cpu: "2" + memory: 250Mi + +metadata: + resources: + requests: + cpu: 100m + memory: 200Mi + limits: + # cpu: "2" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 1 + disableProxyBuffering: true + resources: + requests: + cpu: "100m" + memory: "100Mi" + limits: + # cpu: "2" + memory: "500Mi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "100" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 16Gi + cpu: "4" + limits: + memory: 16Gi + # cpu: "10" \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge-extra-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge-extra-config.yaml new file mode 100644 index 0000000000..00e6099f20 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge-extra-config.yaml @@ -0,0 +1,41 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=65 + -Dartifactory.async.corePoolSize=160 + -Dartifactory.async.poolMaxQueueSize=50000 + -Dartifactory.http.client.max.total.connections=150 + -Dartifactory.http.client.max.connections.per.route=150 + -Dartifactory.access.client.max.connections=150 + -Dartifactory.metadata.event.operator.threads=5 + -XX:MaxMetaspaceSize=512m + -Djdk.nio.maxCachedBufferSize=1048576 + -XX:MaxDirectMemorySize=1024m + tomcat: + connector: + maxThreads: 600 + extraConfig: 'acceptCount="1200" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 150 + +access: + tomcat: + connector: + maxThreads: 150 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 150 + +metadata: + database: + maxOpenConnections: 150 + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge.yaml new file mode 100644 index 0000000000..e77152ee1e --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xlarge.yaml @@ -0,0 +1,126 @@ +############################################################## +# The xlarge sizing +# This size is intended for very large organizations. It can be increased with adding replicas +############################################################## +splitServicesToContainers: true +artifactory: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 4 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "2" + memory: 14Gi + limits: + # cpu: "14" + memory: 16Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "16" + - name : JF_SHARED_NODE_HAENABLED + value: "true" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 500m + memory: 2Gi + limits: + # cpu: 1 + memory: 3Gi + +router: + resources: + requests: + cpu: 200m + memory: 500Mi + limits: + # cpu: "4" + memory: 1Gi + +frontend: + resources: + requests: + cpu: 200m + memory: 300Mi + limits: + # cpu: "3" + memory: 1Gi + +metadata: + resources: + requests: + cpu: 200m + memory: 200Mi + limits: + # cpu: "4" + memory: 1Gi + +event: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 100m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 2 + disableProxyBuffering: true + resources: + requests: + cpu: "4" + memory: "4Gi" + limits: + # cpu: "12" + memory: "8Gi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "2000" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 128Gi + cpu: "32" + limits: + memory: 128Gi + # cpu: "64" \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall-extra-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall-extra-config.yaml new file mode 100644 index 0000000000..39709b691c --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall-extra-config.yaml @@ -0,0 +1,42 @@ +#################################################################################### +# [WARNING] The configuration mentioned in this file are taken inside system.yaml +# hence this configuration will be overridden when enabling systemYamlOverride +#################################################################################### +artifactory: + javaOpts: + other: > + -XX:InitialRAMPercentage=40 + -XX:MaxRAMPercentage=70 + -Dartifactory.async.corePoolSize=10 + -Dartifactory.async.poolMaxQueueSize=2000 + -Dartifactory.http.client.max.total.connections=20 + -Dartifactory.http.client.max.connections.per.route=20 + -Dartifactory.access.client.max.connections=15 + -Dartifactory.metadata.event.operator.threads=2 + -XX:MaxMetaspaceSize=400m + -XX:CompressedClassSpaceSize=96m + -Djdk.nio.maxCachedBufferSize=131072 + -XX:MaxDirectMemorySize=128m + tomcat: + connector: + maxThreads: 50 + extraConfig: 'acceptCount="200" acceptorThreadCount="2" compression="off" connectionLinger="-1" connectionTimeout="120000" enableLookups="false"' + + database: + maxOpenConnections: 15 + +access: + tomcat: + connector: + maxThreads: 15 + javaOpts: + other: > + -XX:InitialRAMPercentage=20 + -XX:MaxRAMPercentage=60 + database: + maxOpenConnections: 15 + +metadata: + database: + maxOpenConnections: 15 + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall.yaml new file mode 100644 index 0000000000..246f830a01 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/sizing/artifactory-xsmall.yaml @@ -0,0 +1,125 @@ +############################################################## +# The xsmall sizing +# This is the minimum size recommended for running Artifactory +############################################################## +splitServicesToContainers: true +artifactory: + # Enterprise and above licenses are required for setting replicaCount greater than 1. + # Count should be equal or above the total number of licenses available for artifactory. + replicaCount: 1 + + # Require multiple Artifactory pods to run on separate nodes + podAntiAffinity: + type: "hard" + + resources: + requests: + cpu: "1" + memory: 3Gi + limits: + # cpu: "10" + memory: 4Gi + + extraEnvironmentVariables: + - name: MALLOC_ARENA_MAX + value: "2" + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + +access: + resources: + requests: + cpu: 500m + memory: 1.5Gi + limits: + # cpu: 1 + memory: 2Gi + +router: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: "1" + memory: 500Mi + +frontend: + resources: + requests: + cpu: 50m + memory: 150Mi + limits: + # cpu: "2" + memory: 250Mi + +metadata: + resources: + requests: + cpu: 50m + memory: 100Mi + limits: + # cpu: "2" + memory: 1Gi + +event: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +observability: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +jfconnect: + resources: + requests: + cpu: 50m + memory: 50Mi + limits: + # cpu: 500m + memory: 250Mi + +nginx: + replicaCount: 1 + disableProxyBuffering: true + resources: + requests: + cpu: "50m" + memory: "50Mi" + limits: + # cpu: "1" + memory: "250Mi" + +postgresql: + postgresqlExtendedConf: + maxConnections: "50" + primary: + affinity: + # Require PostgreSQL pod to run on a different node than Artifactory pods + podAntiAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + - labelSelector: + matchExpressions: + - key: app + operator: In + values: + - artifactory + topologyKey: kubernetes.io/hostname + resources: + requests: + memory: 8Gi + cpu: "2" + limits: + memory: 8Gi + # cpu: "8" + diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/NOTES.txt b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/NOTES.txt new file mode 100644 index 0000000000..76652ac98d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/NOTES.txt @@ -0,0 +1,106 @@ +Congratulations. You have just deployed JFrog Artifactory! +{{- if .Values.artifactory.masterKey }} +{{- if and (not .Values.artifactory.masterKeySecretName) (eq .Values.artifactory.masterKey "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF") }} + + +***************************************** WARNING ****************************************** +* Your Artifactory master key is still set to the provided example: * +* artifactory.masterKey=FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF * +* * +* You should change this to your own generated key: * +* $ export MASTER_KEY=$(openssl rand -hex 32) * +* $ echo ${MASTER_KEY} * +* * +* Pass the created master key to helm with '--set artifactory.masterKey=${MASTER_KEY}' * +* * +* Alternatively, you can use a pre-existing secret with a key called master-key with * +* '--set artifactory.masterKeySecretName=${SECRET_NAME}' * +******************************************************************************************** +{{- end }} +{{- end }} + +{{- if .Values.artifactory.joinKey }} +{{- if eq .Values.artifactory.joinKey "EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE" }} + + +***************************************** WARNING ****************************************** +* Your Artifactory join key is still set to the provided example: * +* artifactory.joinKey=EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE * +* * +* You should change this to your own generated key: * +* $ export JOIN_KEY=$(openssl rand -hex 32) * +* $ echo ${JOIN_KEY} * +* * +* Pass the created master key to helm with '--set artifactory.joinKey=${JOIN_KEY}' * +* * +******************************************************************************************** +{{- end }} +{{- end }} + +{{- if .Values.artifactory.setSecurityContext }} +****************************************** WARNING ********************************************** +* From chart version 107.84.x, `setSecurityContext` has been renamed to `podSecurityContext`, * + please change your values.yaml before upgrade , For more Info , refer to 107.84.x changelog * +************************************************************************************************* +{{- end }} + +{{- if and (or (or (or (or (or ( or ( or ( or (or (or ( or (or .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName) .Values.systemYamlOverride.existingSecret) (or .Values.artifactory.customCertificates.enabled .Values.global.customCertificates.enabled)) .Values.aws.licenseConfigSecretName) .Values.artifactory.persistence.customBinarystoreXmlSecret) .Values.access.customCertificatesSecretName) .Values.systemYamlOverride.existingSecret) .Values.artifactory.license.secret) .Values.artifactory.userPluginSecrets) (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey)) (and .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName)) (or .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName)) .Values.artifactory.unifiedSecretInstallation }} +****************************************** WARNING ************************************************************************************************** +* The unifiedSecretInstallation flag is currently enabled, which creates the unified secret. The existing secrets will continue as separate secrets.* +* Update the values.yaml with the existing secrets to add them to the unified secret. * +***************************************************************************************************************************************************** +{{- end }} + +1. Get the Artifactory URL by running these commands: + + {{- if .Values.ingress.enabled }} + {{- range .Values.ingress.hosts }} + http://{{ . }} + {{- end }} + + {{- else if contains "NodePort" .Values.nginx.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "artifactory.nginx.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.nginx.service.type }} + + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of the service by running 'kubectl get svc --namespace {{ .Release.Namespace }} -w {{ template "artifactory.nginx.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "artifactory.nginx.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + echo http://$SERVICE_IP/ + + {{- else if contains "ClusterIP" .Values.nginx.service.type }} + export POD_NAME=$(kubectl get pods --namespace {{ .Release.Namespace }} -l "component={{ .Values.nginx.name }}" -o jsonpath="{.items[0].metadata.name}") + echo http://127.0.0.1:{{ .Values.nginx.externalPortHttp }} + kubectl port-forward --namespace {{ .Release.Namespace }} $POD_NAME {{ .Values.nginx.externalPortHttp }}:{{ .Values.nginx.internalPortHttp }} + + {{- end }} + +2. Open Artifactory in your browser + Default credential for Artifactory: + user: admin + password: password + +{{ if .Values.artifactory.javaOpts.jmx.enabled }} +JMX configuration: +{{- if not (contains "LoadBalancer" .Values.artifactory.service.type) }} +If you want to access JMX from you computer with jconsole, you should set ".Values.artifactory.service.type=LoadBalancer" !!! +{{ end }} + +1. Get the Artifactory service IP: +export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "artifactory.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}') + +2. Map the service name to the service IP in /etc/hosts: +sudo sh -c "echo \"${SERVICE_IP} {{ template "artifactory.fullname" . }}\" >> /etc/hosts" + +3. Launch jconsole: +jconsole {{ template "artifactory.fullname" . }}:{{ .Values.artifactory.javaOpts.jmx.port }} +{{- end }} + +{{- if and .Values.nginx.enabled .Values.ingress.hosts }} +***************************************** WARNING ***************************************************************************** +* when nginx is enabled , .Values.ingress.hosts will be deprecated in upcoming releases * +* It is recommended to use nginx.hosts instead ingress.hosts +******************************************************************************************************************************* +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_helpers.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_helpers.tpl new file mode 100644 index 0000000000..7cea041f7e --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_helpers.tpl @@ -0,0 +1,528 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "artifactory.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Expand the name nginx service. +*/}} +{{- define "artifactory.nginx.name" -}} +{{- default .Chart.Name .Values.nginx.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 "artifactory.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 default fully qualified nginx name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "artifactory.nginx.fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- printf "%s-%s-%s" .Release.Name $name .Values.nginx.name | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create the name of the service account to use +*/}} +{{- define "artifactory.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} +{{ default (include "artifactory.fullname" .) .Values.serviceAccount.name }} +{{- else -}} +{{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "artifactory.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Generate SSL certificates +*/}} +{{- define "artifactory.gen-certs" -}} +{{- $altNames := list ( printf "%s.%s" (include "artifactory.fullname" .) .Release.Namespace ) ( printf "%s.%s.svc" (include "artifactory.fullname" .) .Release.Namespace ) -}} +{{- $ca := genCA "artifactory-ca" 365 -}} +{{- $cert := genSignedCert ( include "artifactory.fullname" . ) nil $altNames 365 $ca -}} +tls.crt: {{ $cert.Cert | b64enc }} +tls.key: {{ $cert.Key | b64enc }} +{{- end -}} + +{{/* +Scheme (http/https) based on Access or Router TLS enabled/disabled +*/}} +{{- define "artifactory.scheme" -}} +{{- if or .Values.access.accessConfig.security.tls .Values.router.tlsEnabled -}} +{{- printf "%s" "https" -}} +{{- else -}} +{{- printf "%s" "http" -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve joinKey value +*/}} +{{- define "artifactory.joinKey" -}} +{{- if .Values.global.joinKey -}} +{{- .Values.global.joinKey -}} +{{- else if .Values.artifactory.joinKey -}} +{{- .Values.artifactory.joinKey -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve jfConnectToken value +*/}} +{{- define "artifactory.jfConnectToken" -}} +{{- .Values.artifactory.jfConnectToken -}} +{{- end -}} + +{{/* +Resolve masterKey value +*/}} +{{- define "artifactory.masterKey" -}} +{{- if .Values.global.masterKey -}} +{{- .Values.global.masterKey -}} +{{- else if .Values.artifactory.masterKey -}} +{{- .Values.artifactory.masterKey -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve joinKeySecretName value +*/}} +{{- define "artifactory.joinKeySecretName" -}} +{{- if .Values.global.joinKeySecretName -}} +{{- .Values.global.joinKeySecretName -}} +{{- else if .Values.artifactory.joinKeySecretName -}} +{{- .Values.artifactory.joinKeySecretName -}} +{{- else -}} +{{ include "artifactory.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Resolve jfConnectTokenSecretName value +*/}} +{{- define "artifactory.jfConnectTokenSecretName" -}} +{{- if .Values.artifactory.jfConnectTokenSecretName -}} +{{- .Values.artifactory.jfConnectTokenSecretName -}} +{{- else -}} +{{ include "artifactory.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Resolve masterKeySecretName value +*/}} +{{- define "artifactory.masterKeySecretName" -}} +{{- if .Values.global.masterKeySecretName -}} +{{- .Values.global.masterKeySecretName -}} +{{- else if .Values.artifactory.masterKeySecretName -}} +{{- .Values.artifactory.masterKeySecretName -}} +{{- else -}} +{{ include "artifactory.fullname" . }} +{{- end -}} +{{- end -}} + +{{/* +Resolve imagePullSecrets value +*/}} +{{- define "artifactory.imagePullSecrets" -}} +{{- if .Values.global.imagePullSecrets }} +imagePullSecrets: +{{- range .Values.global.imagePullSecrets }} + - name: {{ . }} +{{- end }} +{{- else if .Values.imagePullSecrets }} +imagePullSecrets: +{{- range .Values.imagePullSecrets }} + - name: {{ . }} +{{- end }} +{{- end -}} +{{- end -}} + +{{/* +Resolve customInitContainersBegin value +*/}} +{{- define "artifactory.customInitContainersBegin" -}} +{{- if .Values.global.customInitContainersBegin -}} +{{- .Values.global.customInitContainersBegin -}} +{{- end -}} +{{- if .Values.artifactory.customInitContainersBegin -}} +{{- .Values.artifactory.customInitContainersBegin -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customInitContainers value +*/}} +{{- define "artifactory.customInitContainers" -}} +{{- if .Values.global.customInitContainers -}} +{{- .Values.global.customInitContainers -}} +{{- end -}} +{{- if .Values.artifactory.customInitContainers -}} +{{- .Values.artifactory.customInitContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumes value +*/}} +{{- define "artifactory.customVolumes" -}} +{{- if .Values.global.customVolumes -}} +{{- .Values.global.customVolumes -}} +{{- end -}} +{{- if .Values.artifactory.customVolumes -}} +{{- .Values.artifactory.customVolumes -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumeMounts value +*/}} +{{- define "artifactory.customVolumeMounts" -}} +{{- if .Values.global.customVolumeMounts -}} +{{- .Values.global.customVolumeMounts -}} +{{- end -}} +{{- if .Values.artifactory.customVolumeMounts -}} +{{- .Values.artifactory.customVolumeMounts -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customSidecarContainers value +*/}} +{{- define "artifactory.customSidecarContainers" -}} +{{- if .Values.global.customSidecarContainers -}} +{{- .Values.global.customSidecarContainers -}} +{{- end -}} +{{- if .Values.artifactory.customSidecarContainers -}} +{{- .Values.artifactory.customSidecarContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper artifactory chart image names +*/}} +{{- define "artifactory.getImageInfoByValue" -}} +{{- $dot := index . 0 }} +{{- $indexReference := index . 1 }} +{{- $registryName := index $dot.Values $indexReference "image" "registry" -}} +{{- $repositoryName := index $dot.Values $indexReference "image" "repository" -}} +{{- $tag := default $dot.Chart.AppVersion (index $dot.Values $indexReference "image" "tag") | toString -}} +{{- if $dot.Values.global }} + {{- if and $dot.Values.splitServicesToContainers $dot.Values.global.versions.router (eq $indexReference "router") }} + {{- $tag = $dot.Values.global.versions.router | toString -}} + {{- end -}} + {{- if and $dot.Values.global.versions.initContainers (eq $indexReference "initContainers") }} + {{- $tag = $dot.Values.global.versions.initContainers | toString -}} + {{- end -}} + {{- if and $dot.Values.global.versions.artifactory (or (eq $indexReference "artifactory") (eq $indexReference "nginx") ) }} + {{- $tag = $dot.Values.global.versions.artifactory | toString -}} + {{- end -}} + {{- if $dot.Values.global.imageRegistry }} + {{- printf "%s/%s:%s" $dot.Values.global.imageRegistry $repositoryName $tag -}} + {{- else -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} + {{- end -}} +{{- else -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag -}} +{{- end -}} +{{- end -}} + +{{/* +Return the proper artifactory app version +*/}} +{{- define "artifactory.app.version" -}} +{{- $tag := (splitList ":" ((include "artifactory.getImageInfoByValue" (list . "artifactory" )))) | last | toString -}} +{{- printf "%s" $tag -}} +{{- end -}} + +{{/* +Custom certificate copy command +*/}} +{{- define "artifactory.copyCustomCerts" -}} +echo "Copy custom certificates to {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted"; +mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted; +for file in $(ls -1 /tmp/certs/* | grep -v .key | grep -v ":" | grep -v grep); do if [ -f "${file}" ]; then cp -v ${file} {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted; fi done; +if [ -f {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted/tls.crt ]; then mv -v {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted/tls.crt {{ .Values.artifactory.persistence.mountPath }}/etc/security/keys/trusted/ca.crt; fi; +{{- end -}} + +{{/* +Circle of trust certificates copy command +*/}} +{{- define "artifactory.copyCircleOfTrustCertsCerts" -}} +echo "Copy circle of trust certificates to {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted"; +mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; +for file in $(ls -1 /tmp/circleoftrustcerts/* | grep -v .key | grep -v ":" | grep -v grep); do if [ -f "${file}" ]; then cp -v ${file} {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; fi done; +{{- end -}} + +{{/* +Resolve requiredServiceTypes value +*/}} +{{- define "artifactory.router.requiredServiceTypes" -}} +{{- $requiredTypes := "jfrt,jfac" -}} +{{- if not .Values.access.enabled -}} + {{- $requiredTypes = "jfrt" -}} +{{- end -}} +{{- if .Values.observability.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfob" -}} +{{- end -}} +{{- if .Values.metadata.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfmd" -}} +{{- end -}} +{{- if .Values.event.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfevt" -}} +{{- end -}} +{{- if .Values.frontend.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jffe" -}} +{{- end -}} +{{- if .Values.jfconnect.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfcon" -}} +{{- end -}} +{{- if .Values.evidence.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfevd" -}} +{{- end -}} +{{- if .Values.mc.enabled -}} + {{- $requiredTypes = printf "%s,%s" $requiredTypes "jfmc" -}} +{{- end -}} +{{- $requiredTypes -}} +{{- end -}} + +{{/* +Check if the image is artifactory pro or not +*/}} +{{- define "artifactory.isImageProType" -}} +{{- if not (regexMatch "^.*(oss|cpp-ce|jcr).*$" .Values.artifactory.image.repository) -}} +{{ true }} +{{- else -}} +{{ false }} +{{- end -}} +{{- end -}} + +{{/* +Check if the artifactory is using derby database +*/}} +{{- define "artifactory.isUsingDerby" -}} +{{- if and (eq (default "derby" .Values.database.type) "derby") (not .Values.postgresql.enabled) -}} +{{ true }} +{{- else -}} +{{ false }} +{{- end -}} +{{- end -}} + +{{/* +nginx scheme (http/https) +*/}} +{{- define "nginx.scheme" -}} +{{- if .Values.nginx.http.enabled -}} +{{- printf "%s" "http" -}} +{{- else -}} +{{- printf "%s" "https" -}} +{{- end -}} +{{- end -}} + +{{/* +nginx command +*/}} +{{- define "nginx.command" -}} +{{- if .Values.nginx.customCommand }} +{{ toYaml .Values.nginx.customCommand }} +{{- end }} +{{- end -}} + +{{/* +nginx port (8080/8443) based on http/https enabled +*/}} +{{- define "nginx.port" -}} +{{- if .Values.nginx.http.enabled -}} +{{- .Values.nginx.http.internalPort -}} +{{- else -}} +{{- .Values.nginx.https.internalPort -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customInitContainers value +*/}} +{{- define "artifactory.nginx.customInitContainers" -}} +{{- if .Values.nginx.customInitContainers -}} +{{- .Values.nginx.customInitContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve customVolumes value +*/}} +{{- define "artifactory.nginx.customVolumes" -}} +{{- if .Values.nginx.customVolumes -}} +{{- .Values.nginx.customVolumes -}} +{{- end -}} +{{- end -}} + + +{{/* +Resolve customVolumeMounts nginx value +*/}} +{{- define "artifactory.nginx.customVolumeMounts" -}} +{{- if .Values.nginx.customVolumeMounts -}} +{{- .Values.nginx.customVolumeMounts -}} +{{- end -}} +{{- end -}} + + +{{/* +Resolve customSidecarContainers value +*/}} +{{- define "artifactory.nginx.customSidecarContainers" -}} +{{- if .Values.nginx.customSidecarContainers -}} +{{- .Values.nginx.customSidecarContainers -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve Artifactory pod node selector value +*/}} +{{- define "artifactory.nodeSelector" -}} +nodeSelector: +{{- if .Values.global.nodeSelector }} +{{ toYaml .Values.global.nodeSelector | indent 2 }} +{{- else if .Values.artifactory.nodeSelector }} +{{ toYaml .Values.artifactory.nodeSelector | indent 2 }} +{{- end -}} +{{- end -}} + +{{/* +Resolve Nginx pods node selector value +*/}} +{{- define "nginx.nodeSelector" -}} +nodeSelector: +{{- if .Values.global.nodeSelector }} +{{ toYaml .Values.global.nodeSelector | indent 2 }} +{{- else if .Values.nginx.nodeSelector }} +{{ toYaml .Values.nginx.nodeSelector | indent 2 }} +{{- end -}} +{{- end -}} + +{{/* +Resolve unifiedCustomSecretVolumeName value +*/}} +{{- define "artifactory.unifiedCustomSecretVolumeName" -}} +{{- printf "%s-%s" (include "artifactory.name" .) ("unified-secret-volume") | trunc 63 -}} +{{- end -}} + +{{/* +Check the Duplication of volume names for secrets. If unifiedSecretInstallation is enabled then the method is checking for volume names, +if the volume exists in customVolume then an extra volume with the same name will not be getting added in unifiedSecretInstallation case. +*/}} +{{- define "artifactory.checkDuplicateUnifiedCustomVolume" -}} +{{- if or .Values.global.customVolumes .Values.artifactory.customVolumes -}} +{{- $val := (tpl (include "artifactory.customVolumes" .) .) | toJson -}} +{{- contains (include "artifactory.unifiedCustomSecretVolumeName" .) $val | toString -}} +{{- else -}} +{{- printf "%s" "false" -}} +{{- end -}} +{{- end -}} + +{{/* +Calculate the systemYaml from structured and unstructured text input +*/}} +{{- define "artifactory.finalSystemYaml" -}} +{{ tpl (mergeOverwrite (include "artifactory.systemYaml" . | fromYaml) .Values.artifactory.extraSystemYaml | toYaml) . }} +{{- end -}} + +{{/* +Calculate the systemYaml from the unstructured text input +*/}} +{{- define "artifactory.systemYaml" -}} +{{ include (print $.Template.BasePath "/_system-yaml-render.tpl") . }} +{{- end -}} + +{{/* +Metrics enabled +*/}} +{{- define "metrics.enabled" -}} +shared: + metrics: + enabled: true +{{- end }} + +{{/* +Resolve unified secret prepend release name +*/}} +{{- define "artifactory.unifiedSecretPrependReleaseName" -}} +{{- if .Values.artifactory.unifiedSecretPrependReleaseName }} +{{- printf "%s" (include "artifactory.fullname" .) -}} +{{- else }} +{{- printf "%s" (include "artifactory.name" .) -}} +{{- end }} +{{- end }} + +{{/* +Resolve artifactory metrics +*/}} +{{- define "artifactory.metrics" -}} +{{- if .Values.artifactory.openMetrics -}} +{{- if .Values.artifactory.openMetrics.enabled -}} +{{ include "metrics.enabled" . }} +{{- if .Values.artifactory.openMetrics.filebeat }} +{{- if .Values.artifactory.openMetrics.filebeat.enabled }} +{{ include "metrics.enabled" . }} + filebeat: +{{ tpl (.Values.artifactory.openMetrics.filebeat | toYaml) . | indent 6 }} +{{- end -}} +{{- end -}} +{{- end -}} +{{- else if .Values.artifactory.metrics -}} +{{- if .Values.artifactory.metrics.enabled -}} +{{ include "metrics.enabled" . }} +{{- if .Values.artifactory.metrics.filebeat }} +{{- if .Values.artifactory.metrics.filebeat.enabled }} +{{ include "metrics.enabled" . }} + filebeat: +{{ tpl (.Values.artifactory.metrics.filebeat | toYaml) . | indent 6 }} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Resolve nginx hosts value +*/}} +{{- define "artifactory.nginx.hosts" -}} +{{- if .Values.ingress.hosts }} +{{- range .Values.ingress.hosts -}} + {{- if contains "." . -}} + {{ "" | indent 0 }} ~(?.+)\.{{ . }} + {{- end -}} +{{- end -}} +{{- else if .Values.nginx.hosts }} +{{- range .Values.nginx.hosts -}} + {{- if contains "." . -}} + {{ "" | indent 0 }} ~(?.+)\.{{ . }} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_system-yaml-render.tpl b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_system-yaml-render.tpl new file mode 100644 index 0000000000..deaa773ea9 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/_system-yaml-render.tpl @@ -0,0 +1,5 @@ +{{- if .Values.artifactory.systemYaml -}} +{{- tpl .Values.artifactory.systemYaml . -}} +{{- else -}} +{{ (tpl ( $.Files.Get "files/system.yaml" ) .) }} +{{- end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/additional-resources.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/additional-resources.yaml new file mode 100644 index 0000000000..c4d06f08ad --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/additional-resources.yaml @@ -0,0 +1,3 @@ +{{ if .Values.additionalResources }} +{{ tpl .Values.additionalResources . }} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/admin-bootstrap-creds.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/admin-bootstrap-creds.yaml new file mode 100644 index 0000000000..eb2d613c6c --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/admin-bootstrap-creds.yaml @@ -0,0 +1,15 @@ +{{- if not (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) }} +{{- if and .Values.artifactory.admin.password (not .Values.artifactory.unifiedSecretInstallation) }} +kind: Secret +apiVersion: v1 +metadata: + name: {{ template "artifactory.fullname" . }}-bootstrap-creds + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + bootstrap.creds: {{ (printf "%s@%s=%s" .Values.artifactory.admin.username .Values.artifactory.admin.ip .Values.artifactory.admin.password) | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-access-config.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-access-config.yaml new file mode 100644 index 0000000000..4fcf85d94c --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-access-config.yaml @@ -0,0 +1,15 @@ +{{- if and .Values.access.accessConfig (not .Values.artifactory.unifiedSecretInstallation) }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory.fullname" . }}-access-config + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +type: Opaque +stringData: + access.config.patch.yml: | +{{ tpl (toYaml .Values.access.accessConfig) . | indent 4 }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-binarystore-secret.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-binarystore-secret.yaml new file mode 100644 index 0000000000..6b721dd4c7 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-binarystore-secret.yaml @@ -0,0 +1,18 @@ +{{- if and (not .Values.artifactory.persistence.customBinarystoreXmlSecret) (not .Values.artifactory.unifiedSecretInstallation) }} +kind: Secret +apiVersion: v1 +metadata: + name: {{ template "artifactory.fullname" . }}-binarystore + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +stringData: + binarystore.xml: |- +{{- if .Values.artifactory.persistence.binarystoreXml }} +{{ tpl .Values.artifactory.persistence.binarystoreXml . | indent 4 }} +{{- else }} +{{ tpl ( .Files.Get "files/binarystore.xml" ) . | indent 4 }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-configmaps.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-configmaps.yaml new file mode 100644 index 0000000000..359fa07d25 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-configmaps.yaml @@ -0,0 +1,13 @@ +{{ if .Values.artifactory.configMaps }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory.fullname" . }}-configmaps + labels: + app: {{ template "artifactory.fullname" . }} + chart: {{ .Chart.Name }}-{{ .Chart.Version }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: +{{ tpl .Values.artifactory.configMaps . | indent 2 }} +{{ end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-custom-secrets.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-custom-secrets.yaml new file mode 100644 index 0000000000..4b73e79fc3 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-custom-secrets.yaml @@ -0,0 +1,19 @@ +{{- if and .Values.artifactory.customSecrets (not .Values.artifactory.unifiedSecretInstallation) }} +{{- range .Values.artifactory.customSecrets }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory.fullname" $ }}-{{ .name }} + labels: + app: "{{ template "artifactory.name" $ }}" + chart: "{{ template "artifactory.chart" $ }}" + component: "{{ $.Values.artifactory.name }}" + heritage: {{ $.Release.Service | quote }} + release: {{ $.Release.Name | quote }} +type: Opaque +stringData: + {{ .key }}: | +{{ .data | indent 4 -}} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-database-secrets.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-database-secrets.yaml new file mode 100644 index 0000000000..f98d422e95 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-database-secrets.yaml @@ -0,0 +1,24 @@ +{{- if and (not .Values.database.secrets) (not .Values.postgresql.enabled) (not .Values.artifactory.unifiedSecretInstallation) }} +{{- if or .Values.database.url .Values.database.user .Values.database.password }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory.fullname" . }}-database-creds + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +type: Opaque +data: + {{- with .Values.database.url }} + db-url: {{ tpl . $ | b64enc | quote }} + {{- end }} + {{- with .Values.database.user }} + db-user: {{ tpl . $ | b64enc | quote }} + {{- end }} + {{- with .Values.database.password }} + db-password: {{ tpl . $ | b64enc | quote }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-gcp-credentials-secret.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-gcp-credentials-secret.yaml new file mode 100644 index 0000000000..72dee6bb84 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-gcp-credentials-secret.yaml @@ -0,0 +1,16 @@ +{{- if not .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} +{{- if and (.Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled) (not .Values.artifactory.unifiedSecretInstallation) }} +kind: Secret +apiVersion: v1 +metadata: + name: {{ template "artifactory.fullname" . }}-gcpcreds + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +stringData: + gcp.credentials.json: |- +{{ tpl .Values.artifactory.persistence.googleStorage.gcpServiceAccount.config . | indent 4 }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-hpa.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-hpa.yaml new file mode 100644 index 0000000000..01f8a9fb70 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-hpa.yaml @@ -0,0 +1,29 @@ +{{- if .Values.autoscaling.enabled }} + {{- if semverCompare ">=v1.23.0-0" .Capabilities.KubeVersion.Version }} +apiVersion: autoscaling/v2 + {{- else }} +apiVersion: autoscaling/v2beta2 + {{- end }} +kind: HorizontalPodAutoscaler +metadata: + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + name: {{ template "artifactory.fullname" . }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: StatefulSet + name: {{ template "artifactory.fullname" . }} + minReplicas: {{ .Values.autoscaling.minReplicas }} + maxReplicas: {{ .Values.autoscaling.maxReplicas }} + metrics: + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: {{ .Values.autoscaling.targetCPUUtilizationPercentage }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-installer-info.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-installer-info.yaml new file mode 100644 index 0000000000..cfb95b67d3 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-installer-info.yaml @@ -0,0 +1,16 @@ +kind: ConfigMap +apiVersion: v1 +metadata: + name: {{ template "artifactory.fullname" . }}-installer-info + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + installer-info.json: | +{{- if .Values.installerInfo -}} +{{- tpl .Values.installerInfo . | nindent 4 -}} +{{- else -}} +{{ (tpl ( .Files.Get "files/installer-info.json" | nindent 4 ) .) }} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-license-secret.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-license-secret.yaml new file mode 100644 index 0000000000..ba83aaf24d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-license-secret.yaml @@ -0,0 +1,16 @@ +{{ if and (not .Values.artifactory.unifiedSecretInstallation) (not .Values.artifactory.license.secret) (not .Values.artifactory.license.licenseKey) }} +{{- with .Values.artifactory.license.licenseKey }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "artifactory.fullname" $ }}-license + labels: + app: {{ template "artifactory.name" $ }} + chart: {{ template "artifactory.chart" $ }} + heritage: {{ $.Release.Service }} + release: {{ $.Release.Name }} +type: Opaque +data: + artifactory.lic: {{ . | b64enc | quote }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-migration-scripts.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-migration-scripts.yaml new file mode 100644 index 0000000000..4b1ba40276 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-migration-scripts.yaml @@ -0,0 +1,18 @@ +{{- if .Values.artifactory.migration.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory.fullname" . }}-migration-scripts + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + migrate.sh: | +{{ .Files.Get "files/migrate.sh" | indent 4 }} + migrationHelmInfo.yaml: | +{{ .Files.Get "files/migrationHelmInfo.yaml" | indent 4 }} + migrationStatus.sh: | +{{ .Files.Get "files/migrationStatus.sh" | indent 4 }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-networkpolicy.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-networkpolicy.yaml new file mode 100644 index 0000000000..d24203dc99 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-networkpolicy.yaml @@ -0,0 +1,34 @@ +{{- range .Values.networkpolicy }} +apiVersion: networking.k8s.io/v1 +kind: NetworkPolicy +metadata: + name: {{ template "artifactory.fullname" $ }}-{{ .name }}-networkpolicy + labels: + app: {{ template "artifactory.name" $ }} + chart: {{ template "artifactory.chart" $ }} + release: {{ $.Release.Name }} + heritage: {{ $.Release.Service }} +spec: +{{- if .podSelector }} + podSelector: +{{ .podSelector | toYaml | trimSuffix "\n" | indent 4 -}} +{{ else }} + podSelector: {} +{{- end }} + policyTypes: + {{- if .ingress }} + - Ingress + {{- end }} + {{- if .egress }} + - Egress + {{- end }} +{{- if .ingress }} + ingress: +{{ .ingress | toYaml | trimSuffix "\n" | indent 2 -}} +{{- end }} +{{- if .egress }} + egress: +{{ .egress | toYaml | trimSuffix "\n" | indent 2 -}} +{{- end }} +--- +{{- end -}} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-nfs-pvc.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-nfs-pvc.yaml new file mode 100644 index 0000000000..75d6d0c53d --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-nfs-pvc.yaml @@ -0,0 +1,101 @@ +{{- if eq .Values.artifactory.persistence.type "nfs" }} +### Artifactory HA data +apiVersion: v1 +kind: PersistentVolume +metadata: + name: {{ template "artifactory.fullname" . }}-data-pv + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + id: {{ template "artifactory.name" . }}-data-pv + type: nfs-volume +spec: + {{- if .Values.artifactory.persistence.nfs.mountOptions }} + mountOptions: +{{ toYaml .Values.artifactory.persistence.nfs.mountOptions | indent 4 }} + {{- end }} + capacity: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain + nfs: + server: {{ .Values.artifactory.persistence.nfs.ip }} + path: "{{ .Values.artifactory.persistence.nfs.haDataMount }}" + readOnly: false +--- +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: + name: {{ template "artifactory.fullname" . }}-data-pvc + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + type: nfs-volume +spec: + accessModes: + - ReadWriteOnce + storageClassName: "" + resources: + requests: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + selector: + matchLabels: + id: {{ template "artifactory.name" . }}-data-pv + app: {{ template "artifactory.name" . }} + release: {{ .Release.Name }} +--- +### Artifactory HA backup +apiVersion: v1 +kind: PersistentVolume +metadata: + name: {{ template "artifactory.fullname" . }}-backup-pv + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + id: {{ template "artifactory.name" . }}-backup-pv + type: nfs-volume +spec: + {{- if .Values.artifactory.persistence.nfs.mountOptions }} + mountOptions: +{{ toYaml .Values.artifactory.persistence.nfs.mountOptions | indent 4 }} + {{- end }} + capacity: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain + nfs: + server: {{ .Values.artifactory.persistence.nfs.ip }} + path: "{{ .Values.artifactory.persistence.nfs.haBackupMount }}" + readOnly: false +--- +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: + name: {{ template "artifactory.fullname" . }}-backup-pvc + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + type: nfs-volume +spec: + accessModes: + - ReadWriteOnce + storageClassName: "" + resources: + requests: + storage: {{ .Values.artifactory.persistence.nfs.capacity }} + selector: + matchLabels: + id: {{ template "artifactory.name" . }}-backup-pv + app: {{ template "artifactory.name" . }} + release: {{ .Release.Name }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-pdb.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-pdb.yaml new file mode 100644 index 0000000000..68876d23b0 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/artifactory-pdb.yaml @@ -0,0 +1,24 @@ +{{- if .Values.artifactory.minAvailable -}} +{{- if semverCompare "= 107.79.x), just set databaseUpgradeReady=true \n" .Values.databaseUpgradeReady | quote }} +{{- end }} +{{- with .Values.artifactory.statefulset.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +{{- if and (eq (include "artifactory.isUsingDerby" .) "true") (gt (.Values.artifactory.replicaCount | int64) 1) }} + {{- fail "Derby database is not supported in HA mode" }} +{{- end }} +{{- if .Values.artifactory.postStartCommand }} + {{- fail ".Values.artifactory.postStartCommand is not supported and should be replaced with .Values.artifactory.lifecycle.postStart.exec.command" }} +{{- end }} +{{- if eq .Values.artifactory.persistence.type "aws-s3" }} + {{- fail "\nPersistence storage type 'aws-s3' is deprecated and is not supported and should be replaced with 'aws-s3-v3'" }} +{{- end }} +{{- if or .Values.artifactory.persistence.googleStorage.identity .Values.artifactory.persistence.googleStorage.credential }} + {{- fail "\nGCP Bucket Authentication with Identity and Credential is deprecated" }} +{{- end }} +{{- if (eq (.Values.artifactory.setSecurityContext | toString) "false" ) }} + {{- fail "\n You need to set security context at the pod level. .Values.artifactory.setSecurityContext is no longer supported. Replace it with .Values.artifactory.podSecurityContext" }} +{{- end }} +{{- if or .Values.artifactory.uid .Values.artifactory.gid }} +{{- if or (not (eq (.Values.artifactory.uid | toString) "1030" )) (not (eq (.Values.artifactory.gid | toString) "1030" )) }} + {{- fail "\n .Values.artifactory.uid and .Values.artifactory.gid are no longer supported. You need to set these values at the pod security context level. Replace them with .Values.artifactory.podSecurityContext.runAsUser .Values.artifactory.podSecurityContext.runAsGroup and .Values.artifactory.podSecurityContext.fsGroup" }} +{{- end }} +{{- end }} +{{- if or .Values.artifactory.fsGroupChangePolicy .Values.artifactory.seLinuxOptions }} + {{- fail "\n .Values.artifactory.fsGroupChangePolicy and .Values.artifactory.seLinuxOptions are no longer supported. You need to set these values at the pod security context level. Replace them with .Values.artifactory.podSecurityContext.fsGroupChangePolicy and .Values.artifactory.podSecurityContext.seLinuxOptions" }} +{{- end }} +{{- if .Values.initContainerImage }} + {{- fail "\n .Values.initContainerImage is no longer supported. Replace it with .Values.initContainers.image.registry .Values.initContainers.image.repository and .Values.initContainers.image.tag" }} +{{- end }} +spec: + serviceName: {{ template "artifactory.name" . }} + replicas: {{ .Values.artifactory.replicaCount }} + updateStrategy: {{- toYaml .Values.artifactory.updateStrategy | nindent 4 }} + selector: + matchLabels: + app: {{ template "artifactory.name" . }} + role: {{ template "artifactory.name" . }} + release: {{ .Release.Name }} + template: + metadata: + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + role: {{ template "artifactory.name" . }} + component: {{ .Values.artifactory.name }} + release: {{ .Release.Name }} + {{- with .Values.artifactory.labels }} +{{ toYaml . | indent 8 }} + {{- end }} + annotations: + {{- if not .Values.artifactory.unifiedSecretInstallation }} + checksum/database-secrets: {{ include (print $.Template.BasePath "/artifactory-database-secrets.yaml") . | sha256sum }} + checksum/binarystore: {{ include (print $.Template.BasePath "/artifactory-binarystore-secret.yaml") . | sha256sum }} + checksum/systemyaml: {{ include (print $.Template.BasePath "/artifactory-system-yaml.yaml") . | sha256sum }} + {{- if .Values.access.accessConfig }} + checksum/access-config: {{ include (print $.Template.BasePath "/artifactory-access-config.yaml") . | sha256sum }} + {{- end }} + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + checksum/gcpcredentials: {{ include (print $.Template.BasePath "/artifactory-gcp-credentials-secret.yaml") . | sha256sum }} + {{- end }} + {{- if not (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) }} + checksum/admin-creds: {{ include (print $.Template.BasePath "/admin-bootstrap-creds.yaml") . | sha256sum }} + {{- end }} + {{- else }} + checksum/artifactory-unified-secret: {{ include (print $.Template.BasePath "/artifactory-unified-secret.yaml") . | sha256sum }} + {{- end }} + {{- with .Values.artifactory.annotations }} +{{ toYaml . | indent 8 }} + {{- end }} + spec: + {{- if .Values.artifactory.schedulerName }} + schedulerName: {{ .Values.artifactory.schedulerName | quote }} + {{- end }} + {{- if .Values.artifactory.priorityClass.existingPriorityClass }} + priorityClassName: {{ .Values.artifactory.priorityClass.existingPriorityClass }} + {{- else -}} + {{- if .Values.artifactory.priorityClass.create }} + priorityClassName: {{ default (include "artifactory.fullname" .) .Values.artifactory.priorityClass.name }} + {{- end }} + {{- end }} + serviceAccountName: {{ template "artifactory.serviceAccountName" . }} + terminationGracePeriodSeconds: {{ add .Values.artifactory.terminationGracePeriodSeconds 10 }} + {{- if or .Values.imagePullSecrets .Values.global.imagePullSecrets }} +{{- include "artifactory.imagePullSecrets" . | indent 6 }} + {{- end }} + {{- if .Values.artifactory.podSecurityContext.enabled }} + securityContext: {{- omit .Values.artifactory.podSecurityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.artifactory.topologySpreadConstraints }} + topologySpreadConstraints: +{{ tpl (toYaml .Values.artifactory.topologySpreadConstraints) . | indent 8 }} + {{- end }} + initContainers: + {{- if or .Values.artifactory.customInitContainersBegin .Values.global.customInitContainersBegin }} +{{ tpl (include "artifactory.customInitContainersBegin" .) . | indent 6 }} + {{- end }} + {{- if .Values.artifactory.persistence.enabled }} + {{- if .Values.artifactory.deleteDBPropertiesOnStartup }} + - name: "delete-db-properties" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - 'rm -fv {{ .Values.artifactory.persistence.mountPath }}/etc/db.properties' + volumeMounts: + - name: artifactory-volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + {{- end }} + {{- end }} + {{- if or (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) .Values.artifactory.admin.password }} + - name: "access-bootstrap-creds" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > + echo "Preparing {{ .Values.artifactory.persistence.mountPath }}/etc/access/bootstrap.creds"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access; + cp -Lrf /tmp/access/bootstrap.creds {{ .Values.artifactory.persistence.mountPath }}/etc/access/bootstrap.creds; + chmod 600 {{ .Values.artifactory.persistence.mountPath }}/etc/access/bootstrap.creds; + volumeMounts: + - name: artifactory-volume + mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + {{- if or (not .Values.artifactory.unifiedSecretInstallation) (and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey) }} + - name: access-bootstrap-creds + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/access/bootstrap.creds" + {{- if and .Values.artifactory.admin.secret .Values.artifactory.admin.dataKey }} + subPath: {{ .Values.artifactory.admin.dataKey }} + {{- else }} + subPath: "bootstrap.creds" + {{- end }} + {{- end }} + - name: 'copy-system-configurations' + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - '/bin/bash' + - '-c' + - > + if [[ -e "{{ .Values.artifactory.persistence.mountPath }}/etc/filebeat.yaml" ]]; then chmod 644 {{ .Values.artifactory.persistence.mountPath }}/etc/filebeat.yaml; fi; + echo "Copy system.yaml to {{ .Values.artifactory.persistence.mountPath }}/etc"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access/keys/trusted; + {{- if .Values.systemYamlOverride.existingSecret }} + cp -fv /tmp/etc/{{ .Values.systemYamlOverride.dataKey }} {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml; + {{- else }} + cp -fv /tmp/etc/system.yaml {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml; + {{- end }} + echo "Copy binarystore.xml file"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/artifactory; + cp -fv /tmp/etc/artifactory/binarystore.xml {{ .Values.artifactory.persistence.mountPath }}/etc/artifactory/binarystore.xml; + {{- if .Values.access.accessConfig }} + echo "Copy access.config.patch.yml to {{ .Values.artifactory.persistence.mountPath }}/etc/access"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/access; + cp -fv /tmp/etc/access.config.patch.yml {{ .Values.artifactory.persistence.mountPath }}/etc/access/access.config.patch.yml; + {{- end }} + {{- if .Values.access.resetAccessCAKeys }} + echo "Resetting Access CA Keys"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys; + touch {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys/reset_ca_keys; + {{- end }} + {{- if .Values.access.customCertificatesSecretName }} + echo "Copying custom certificates to {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys; + cp -fv /tmp/etc/tls.crt {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys/ca.crt; + cp -fv /tmp/etc/tls.key {{ .Values.artifactory.persistence.mountPath }}/bootstrap/etc/access/keys/ca.private.key; + {{- end }} + {{- if or .Values.artifactory.joinKey .Values.global.joinKey .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName }} + echo "Copy joinKey to {{ .Values.artifactory.persistence.mountPath }}/bootstrap/access/etc/security"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/access/etc/security; + echo -n ${ARTIFACTORY_JOIN_KEY} > {{ .Values.artifactory.persistence.mountPath }}/bootstrap/access/etc/security/join.key; + {{- end }} + {{- if or .Values.artifactory.jfConnectToken .Values.artifactory.jfConnectTokenSecretName }} + echo "Copy jfConnectToken to {{ .Values.artifactory.persistence.mountPath }}/bootstrap/jfconnect/registration_token"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/bootstrap/jfconnect/; + echo -n ${ARTIFACTORY_JFCONNECT_TOKEN} > {{ .Values.artifactory.persistence.mountPath }}/bootstrap/jfconnect/registration_token; + {{- end }} + {{- if or .Values.artifactory.masterKey .Values.global.masterKey .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + echo "Copy masterKey to {{ .Values.artifactory.persistence.mountPath }}/etc/security"; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/etc/security; + echo -n ${ARTIFACTORY_MASTER_KEY} > {{ .Values.artifactory.persistence.mountPath }}/etc/security/master.key; + {{- end }} + env: + {{- if or .Values.artifactory.joinKey .Values.global.joinKey .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName }} + - name: ARTIFACTORY_JOIN_KEY + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.joinKeySecretName .Values.global.joinKeySecretName }} + name: {{ include "artifactory.joinKeySecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: join-key + {{- end }} + {{- if or .Values.artifactory.jfConnectToken .Values.artifactory.jfConnectSecretName }} + - name: ARTIFACTORY_JFCONNECT_TOKEN + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.jfConnectTokenSecretName }} + name: {{ include "artifactory.jfConnectTokenSecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: jfconnect-token + {{- end }} + {{- if or .Values.artifactory.masterKey .Values.global.masterKey .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + - name: ARTIFACTORY_MASTER_KEY + valueFrom: + secretKeyRef: + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.masterKeySecretName .Values.global.masterKeySecretName }} + name: {{ include "artifactory.masterKeySecretName" . }} + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: master-key + {{- end }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.systemYamlOverride.existingSecret }} + - name: systemyaml + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + {{- if .Values.systemYamlOverride.existingSecret }} + mountPath: "/tmp/etc/{{.Values.systemYamlOverride.dataKey}}" + subPath: {{ .Values.systemYamlOverride.dataKey }} + {{- else }} + mountPath: "/tmp/etc/system.yaml" + subPath: "system.yaml" + {{- end }} + + ######################## Binarystore ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Access config ########################## + {{- if .Values.access.accessConfig }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + - name: access-config + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/access.config.patch.yml" + subPath: "access.config.patch.yml" + {{- end }} + + ######################## Access certs external secret ########################## + {{- if .Values.access.customCertificatesSecretName }} + - name: access-certs + mountPath: "/tmp/etc/tls.crt" + subPath: tls.crt + - name: access-certs + mountPath: "/tmp/etc/tls.key" + subPath: tls.key + {{- end }} + + {{- if or .Values.artifactory.customCertificates.enabled .Values.global.customCertificates.enabled }} + - name: copy-custom-certificates + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > +{{ include "artifactory.copyCustomCerts" . | indent 10 }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath }} + - name: ca-certs + mountPath: "/tmp/certs" + {{- end }} + + {{- if .Values.artifactory.circleOfTrustCertificatesSecret }} + - name: copy-circle-of-trust-certificates + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - 'bash' + - '-c' + - > +{{ include "artifactory.copyCircleOfTrustCertsCerts" . | indent 10 }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath }} + - name: circle-of-trust-certs + mountPath: "/tmp/circleoftrustcerts" + {{- end }} + + {{- if .Values.waitForDatabase }} + {{- if .Values.postgresql.enabled }} + - name: "wait-for-db" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + resources: +{{ toYaml .Values.initContainers.resources | indent 10 }} + command: + - /bin/bash + - -c + - | + echo "Waiting for postgresql to come up" + ready=false; + while ! $ready; do echo waiting; + timeout 2s bash -c " + {{- if .Values.artifactory.migration.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.artifactory.migration.preStartCommand . }}; + {{- end }} + scriptsPath="/opt/jfrog/artifactory/app/bin"; + mkdir -p $scriptsPath; + echo "Copy migration scripts and Run migration"; + cp -fv /tmp/migrate.sh $scriptsPath/migrate.sh; + cp -fv /tmp/migrationHelmInfo.yaml $scriptsPath/migrationHelmInfo.yaml; + cp -fv /tmp/migrationStatus.sh $scriptsPath/migrationStatus.sh; + mkdir -p {{ .Values.artifactory.persistence.mountPath }}/log; + bash $scriptsPath/migrationStatus.sh {{ include "artifactory.app.version" . }} {{ .Values.artifactory.migration.timeoutSeconds }} > >(tee {{ .Values.artifactory.persistence.mountPath }}/log/helm-migration.log) 2>&1; + env: + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.artifactory.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: migration-scripts + mountPath: "/tmp/migrate.sh" + subPath: migrate.sh + - name: migration-scripts + mountPath: "/tmp/migrationHelmInfo.yaml" + subPath: migrationHelmInfo.yaml + - name: migration-scripts + mountPath: "/tmp/migrationStatus.sh" + subPath: migrationStatus.sh + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystore Xml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: "binarystore.xml" + + ######################## Artifactory persistence google storage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + {{- end }} + + ######################## CustomVolumeMounts ########################## + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory.customVolumeMounts" .) . | indent 8 }} + {{- end }} +{{- end }} + {{- if .Values.hostAliases }} + hostAliases: +{{ toYaml .Values.hostAliases | indent 6 }} + {{- end }} + containers: + {{- if .Values.splitServicesToContainers }} + - name: {{ .Values.router.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "router") }} + imagePullPolicy: {{ .Values.router.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/router/app/bin/entrypoint-router.sh + {{- with .Values.router.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_ROUTER_TOPOLOGY_LOCAL_REQUIREDSERVICETYPES + value: {{ include "artifactory.router.requiredServiceTypes" . }} +{{- with .Values.router.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - name: http + containerPort: {{ .Values.router.internalPort }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.router.persistence.mountPath | quote }} +{{- with .Values.router.customVolumeMounts }} +{{ tpl . $ | indent 8 }} +{{- end }} + resources: +{{ toYaml .Values.router.resources | indent 10 }} + {{- if .Values.router.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.router.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.router.readinessProbe.enabled }} + readinessProbe: +{{ tpl .Values.router.readinessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.router.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.router.livenessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.frontend.enabled }} + - name: {{ .Values.frontend.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/third-party/node/bin/node /opt/jfrog/artifactory/app/frontend/bin/server/dist/bundle.js /opt/jfrog/artifactory/app/frontend + {{- with .Values.frontend.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if and (gt (.Values.artifactory.replicaCount | int64) 1) (eq (include "artifactory.isImageProType" .) "true") (eq (include "artifactory.isUsingDerby" .) "false") }} + - name : JF_SHARED_NODE_HAENABLED + value: "true" + {{- end }} +{{- with .Values.frontend.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.frontend.resources | indent 10 }} + {{- if .Values.frontend.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.frontend.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.frontend.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.frontend.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.evidence.enabled }} + - name: {{ .Values.evidence.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/evidence/bin/jf-evidence start + {{- with .Values.evidence.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.evidence.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.evidence.internalPort }} + name: http-evidence + - containerPort: {{ .Values.evidence.externalPort }} + name: grpc-evidence + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.evidence.resources | indent 10 }} + {{- if .Values.evidence.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.evidence.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.evidence.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.evidence.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.metadata.enabled }} + - name: {{ .Values.metadata.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/metadata/bin/jf-metadata start + {{- with .Values.metadata.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.metadata.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory.customVolumeMounts" .) . | indent 8 }} + {{- end }} + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.metadata.resources | indent 10 }} + {{- if .Values.metadata.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.metadata.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.metadata.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.metadata.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.event.enabled }} + - name: {{ .Values.event.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/event/bin/jf-event start + {{- with .Values.event.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.event.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.event.resources | indent 10 }} + {{- if .Values.event.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.event.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.event.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.event.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if and .Values.jfconnect.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" .Values.artifactory.image.repository)) }} + - name: {{ .Values.jfconnect.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/jfconnect/bin/jf-connect start + {{- with .Values.jfconnect.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.jfconnect.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.jfconnect.resources | indent 10 }} + {{- if .Values.jfconnect.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.jfconnect.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.jfconnect.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.jfconnect.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if and .Values.access.enabled (not (.Values.access.runOnArtifactoryTomcat | default false)) }} + - name: {{ .Values.access.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.access.resources }} + resources: +{{ toYaml .Values.access.resources | indent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + set -e; + {{- if .Values.access.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.access.preStartCommand . }}; + {{- end }} + exec /opt/jfrog/artifactory/app/access/bin/entrypoint-access.sh + {{- with .Values.access.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + {{- if and (gt (.Values.artifactory.replicaCount | int64) 1) (eq (include "artifactory.isImageProType" .) "true") (eq (include "artifactory.isUsingDerby" .) "false") }} + - name : JF_SHARED_NODE_HAENABLED + value: "true" + {{- end }} + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.access.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + {{- if .Values.artifactory.customPersistentVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentVolumeClaim.mountPath }} + {{- end }} + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence googleStorage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + {{- end }} + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory.customVolumeMounts" .) . | indent 8 }} + {{- end }} + {{- if .Values.access.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.access.startupProbe.config . | indent 10 }} + {{- end }} + {{- if semverCompare " + exec /opt/jfrog/artifactory/app/third-party/java/bin/java {{ .Values.federation.extraJavaOpts }} -jar /opt/jfrog/artifactory/app/rtfs/lib/jf-rtfs + {{- with .Values.federation.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name + # TODO - Password,Url,Username - should be derived from env variable +{{- with .Values.federation.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.federation.internalPort }} + name: http-rtfs + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.federation.resources | indent 10 }} + {{- if .Values.federation.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.federation.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.federation.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.federation.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- if .Values.observability.enabled }} + - name: {{ .Values.observability.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + exec /opt/jfrog/artifactory/app/observability/bin/jf-observability start + {{- with .Values.observability.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + - name: JF_SHARED_NODE_ID + valueFrom: + fieldRef: + fieldPath: metadata.name +{{- with .Values.observability.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + volumeMounts: + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + resources: +{{ toYaml .Values.observability.resources | indent 10 }} + {{- if .Values.observability.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.observability.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.observability.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.observability.livenessProbe.config . | indent 10 }} + {{- end }} + {{- end }} + {{- end }} + - name: {{ .Values.artifactory.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "artifactory") }} + imagePullPolicy: {{ .Values.artifactory.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.artifactory.resources }} + resources: +{{ toYaml .Values.artifactory.resources | indent 10 }} + {{- end }} + command: + - '/bin/bash' + - '-c' + - > + set -e; + if [ -d /artifactory_extra_conf ] && [ -d /artifactory_bootstrap ]; then + echo "Copying bootstrap config from /artifactory_extra_conf to /artifactory_bootstrap"; + cp -Lrfv /artifactory_extra_conf/ /artifactory_bootstrap/; + fi; + {{- if .Values.artifactory.configMapName }} + echo "Copying bootstrap configs"; + cp -Lrf /bootstrap/* /artifactory_bootstrap/; + {{- end }} + {{- if .Values.artifactory.userPluginSecrets }} + echo "Copying plugins"; + cp -Lrf /tmp/plugin/*/* /artifactory_bootstrap/plugins; + {{- end }} + {{- range .Values.artifactory.copyOnEveryStartup }} + {{- $targetPath := printf "%s/%s" $.Values.artifactory.persistence.mountPath .target }} + {{- $baseDirectory := regexFind ".*/" $targetPath }} + mkdir -p {{ $baseDirectory }}; + cp -Lrf {{ .source }} {{ $.Values.artifactory.persistence.mountPath }}/{{ .target }}; + {{- end }} + {{- if .Values.artifactory.preStartCommand }} + echo "Running custom preStartCommand command"; + {{ tpl .Values.artifactory.preStartCommand . }}; + {{- end }} + exec /entrypoint-artifactory.sh + {{- with .Values.artifactory.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + env: + {{- if and (gt (.Values.artifactory.replicaCount | int64) 1) (eq (include "artifactory.isImageProType" .) "true") (eq (include "artifactory.isUsingDerby" .) "false") }} + - name : JF_SHARED_NODE_HAENABLED + value: "true" + {{- end }} + {{- if .Values.aws.license.enabled }} + - name: IS_AWS_LICENSE + value: "true" + - name: AWS_REGION + value: {{ .Values.aws.region | quote }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: AWS_WEB_IDENTITY_REFRESH_TOKEN_FILE + value: "/var/run/secrets/product-license/license_token" + - name: AWS_ROLE_ARN + valueFrom: + secretKeyRef: + name: {{ .Values.aws.licenseConfigSecretName }} + key: iam_role + {{- end }} + {{- end }} + {{- if .Values.splitServicesToContainers }} + - name : JF_ROUTER_ENABLED + value: "true" + - name : JF_ROUTER_SERVICE_ENABLED + value: "false" + - name : JF_EVENT_ENABLED + value: "false" + - name : JF_METADATA_ENABLED + value: "false" + - name : JF_FRONTEND_ENABLED + value: "false" + - name: JF_FEDERATION_ENABLED + value: "false" + - name : JF_OBSERVABILITY_ENABLED + value: "false" + - name : JF_JFCONNECT_SERVICE_ENABLED + value: "false" + - name : JF_EVIDENCE_ENABLED + value: "false" + {{- if not (.Values.access.runOnArtifactoryTomcat | default false) }} + - name : JF_ACCESS_ENABLED + value: "false" + {{- end}} + {{- end}} + {{- if and (not .Values.waitForDatabase) (not .Values.postgresql.enabled) }} + - name: SKIP_WAIT_FOR_EXTERNAL_DB + value: "true" + {{- end }} + {{- if or .Values.database.secrets.user .Values.database.user }} + - name: JF_SHARED_DATABASE_USERNAME + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.user }} + name: {{ tpl .Values.database.secrets.user.name . }} + key: {{ tpl .Values.database.secrets.user.key . }} + {{- else if .Values.database.user }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-user + {{- end }} + {{- end }} + {{ if or .Values.database.secrets.password .Values.database.password .Values.postgresql.enabled }} + - name: JF_SHARED_DATABASE_PASSWORD + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.password }} + name: {{ tpl .Values.database.secrets.password.name . }} + key: {{ tpl .Values.database.secrets.password.key . }} + {{- else if .Values.database.password }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-password + {{- else if .Values.postgresql.enabled }} + name: {{ .Release.Name }}-postgresql + key: postgresql-password + {{- end }} + {{- end }} + {{- if or .Values.database.secrets.url .Values.database.url }} + - name: JF_SHARED_DATABASE_URL + valueFrom: + secretKeyRef: + {{- if .Values.database.secrets.url }} + name: {{ tpl .Values.database.secrets.url.name . }} + key: {{ tpl .Values.database.secrets.url.key . }} + {{- else if .Values.database.url }} + {{- if not .Values.artifactory.unifiedSecretInstallation }} + name: {{ template "artifactory.fullname" . }}-database-creds + {{- else }} + name: "{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret" + {{- end }} + key: db-url + {{- end }} + {{- end }} +{{- with .Values.artifactory.extraEnvironmentVariables }} +{{ tpl (toYaml .) $ | indent 8 }} +{{- end }} + ports: + - containerPort: {{ .Values.artifactory.internalPort }} + name: http + - containerPort: {{ .Values.artifactory.internalArtifactoryPort }} + name: http-internal + - containerPort: {{ .Values.federation.internalPort }} + name: http-rtfs + {{- if .Values.artifactory.javaOpts.jmx.enabled }} + - containerPort: {{ .Values.artifactory.javaOpts.jmx.port }} + name: tcp-jmx + {{- end }} + {{- if .Values.artifactory.ssh.enabled }} + - containerPort: {{ .Values.artifactory.ssh.internalPort }} + name: tcp-ssh + {{- end }} + volumeMounts: + {{- if .Values.artifactory.customPersistentVolumeClaim }} + - name: {{ .Values.artifactory.customPersistentVolumeClaim.name }} + mountPath: {{ .Values.artifactory.customPersistentVolumeClaim.mountPath }} + {{- end }} + {{- if .Values.aws.licenseConfigSecretName }} + - name: awsmp-product-license + mountPath: "/var/run/secrets/product-license" + {{- end }} + {{- if .Values.artifactory.userPluginSecrets }} + - name: bootstrap-plugins + mountPath: "/artifactory_bootstrap/plugins/" + {{- range .Values.artifactory.userPluginSecrets }} + - name: {{ tpl . $ }} + mountPath: "/tmp/plugin/{{ tpl . $ }}" + {{- end }} + {{- end }} + - name: artifactory-volume + mountPath: {{ .Values.artifactory.persistence.mountPath | quote }} + + ######################## Artifactory config map ########################## + {{- if .Values.artifactory.configMapName }} + - name: bootstrap-config + mountPath: "/bootstrap/" + {{- end }} + + ######################## Artifactory persistence nfs ########################## + {{- if eq .Values.artifactory.persistence.type "nfs" }} + - name: artifactory-data + mountPath: "{{ .Values.artifactory.persistence.nfs.dataDir }}" + - name: artifactory-backup + mountPath: "{{ .Values.artifactory.persistence.nfs.backupDir }}" + {{- else }} + + ######################## Artifactory persistence binarystoreXml ########################## + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.customBinarystoreXmlSecret }} + - name: binarystore-xml + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/tmp/etc/artifactory/binarystore.xml" + subPath: binarystore.xml + + ######################## Artifactory persistence googleStorage ########################## + {{- if .Values.artifactory.persistence.googleStorage.gcpServiceAccount.enabled }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.persistence.googleStorage.gcpServiceAccount.customSecretName }} + - name: gcpcreds-json + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/gcp.credentials.json" + subPath: gcp.credentials.json + {{- end }} + {{- end }} + + ######################## Artifactory license ########################## + {{- if or .Values.artifactory.license.secret .Values.artifactory.license.licenseKey }} + {{- if or (not .Values.artifactory.unifiedSecretInstallation) .Values.artifactory.license.secret }} + - name: artifactory-license + {{- else }} + - name: {{ include "artifactory.unifiedCustomSecretVolumeName" . }} + {{- end }} + mountPath: "/artifactory_bootstrap/artifactory.cluster.license" + {{- if .Values.artifactory.license.secret }} + subPath: {{ .Values.artifactory.license.dataKey }} + {{- else if .Values.artifactory.license.licenseKey }} + subPath: artifactory.lic + {{- end }} + {{- end }} + + - name: installer-info + mountPath: "/artifactory_bootstrap/info/installer-info.json" + subPath: installer-info.json + {{- if or .Values.artifactory.customVolumeMounts .Values.global.customVolumeMounts }} +{{ tpl (include "artifactory.customVolumeMounts" .) . | indent 8 }} + {{- end }} + {{- if .Values.artifactory.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.artifactory.startupProbe.config . | indent 10 }} + {{- end }} + {{- if and (not .Values.splitServicesToContainers) (semverCompare "=1.18.0-0" .Capabilities.KubeVersion.GitVersion) }} + ingressClassName: {{ .Values.ingress.className }} + {{- end }} + {{- if .Values.ingress.defaultBackend.enabled }} + {{- if .Capabilities.APIVersions.Has "networking.k8s.io/v1" }} + defaultBackend: + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- else }} + backend: + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- end }} + {{- end }} + rules: +{{- if .Values.ingress.hosts }} + {{- if .Capabilities.APIVersions.Has "networking.k8s.io/v1" }} + {{- range $host := .Values.ingress.hosts }} + - host: {{ $host | quote }} + http: + paths: + - path: {{ $.Values.ingress.routerPath }} + pathType: ImplementationSpecific + backend: + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + {{- if not $.Values.ingress.disableRouterBypass }} + - path: {{ $.Values.ingress.artifactoryPath }} + pathType: ImplementationSpecific + backend: + service: + name: {{ $serviceName }} + port: + number: {{ $artifactoryServicePort }} + {{- end }} + {{- if and $.Values.federation.enabled (not (regexMatch "^.*(oss|cpp-ce|jcr).*$" $.Values.artifactory.image.repository)) }} + - path: {{ $.Values.ingress.rtfsPath }} + pathType: ImplementationSpecific + backend: + service: + name: {{ $serviceName }} + port: + number: {{ $.Values.federation.internalPort }} + {{- end }} + {{- end }} + {{- else }} + {{- range $host := .Values.ingress.hosts }} + - host: {{ $host | quote }} + http: + paths: + - path: {{ $.Values.ingress.routerPath }} + backend: + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} + {{- if not $.Values.ingress.disableRouterBypass }} + - path: {{ $.Values.ingress.artifactoryPath }} + backend: + serviceName: {{ $serviceName }} + servicePort: {{ $artifactoryServicePort }} + {{- end }} + {{- end }} + {{- end }} +{{- end -}} + {{- with .Values.ingress.additionalRules }} +{{ tpl . $ | indent 2 }} + {{- end }} + + {{- if .Values.ingress.tls }} + tls: +{{ toYaml .Values.ingress.tls | indent 4 }} + {{- end -}} + +{{- if .Values.customIngress }} +--- +{{ .Values.customIngress | toYaml | trimSuffix "\n" }} +{{- end -}} +{{- end -}} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/logger-configmap.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/logger-configmap.yaml new file mode 100644 index 0000000000..41a078b024 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/logger-configmap.yaml @@ -0,0 +1,63 @@ +{{- if or .Values.artifactory.loggers .Values.artifactory.catalinaLoggers }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory.fullname" . }}-logger + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + tail-log.sh: | + #!/bin/sh + + LOG_DIR=$1 + LOG_NAME=$2 + PID= + + # Wait for log dir to appear + while [ ! -d ${LOG_DIR} ]; do + sleep 1 + done + + cd ${LOG_DIR} + + LOG_PREFIX=$(echo ${LOG_NAME} | sed 's/.log$//g') + + # Find the log to tail + LOG_FILE=$(ls -1t ./${LOG_PREFIX}.log 2>/dev/null) + + # Wait for the log file + while [ -z "${LOG_FILE}" ]; do + sleep 1 + LOG_FILE=$(ls -1t ./${LOG_PREFIX}.log 2>/dev/null) + done + + echo "Log file ${LOG_FILE} is ready!" + + # Get inode number + INODE_ID=$(ls -i ${LOG_FILE}) + + # echo "Tailing ${LOG_FILE}" + tail -F ${LOG_FILE} & + PID=$! + + # Loop forever to see if a new log was created + while true; do + # Check inode number + NEW_INODE_ID=$(ls -i ${LOG_FILE}) + + # If inode number changed, this means log was rotated and need to start a new tail + if [ "${INODE_ID}" != "${NEW_INODE_ID}" ]; then + kill -9 ${PID} 2>/dev/null + INODE_ID="${NEW_INODE_ID}" + + # Start a new tail + tail -F ${LOG_FILE} & + PID=$! + fi + sleep 1 + done + +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-artifactory-conf.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-artifactory-conf.yaml new file mode 100644 index 0000000000..3434489943 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-artifactory-conf.yaml @@ -0,0 +1,18 @@ +{{- if and (not .Values.nginx.customArtifactoryConfigMap) .Values.nginx.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory.fullname" . }}-nginx-artifactory-conf + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + artifactory.conf: | +{{- if .Values.nginx.artifactoryConf }} +{{ tpl .Values.nginx.artifactoryConf . | indent 4 }} +{{- else }} +{{ tpl ( .Files.Get "files/nginx-artifactory-conf.yaml" ) . | indent 4 }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-certificate-secret.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-certificate-secret.yaml new file mode 100644 index 0000000000..f13d401747 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-certificate-secret.yaml @@ -0,0 +1,14 @@ +{{- if and (not .Values.nginx.tlsSecretName) .Values.nginx.enabled .Values.nginx.https.enabled }} +apiVersion: v1 +kind: Secret +type: kubernetes.io/tls +metadata: + name: {{ template "artifactory.fullname" . }}-nginx-certificate + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: +{{ ( include "artifactory.gen-certs" . ) | indent 2 }} +{{- end }} diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-conf.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-conf.yaml new file mode 100644 index 0000000000..31219d58a9 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-conf.yaml @@ -0,0 +1,18 @@ +{{- if and (not .Values.nginx.customConfigMap) .Values.nginx.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "artifactory.fullname" . }}-nginx-conf + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +data: + nginx.conf: | +{{- if .Values.nginx.mainConf }} +{{ tpl .Values.nginx.mainConf . | indent 4 }} +{{- else }} +{{ tpl ( .Files.Get "files/nginx-main-conf.yaml" ) . | indent 4 }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-deployment.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-deployment.yaml new file mode 100644 index 0000000000..774bedccaf --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-deployment.yaml @@ -0,0 +1,223 @@ +{{- if .Values.nginx.enabled -}} +{{- $serviceName := include "artifactory.fullname" . -}} +{{- $servicePort := .Values.artifactory.externalPort -}} +apiVersion: apps/v1 +kind: {{ .Values.nginx.kind }} +metadata: + name: {{ template "artifactory.nginx.fullname" . }} + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} + component: {{ .Values.nginx.name }} +{{- if .Values.nginx.labels }} +{{ toYaml .Values.nginx.labels | indent 4 }} +{{- end }} +{{- with .Values.nginx.deployment.annotations }} + annotations: +{{ toYaml . | indent 4 }} +{{- end }} +spec: +{{- if eq .Values.nginx.kind "StatefulSet" }} + serviceName: {{ template "artifactory.nginx.fullname" . }} +{{- end }} +{{- if ne .Values.nginx.kind "DaemonSet" }} + replicas: {{ .Values.nginx.replicaCount }} +{{- end }} + selector: + matchLabels: + app: {{ template "artifactory.name" . }} + release: {{ .Release.Name }} + component: {{ .Values.nginx.name }} + template: + metadata: + annotations: + checksum/nginx-conf: {{ include (print $.Template.BasePath "/nginx-conf.yaml") . | sha256sum }} + checksum/nginx-artifactory-conf: {{ include (print $.Template.BasePath "/nginx-artifactory-conf.yaml") . | sha256sum }} + {{- range $key, $value := .Values.nginx.annotations }} + {{ $key }}: {{ $value | quote }} + {{- end }} + labels: + app: {{ template "artifactory.name" . }} + chart: {{ template "artifactory.chart" . }} + component: {{ .Values.nginx.name }} + heritage: {{ .Release.Service }} + release: {{ .Release.Name }} +{{- if .Values.nginx.labels }} +{{ toYaml .Values.nginx.labels | indent 8 }} +{{- end }} + spec: + {{- if .Values.nginx.podSecurityContext.enabled }} + securityContext: {{- omit .Values.nginx.podSecurityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + serviceAccountName: {{ template "artifactory.serviceAccountName" . }} + terminationGracePeriodSeconds: {{ .Values.nginx.terminationGracePeriodSeconds }} + {{- if or .Values.imagePullSecrets .Values.global.imagePullSecrets }} +{{- include "artifactory.imagePullSecrets" . | indent 6 }} + {{- end }} + {{- if .Values.nginx.priorityClassName }} + priorityClassName: {{ .Values.nginx.priorityClassName | quote }} + {{- end }} + {{- if .Values.nginx.topologySpreadConstraints }} + topologySpreadConstraints: +{{ tpl (toYaml .Values.nginx.topologySpreadConstraints) . | indent 8 }} + {{- end }} + initContainers: + {{- if .Values.nginx.customInitContainers }} +{{ tpl (include "artifactory.nginx.customInitContainers" .) . | indent 6 }} + {{- end }} + - name: "setup" + image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + {{- if .Values.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + command: + - '/bin/sh' + - '-c' + - > + rm -rfv {{ .Values.nginx.persistence.mountPath }}/lost+found; + mkdir -p {{ .Values.nginx.persistence.mountPath }}/logs; + resources: + {{- toYaml .Values.initContainers.resources | nindent 10 }} + volumeMounts: + - mountPath: {{ .Values.nginx.persistence.mountPath | quote }} + name: nginx-volume + containers: + - name: {{ .Values.nginx.name }} + image: {{ include "artifactory.getImageInfoByValue" (list . "nginx") }} + imagePullPolicy: {{ .Values.nginx.image.pullPolicy }} + {{- if .Values.nginx.containerSecurityContext.enabled }} + securityContext: {{- omit .Values.nginx.containerSecurityContext "enabled" | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.nginx.customCommand }} + command: +{{- tpl (include "nginx.command" .) . | indent 10 }} + {{- end }} + ports: +{{ if .Values.nginx.customPorts }} +{{ toYaml .Values.nginx.customPorts | indent 8 }} +{{ end }} + # DEPRECATION NOTE: The following is to maintain support for values pre 1.3.1 and + # will be cleaned up in a later version + {{- if .Values.nginx.http }} + {{- if .Values.nginx.http.enabled }} + - containerPort: {{ .Values.nginx.http.internalPort }} + name: http + {{- end }} + {{- else }} # DEPRECATED + - containerPort: {{ .Values.nginx.internalPortHttp }} + name: http-internal + {{- end }} + {{- if .Values.nginx.https }} + {{- if .Values.nginx.https.enabled }} + - containerPort: {{ .Values.nginx.https.internalPort }} + name: https + {{- end }} + {{- else }} # DEPRECATED + - containerPort: {{ .Values.nginx.internalPortHttps }} + name: https-internal + {{- end }} + {{- if .Values.artifactory.ssh.enabled }} + - containerPort: {{ .Values.nginx.ssh.internalPort }} + name: tcp-ssh + {{- end }} + {{- with .Values.nginx.lifecycle }} + lifecycle: +{{ toYaml . | indent 10 }} + {{- end }} + volumeMounts: + - name: nginx-conf + mountPath: /etc/nginx/nginx.conf + subPath: nginx.conf + - name: nginx-artifactory-conf + mountPath: "{{ .Values.nginx.persistence.mountPath }}/conf.d/" + - name: nginx-volume + mountPath: {{ .Values.nginx.persistence.mountPath | quote }} + {{- if .Values.nginx.https.enabled }} + - name: ssl-certificates + mountPath: "{{ .Values.nginx.persistence.mountPath }}/ssl" + {{- end }} + {{- if .Values.nginx.customVolumeMounts }} +{{ tpl (include "artifactory.nginx.customVolumeMounts" .) . | indent 8 }} + {{- end }} + resources: +{{ toYaml .Values.nginx.resources | indent 10 }} + {{- if .Values.nginx.startupProbe.enabled }} + startupProbe: +{{ tpl .Values.nginx.startupProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.nginx.readinessProbe.enabled }} + readinessProbe: +{{ tpl .Values.nginx.readinessProbe.config . | indent 10 }} + {{- end }} + {{- if .Values.nginx.livenessProbe.enabled }} + livenessProbe: +{{ tpl .Values.nginx.livenessProbe.config . | indent 10 }} + {{- end }} + {{- $mountPath := .Values.nginx.persistence.mountPath }} + {{- range .Values.nginx.loggers }} + - name: {{ . | replace "_" "-" | replace "." "-" }} + image: {{ include "artifactory.getImageInfoByValue" (list $ "initContainers") }} + imagePullPolicy: {{ $.Values.initContainers.image.pullPolicy }} + command: + - tail + args: + - '-F' + - '{{ $mountPath }}/logs/{{ . }}' + volumeMounts: + - name: nginx-volume + mountPath: {{ $mountPath }} + resources: +{{ toYaml $.Values.nginx.loggersResources | indent 10 }} + {{- end }} + {{- if .Values.nginx.customSidecarContainers }} +{{ tpl (include "artifactory.nginx.customSidecarContainers" .) . | indent 6 }} + {{- end }} + {{- if or .Values.nginx.nodeSelector .Values.global.nodeSelector }} +{{ tpl (include "nginx.nodeSelector" .) . | indent 6 }} + {{- end }} + {{- with .Values.nginx.affinity }} + affinity: +{{ toYaml . | indent 8 }} + {{- end }} + {{- with .Values.nginx.tolerations }} + tolerations: +{{ toYaml . | indent 8 }} + {{- end }} + volumes: + {{- if .Values.nginx.customVolumes }} +{{ tpl (include "artifactory.nginx.customVolumes" .) . | indent 6 }} + {{- end }} + - name: nginx-conf + configMap: + {{- if .Values.nginx.customConfigMap }} + name: {{ .Values.nginx.customConfigMap }} + {{- else }} + name: {{ template "artifactory.fullname" . }}-nginx-conf + {{- end }} + - name: nginx-artifactory-conf + configMap: + {{- if .Values.nginx.customArtifactoryConfigMap }} + name: {{ .Values.nginx.customArtifactoryConfigMap }} + {{- else }} + name: {{ template "artifactory.fullname" . }}-nginx-artifactory-conf + {{- end }} + - name: nginx-volume + {{- if .Values.nginx.persistence.enabled }} + persistentVolumeClaim: + claimName: {{ .Values.nginx.persistence.existingClaim | default (include "artifactory.nginx.fullname" .) }} + {{- else }} + emptyDir: {} + {{- end }} + {{- if .Values.nginx.https.enabled }} + - name: ssl-certificates + secret: + {{- if .Values.nginx.tlsSecretName }} + secretName: {{ .Values.nginx.tlsSecretName }} + {{- else }} + secretName: {{ template "artifactory.fullname" . }}-nginx-certificate + {{- end }} + {{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-pdb.yaml b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-pdb.yaml new file mode 100644 index 0000000000..dff0c23a3e --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/charts/artifactory/templates/nginx-pdb.yaml @@ -0,0 +1,23 @@ +{{- if .Values.nginx.enabled -}} +{{- if semverCompare "; + # kubernetes.io/tls-acme: "true" + # nginx.ingress.kubernetes.io/proxy-body-size: "0" + labels: {} + # traffic-type: external + # traffic-type: internal + tls: [] + ## Secrets must be manually created in the namespace. + # - secretName: chart-example-tls + # hosts: + # - artifactory.domain.example + + ## Additional ingress rules + additionalRules: [] + ## This is an experimental feature, enabling this feature will route all traffic through the Router. + disableRouterBypass: false +## Allows to add custom ingress +customIngress: "" +networkpolicy: [] +## Allows all ingress and egress +# - name: artifactory +# podSelector: +# matchLabels: +# app: artifactory +# egress: +# - {} +# ingress: +# - {} +## Uncomment to allow only artifactory pods to communicate with postgresql (if postgresql.enabled is true) +# - name: postgresql +# podSelector: +# matchLabels: +# app: postgresql +# ingress: +# - from: +# - podSelector: +# matchLabels: +# app: artifactory + +## Apply horizontal pod auto scaling on artifactory pods +## Ref: https://kubernetes.io/docs/tasks/run-application/horizontal-pod-autoscale/ +autoscaling: + enabled: false + minReplicas: 1 + maxReplicas: 3 + targetCPUUtilizationPercentage: 70 +## You can use a pre-existing secret with keys license_token and iam_role by specifying licenseConfigSecretName +## Example : Create a generic secret using `kubectl create secret generic --from-literal=license_token=${TOKEN} --from-literal=iam_role=${ROLE_ARN}` +aws: + license: + enabled: false + licenseConfigSecretName: + region: us-east-1 +## Container Security Context +## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/#set-the-security-context-for-a-container +## @param containerSecurityContext.enabled Enabled containers' Security Context +## @param containerSecurityContext.runAsNonRoot Set container's Security Context runAsNonRoot +## @param containerSecurityContext.privileged Set container's Security Context privileged +## @param containerSecurityContext.allowPrivilegeEscalation Set container's Security Context allowPrivilegeEscalation +## @param containerSecurityContext.capabilities.drop List of capabilities to be dropped +## @param containerSecurityContext.seccompProfile.type Set container's Security Context seccomp profile +## +containerSecurityContext: + enabled: true + runAsNonRoot: true + privileged: false + allowPrivilegeEscalation: false + seccompProfile: + type: RuntimeDefault + capabilities: + drop: + - ALL +## The following router settings are to configure only when splitServicesToContainers set to true +router: + name: router + image: + registry: releases-docker.jfrog.io + repository: jfrog/router + tag: 7.118.3 + pullPolicy: IfNotPresent + serviceRegistry: + ## Service registry (Access) TLS verification skipped if enabled + insecure: false + internalPort: 8082 + externalPort: 8082 + tlsEnabled: false + ## Extra environment variables that can be used to tune router to your needs. + ## Uncomment and set value as needed + extraEnvironmentVariables: + # - name: MY_ENV_VAR + # value: "" + resources: {} + # requests: + # memory: "100Mi" + # cpu: "100m" + # limits: + # memory: "1Gi" + # cpu: "1" + + ## Add lifecycle hooks for router container + lifecycle: + ## From Artifactory versions 7.52.x, Wait for Artifactory to complete any open uploads or downloads before terminating + preStop: + exec: + command: ["sh", "-c", "while [[ $(curl --fail --silent --connect-timeout 2 http://localhost:8081/artifactory/api/v1/system/liveness) =~ OK ]]; do echo Artifactory is still alive; sleep 2; done"] + # postStart: + # exec: + # command: ["/bin/sh", "-c", "echo Hello from the postStart handler"] + ## Add custom volumesMounts + customVolumeMounts: | + # - name: custom-script + # mountPath: /scripts/script.sh + # subPath: script.sh + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl -s -k --fail --max-time {{ .Values.probes.timeoutSeconds }} {{ include "artifactory.scheme" . }}://localhost:{{ .Values.router.internalPort }}/router/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " prepended. + unifiedSecretPrependReleaseName: true + ## For HA installation, set this value > 1. This is only supported in Artifactory 7.25.x (appVersions) and above. + replicaCount: 1 + # minAvailable: 1 + + ## Note that by default we use appVersion to get image tag/version + image: + registry: releases-docker.jfrog.io + repository: jfrog/artifactory-pro + # tag: + pullPolicy: IfNotPresent + labels: {} + updateStrategy: + type: RollingUpdate + ## ref: https://kubernetes.io/docs/tasks/administer-cluster/configure-multiple-schedulers/ + schedulerName: + ## Create a priority class for the Artifactory pod or use an existing one + ## NOTE - Maximum allowed value of a user defined priority is 1000000000 + priorityClass: + create: false + value: 1000000000 + ## Override default name + # name: + ## Use an existing priority class + # existingPriorityClass: + ## Spread Artifactory pods evenly across your nodes or some other topology + topologySpreadConstraints: [] + # - maxSkew: 1 + # topologyKey: kubernetes.io/hostname + # whenUnsatisfiable: DoNotSchedule + # labelSelector: + # matchLabels: + # app: '{{ template "artifactory.name" . }}' + # role: '{{ template "artifactory.name" . }}' + # release: "{{ .Release.Name }}" + + ## Delete the db.properties file in ARTIFACTORY_HOME/etc/db.properties + deleteDBPropertiesOnStartup: true + ## certificates added to this secret will be copied to $JFROG_HOME/artifactory/var/etc/security/keys/trusted directory + customCertificates: + enabled: false + # certificateSecretName: + database: + maxOpenConnections: 80 + tomcat: + maintenanceConnector: + port: 8091 + connector: + maxThreads: 200 + sendReasonPhrase: false + extraConfig: 'acceptCount="400"' + ## Support for metrics is only available for Artifactory 7.7.x (appVersions) and above. + ## To enable set `.Values.artifactory.metrics.enabled` to `true` + ## Note : Depricated openMetrics as part of 7.87.x and renamed to `metrics` + ## Refer - https://www.jfrog.com/confluence/display/JFROG/Open+Metrics + metrics: + enabled: false + ## Settings for pushing metrics to Insight - enable filebeat to true + filebeat: + enabled: false + log: + enabled: false + ## Log level for filebeat. Possible values: debug, info, warning, or error. + level: "info" + ## Elasticsearch details for filebeat to connect + elasticsearch: + url: "Elasticsearch url where JFrog Insight is installed For example, http://:8082" + username: "" + password: "" + ## Support for Cold Artifact Storage + ## set 'coldStorage.enabled' to 'true' only for Artifactory instance that you are designating as the Cold instance + ## Refer - https://jfrog.com/help/r/jfrog-platform-administration-documentation/setting-up-cold-artifact-storage + coldStorage: + enabled: false + ## This directory is intended for use with NFS eventual configuration for HA + haDataDir: + enabled: false + path: + haBackupDir: + enabled: false + path: + ## Files to copy to ARTIFACTORY_HOME/ on each Artifactory startup + ## Note : From 107.46.x chart versions, copyOnEveryStartup is not needed for binarystore.xml, it is always copied via initContainers + copyOnEveryStartup: + ## Absolute path + # - source: /artifactory_bootstrap/artifactory.lic + ## Relative to ARTIFACTORY_HOME/ + # target: etc/artifactory/ + + ## Sidecar containers for tailing Artifactory logs + loggers: [] + # - access-audit.log + # - access-request.log + # - access-security-audit.log + # - access-service.log + # - artifactory-access.log + # - artifactory-event.log + # - artifactory-import-export.log + # - artifactory-request.log + # - artifactory-service.log + # - frontend-request.log + # - frontend-service.log + # - metadata-request.log + # - metadata-service.log + # - router-request.log + # - router-service.log + # - router-traefik.log + # - derby.log + + ## Loggers containers resources + loggersResources: {} + # requests: + # memory: "10Mi" + # cpu: "10m" + # limits: + # memory: "100Mi" + # cpu: "50m" + + ## Sidecar containers for tailing Tomcat (catalina) logs + catalinaLoggers: [] + # - tomcat-catalina.log + # - tomcat-localhost.log + + ## Tomcat (catalina) loggers resources + catalinaLoggersResources: {} + # requests: + # memory: "10Mi" + # cpu: "10m" + # limits: + # memory: "100Mi" + # cpu: "50m" + + ## Migration support from 6.x to 7.x + migration: + enabled: false + timeoutSeconds: 3600 + ## Extra pre-start command in migration Init Container to install JDBC driver for MySql/MariaDb/Oracle + # preStartCommand: "mkdir -p /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib; cd /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib && curl -o /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib/mysql-connector-java-5.1.41.jar https://repo1.maven.org/maven2/mysql/mysql-connector-java/5.1.41/mysql-connector-java-5.1.41.jar" + ## Add custom init containers execution before predefined init containers + customInitContainersBegin: | + # - name: "custom-setup" + # image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + # imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + # securityContext: + # runAsNonRoot: true + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - NET_RAW + # command: + # - 'sh' + # - '-c' + # - 'touch {{ .Values.artifactory.persistence.mountPath }}/example-custom-setup' + # volumeMounts: + # - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + # name: artifactory-volume + ## Add custom init containers execution after predefined init containers + customInitContainers: | + # - name: "custom-systemyaml-setup" + # image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + # imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + # securityContext: + # runAsNonRoot: true + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - NET_RAW + # command: + # - 'sh' + # - '-c' + # - 'curl -o {{ .Values.artifactory.persistence.mountPath }}/etc/system.yaml https:///systemyaml' + # volumeMounts: + # - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + # name: artifactory-volume + ## Add custom sidecar containers + ## - The provided example uses a custom volume (customVolumes) + customSidecarContainers: | + # - name: "sidecar-list-etc" + # image: {{ include "artifactory.getImageInfoByValue" (list . "initContainers") }} + # imagePullPolicy: {{ .Values.initContainers.image.pullPolicy }} + # securityContext: + # runAsNonRoot: true + # allowPrivilegeEscalation: false + # capabilities: + # drop: + # - NET_RAW + # command: + # - 'sh' + # - '-c' + # - 'sh /scripts/script.sh' + # volumeMounts: + # - mountPath: "{{ .Values.artifactory.persistence.mountPath }}" + # name: artifactory-volume + # - mountPath: "/scripts/script.sh" + # name: custom-script + # subPath: script.sh + # resources: + # requests: + # memory: "32Mi" + # cpu: "50m" + # limits: + # memory: "128Mi" + # cpu: "100m" + ## Add custom volumes + ## If .Values.artifactory.unifiedSecretInstallation is true then secret name should be '{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret' + customVolumes: | + # - name: custom-script + # configMap: + # name: custom-script + ## Add custom volumesMounts + customVolumeMounts: | + # - name: custom-script + # mountPath: "/scripts/script.sh" + # subPath: script.sh + # - name: posthook-start + # mountPath: "/scripts/posthoook-start.sh" + # subPath: posthoook-start.sh + # - name: prehook-start + # mountPath: "/scripts/prehook-start.sh" + # subPath: prehook-start.sh + ## Add custom persistent volume mounts - Available to the entire namespace + customPersistentVolumeClaim: {} + # name: + # mountPath: + # accessModes: + # - "-" + # size: + # storageClassName: + + ## Artifactory license. + license: + ## licenseKey is the license key in plain text. Use either this or the license.secret setting + licenseKey: + ## If artifactory.license.secret is passed, it will be mounted as + ## ARTIFACTORY_HOME/etc/artifactory.lic and loaded at run time. + secret: + ## The dataKey should be the name of the secret data key created. + dataKey: + ## Create configMap with artifactory.config.import.xml and security.import.xml and pass name of configMap in following parameter + configMapName: + ## Add any list of configmaps to Artifactory + configMaps: | + # posthook-start.sh: |- + # echo "This is a post start script" + # posthook-end.sh: |- + # echo "This is a post end script" + ## List of secrets for Artifactory user plugins. + ## One Secret per plugin's files. + userPluginSecrets: + # - archive-old-artifacts + # - build-cleanup + # - webhook + # - '{{ template "my-chart.fullname" . }}' + + ## Artifactory requires a unique master key. + ## You can generate one with the command: "openssl rand -hex 32" + ## An initial one is auto generated by Artifactory on first startup. + # masterKey: FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF + ## Alternatively, you can use a pre-existing secret with a key called master-key by specifying masterKeySecretName + # masterKeySecretName: + + ## Join Key to connect other services to Artifactory + ## IMPORTANT: Setting this value overrides the existing joinKey + ## IMPORTANT: You should NOT use the example joinKey for a production deployment! + # joinKey: EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE + ## Alternatively, you can use a pre-existing secret with a key called join-key by specifying joinKeySecretName + # joinKeySecretName: + + ## Registration Token for JFConnect + # jfConnectToken: + ## Alternatively, you can use a pre-existing secret with a key called jfconnect-token by specifying jfConnectTokenSecretName + # jfConnectTokenSecretName: + + ## Add custom secrets - secret per file + ## If .Values.artifactory.unifiedSecretInstallation is true then secret name should be '{{ template "artifactory.unifiedSecretPrependReleaseName" . }}-unified-secret' common to all secrets + customSecrets: + # - name: custom-secret + # key: custom-secret.yaml + # data: > + # custom_secret_config: + # parameter1: value1 + # parameter2: value2 + # - name: custom-secret2 + # key: custom-secret2.config + # data: | + # here the custom secret 2 config + + ## If false, all service console logs will not redirect to a common console.log + consoleLog: false + ## admin allows to set the password for the default admin user. + ## See: https://www.jfrog.com/confluence/display/JFROG/Users+and+Groups#UsersandGroups-RecreatingtheDefaultAdminUserrecreate + admin: + ip: "127.0.0.1" + username: "admin" + password: + secret: + dataKey: + ## Extra pre-start command to install JDBC driver for MySql/MariaDb/Oracle + # preStartCommand: "mkdir -p /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib; cd /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib && curl -o /opt/jfrog/artifactory/var/bootstrap/artifactory/tomcat/lib/mysql-connector-java-5.1.41.jar https://repo1.maven.org/maven2/mysql/mysql-connector-java/5.1.41/mysql-connector-java-5.1.41.jar" + + ## Add lifecycle hooks for artifactory container + lifecycle: {} + # postStart: + # exec: + # command: ["/bin/sh", "-c", "echo Hello from the postStart handler"] + # preStop: + # exec: + # command: ["/bin/sh","-c","echo Hello from the preStop handler"] + + ## Extra environment variables that can be used to tune Artifactory to your needs. + ## Uncomment and set value as needed + extraEnvironmentVariables: + # - name: SERVER_XML_ARTIFACTORY_PORT + # value: "8081" + # - name: SERVER_XML_ARTIFACTORY_MAX_THREADS + # value: "200" + # - name: SERVER_XML_ACCESS_MAX_THREADS + # value: "50" + # - name: SERVER_XML_ARTIFACTORY_EXTRA_CONFIG + # value: "" + # - name: SERVER_XML_ACCESS_EXTRA_CONFIG + # value: "" + # - name: SERVER_XML_EXTRA_CONNECTOR + # value: "" + # - name: DB_POOL_MAX_ACTIVE + # value: "100" + # - name: DB_POOL_MAX_IDLE + # value: "10" + # - name: MY_SECRET_ENV_VAR + # valueFrom: + # secretKeyRef: + # name: my-secret-name + # key: my-secret-key + + ## System YAML entries now reside under files/system.yaml. + ## You can provide the specific values that you want to add or override under 'artifactory.extraSystemYaml'. + ## For example: + ## extraSystemYaml: + ## shared: + ## node: + ## id: my-instance + ## The entries provided under 'artifactory.extraSystemYaml' are merged with files/system.yaml to create the final system.yaml. + ## If you have already provided system.yaml under, 'artifactory.systemYaml', the values in that entry take precedence over files/system.yaml + ## You can modify specific entries with your own value under `artifactory.extraSystemYaml`, The values under extraSystemYaml overrides the values under 'artifactory.systemYaml' and files/system.yaml + extraSystemYaml: {} + ## systemYaml is intentionally commented and the previous content has been moved under files/system.yaml. + ## You have to add the all entries of the system.yaml file here, and it overrides the values in files/system.yaml. + # systemYaml: + annotations: {} + service: + name: artifactory + type: ClusterIP + ## @param service.ipFamilyPolicy Controller Service ipFamilyPolicy (optional, cloud specific) + ## This can be either SingleStack, PreferDualStack or RequireDualStack + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilyPolicy: "" + ## @param service.ipFamilies Controller Service ipFamilies (optional, cloud specific) + ## This can be either ["IPv4"], ["IPv6"], ["IPv4", "IPv6"] or ["IPv6", "IPv4"] + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilies: [] + ## For supporting whitelist on the Artifactory service (useful if setting service.type=LoadBalancer) + ## Set this to a list of IP CIDR ranges + ## Example: loadBalancerSourceRanges: ['10.10.10.5/32', '10.11.10.5/32'] + ## or pass from helm command line + ## Example: helm install ... --set nginx.service.loadBalancerSourceRanges='{10.10.10.5/32,10.11.10.5/32}' + loadBalancerSourceRanges: [] + annotations: {} + ## If the type is NodePort you can set a fixed port + # nodePort: 32082 + statefulset: + annotations: {} + ## IMPORTANT: If overriding artifactory.internalPort: + ## DO NOT use port lower than 1024 as Artifactory runs as non-root and cannot bind to ports lower than 1024! + externalPort: 8082 + internalPort: 8082 + externalArtifactoryPort: 8081 + internalArtifactoryPort: 8081 + terminationGracePeriodSeconds: 30 + ## Pod Security Context + ## ref: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ + ## @param artifactory.podSecurityContext.enabled Enable security context + ## @param artifactory.podSecurityContext.runAsNonRoot Set pod's Security Context runAsNonRoot + ## @param artifactory.podSecurityContext.runAsUser User ID for the pod + ## @param artifactory.podSecurityContext.runASGroup Group ID for the pod + ## @param artifactory.podSecurityContext.fsGroup Group ID for the pod + ## + podSecurityContext: + enabled: true + runAsNonRoot: true + runAsUser: 1030 + runAsGroup: 1030 + fsGroup: 1030 + # fsGroupChangePolicy: "Always" + # seLinuxOptions: {} + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl -s -k --fail --max-time {{ .Values.probes.timeoutSeconds }} http://localhost:{{ .Values.artifactory.tomcat.maintenanceConnector.port }}/artifactory/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare "", + # "private_key_id": "?????", + # "private_key": "-----BEGIN PRIVATE KEY-----\n????????==\n-----END PRIVATE KEY-----\n", + # "client_email": "???@j.iam.gserviceaccount.com", + # "client_id": "???????", + # "auth_uri": "https://accounts.google.com/o/oauth2/auth", + # "token_uri": "https://oauth2.googleapis.com/token", + # "auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs", + # "client_x509_cert_url": "https://www.googleapis.com/robot/v1....." + # } + endpoint: commondatastorage.googleapis.com + httpsOnly: false + ## Set a unique bucket name + bucketName: "artifactory-gcp" + ## GCP Bucket Authentication with Identity and Credential is deprecated. + ## identity: + ## credential: + path: "artifactory/filestore" + bucketExists: false + useInstanceCredentials: false + enableSignedUrlRedirect: false + ## For artifactory.persistence.type aws-s3-v3, s3-storage-v3-direct, cluster-s3-storage-v3, s3-storage-v3-archive + awsS3V3: + testConnection: false + identity: + credential: + region: + bucketName: artifactory-aws + path: artifactory/filestore + endpoint: + port: + useHttp: + maxConnections: 50 + connectionTimeout: + socketTimeout: + kmsServerSideEncryptionKeyId: + kmsKeyRegion: + kmsCryptoMode: + useInstanceCredentials: true + usePresigning: false + signatureExpirySeconds: 300 + signedUrlExpirySeconds: 30 + cloudFrontDomainName: + cloudFrontKeyPairId: + cloudFrontPrivateKey: + enableSignedUrlRedirect: false + enablePathStyleAccess: false + multiPartLimit: + multipartElementSize: + ## For artifactory.persistence.type azure-blob, azure-blob-storage-direct, cluster-azure-blob-storage, azure-blob-storage-v2-direct + azureBlob: + accountName: + accountKey: + endpoint: + containerName: + multiPartLimit: 100000000 + multipartElementSize: 50000000 + testConnection: false + ## artifactory 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) + ## + # storageClassName: "-" + ## Annotations for the Persistent Volume Claim + annotations: {} + ## Uncomment the following resources definitions or pass them from command line + ## to control the cpu and memory resources allocated by the Kubernetes cluster + resources: {} + # requests: + # memory: "1Gi" + # cpu: "500m" + # limits: + # memory: "2Gi" + # cpu: "1" + ## The following Java options are passed to the java process running Artifactory. + ## You should set them according to the resources set above + javaOpts: + # xms: "1g" + # xmx: "2g" + jmx: + enabled: false + port: 9010 + host: + ssl: false + ## When authenticate is true, accessFile and passwordFile are required + authenticate: false + accessFile: + passwordFile: + # corePoolSize: 24 + # other: "" + nodeSelector: {} + tolerations: [] + affinity: {} + ## Only used if "affinity" is empty + podAntiAffinity: + ## Valid values are "soft" or "hard"; any other value indicates no anti-affinity + type: "soft" + topologyKey: "kubernetes.io/hostname" + ssh: + enabled: false + internalPort: 1339 + externalPort: 1339 +frontend: + name: frontend + enabled: true + internalPort: 8070 + ## Extra environment variables that can be used to tune frontend to your needs. + ## Uncomment and set value as needed + extraEnvironmentVariables: + # - name: MY_ENV_VAR + # value: "" + resources: {} + # requests: + # memory: "100Mi" + # cpu: "100m" + # limits: + # memory: "1Gi" + # cpu: "1" + + ## Add lifecycle hooks for frontend container + lifecycle: {} + # postStart: + # exec: + # command: ["/bin/sh", "-c", "echo Hello from the postStart handler"] + # preStop: + # exec: + # command: ["/bin/sh","-c","echo Hello from the preStop handler"] + + ## Session settings + session: + ## Time in minutes after which the frontend token will need to be refreshed + timeoutMinutes: '30' + ## The following settings are to configure the frequency of the liveness and startup probes when splitServicesToContainers set to true + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl --fail --max-time {{ .Values.probes.timeoutSeconds }} http://localhost:{{ .Values.frontend.internalPort }}/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " --cert=ca.crt --key=ca.private.key` + # customCertificatesSecretName: + + ## When resetAccessCAKeys is true, Access will regenerate the CA certificate and matching private key + # resetAccessCAKeys: false + database: + maxOpenConnections: 80 + tomcat: + connector: + maxThreads: 50 + sendReasonPhrase: false + extraConfig: 'acceptCount="100"' + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl --fail --max-time {{ .Values.probes.timeoutSeconds }} http://localhost:8040/access/api/v1/system/liveness + initialDelaySeconds: {{ if semverCompare " /var/opt/jfrog/nginx/message"] + # preStop: + # exec: + # command: ["/bin/sh","-c","nginx -s quit; while killall -0 nginx; do sleep 1; done"] + + ## Sidecar containers for tailing Nginx logs + loggers: [] + # - access.log + # - error.log + + ## Loggers containers resources + loggersResources: {} + # requests: + # memory: "64Mi" + # cpu: "25m" + # limits: + # memory: "128Mi" + # cpu: "50m" + + ## Logs options + logs: + stderr: false + stdout: false + level: warn + ## A list of custom ports to expose on the NGINX pod. Follows the conventional Kubernetes yaml syntax for container ports. + customPorts: [] + # - containerPort: 8066 + # name: docker + + ## The nginx main conf was moved to files/nginx-main-conf.yaml. This key is commented out to keep support for the old configuration + # mainConf: | + + ## The nginx artifactory conf was moved to files/nginx-artifactory-conf.yaml. This key is commented out to keep support for the old configuration + # artifactoryConf: | + customInitContainers: "" + customSidecarContainers: "" + customVolumes: "" + customVolumeMounts: "" + customCommand: + ## allows overwriting the command for the nginx container. + ## defaults to [ 'nginx', '-g', 'daemon off;' ] + + service: + ## For minikube, set this to NodePort, elsewhere use LoadBalancer + type: LoadBalancer + ssloffload: false + ## @param service.ipFamilyPolicy Controller Service ipFamilyPolicy (optional, cloud specific) + ## This can be either SingleStack, PreferDualStack or RequireDualStack + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilyPolicy: "" + ## @param service.ipFamilies Controller Service ipFamilies (optional, cloud specific) + ## This can be either ["IPv4"], ["IPv6"], ["IPv4", "IPv6"] or ["IPv6", "IPv4"] + ## ref: https://kubernetes.io/docs/concepts/services-networking/dual-stack/#services + ## + ipFamilies: [] + ## For supporting whitelist on the Nginx LoadBalancer service + ## Set this to a list of IP CIDR ranges + ## Example: loadBalancerSourceRanges: ['10.10.10.5/32', '10.11.10.5/32'] + ## or pass from helm command line + ## Example: helm install ... --set nginx.service.loadBalancerSourceRanges='{10.10.10.5/32,10.11.10.5/32}' + loadBalancerSourceRanges: [] + annotations: {} + ## Provide static ip address + loadBalancerIP: + ## There are two available options: "Cluster" (default) and "Local". + externalTrafficPolicy: Cluster + ## If the type is NodePort you can set a fixed port + # nodePort: 32082 + ## A list of custom ports to be exposed on nginx service. Follows the conventional Kubernetes yaml syntax for service ports. + customPorts: [] + # - port: 8066 + # targetPort: 8066 + # protocol: TCP + # name: docker + ## Renamed nginx internalPort 80,443 to 8080,8443 to support openshift + http: + enabled: true + externalPort: 80 + internalPort: 8080 + https: + enabled: true + externalPort: 443 + internalPort: 8443 + ssh: + internalPort: 1339 + externalPort: 1339 + ## DEPRECATED: The following will be removed in a future release + # externalPortHttp: 8080 + # internalPortHttp: 8080 + # externalPortHttps: 8443 + # internalPortHttps: 8443 + + ## The following settings are to configure the frequency of the liveness and readiness probes. + livenessProbe: + enabled: true + config: | + exec: + command: + - sh + - -c + - curl -s -k --fail --max-time {{ .Values.probes.timeoutSeconds }} {{ include "nginx.scheme" . }}://localhost:{{ include "nginx.port" . }}/ + initialDelaySeconds: {{ if semverCompare " + ## 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) + ## + # storageClassName: "-" + resources: {} + # requests: + # memory: "250Mi" + # cpu: "100m" + # limits: + # memory: "250Mi" + # cpu: "500m" + nodeSelector: {} + tolerations: [] + affinity: {} +## Database configurations +## Use the wait-for-db init container. Set to false to skip +waitForDatabase: true +## Configuration values for the PostgreSQL dependency sub-chart +## ref: https://github.com/bitnami/charts/blob/master/bitnami/postgresql/README.md +postgresql: + enabled: true + image: + registry: releases-docker.jfrog.io + repository: bitnami/postgresql + tag: 15.6.0-debian-11-r16 + postgresqlUsername: artifactory + postgresqlPassword: "" + postgresqlDatabase: artifactory + postgresqlExtendedConf: + listenAddresses: "*" + maxConnections: "1500" + persistence: + enabled: true + size: 200Gi + # existingClaim: + service: + port: 5432 + primary: + nodeSelector: {} + affinity: {} + tolerations: [] + readReplicas: + nodeSelector: {} + affinity: {} + tolerations: [] + resources: {} + securityContext: + enabled: true + containerSecurityContext: + enabled: true + # requests: + # memory: "512Mi" + # cpu: "100m" + # limits: + # memory: "1Gi" + # cpu: "500m" +## If NOT using the PostgreSQL in this chart (postgresql.enabled=false), +## specify custom database details here or leave empty and Artifactory will use embedded derby +database: + ## To run Artifactory with any database other than PostgreSQL allowNonPostgresql set to true. + allowNonPostgresql: false + type: + driver: + ## If you set the url, leave host and port empty + url: + ## If you would like this chart to create the secret containing the db + ## password, use these values + user: + password: + ## If you have existing Kubernetes secrets containing db credentials, use + ## these values + secrets: {} + # user: + # name: "rds-artifactory" + # key: "db-user" + # password: + # name: "rds-artifactory" + # key: "db-password" + # url: + # name: "rds-artifactory" + # key: "db-url" +## Filebeat Sidecar container +## The provided filebeat configuration is for Artifactory logs. It assumes you have a logstash installed and configured properly. +filebeat: + enabled: false + name: artifactory-filebeat + image: + repository: "docker.elastic.co/beats/filebeat" + version: 7.16.2 + logstashUrl: "logstash:5044" + livenessProbe: + exec: + command: + - sh + - -c + - | + #!/usr/bin/env bash -e + curl --fail 127.0.0.1:5066 + failureThreshold: 3 + initialDelaySeconds: 10 + periodSeconds: 10 + timeoutSeconds: 5 + readinessProbe: + exec: + command: + - sh + - -c + - | + #!/usr/bin/env bash -e + filebeat test output + failureThreshold: 3 + initialDelaySeconds: 10 + periodSeconds: 10 + timeoutSeconds: 5 + resources: {} + # requests: + # memory: "100Mi" + # cpu: "100m" + # limits: + # memory: "100Mi" + # cpu: "100m" + + filebeatYml: | + logging.level: info + path.data: {{ .Values.artifactory.persistence.mountPath }}/log/filebeat + name: artifactory-filebeat + queue.spool: + file: + permissions: 0760 + filebeat.inputs: + - type: log + enabled: true + close_eof: ${CLOSE:false} + paths: + - {{ .Values.artifactory.persistence.mountPath }}/log/*.log + fields: + service: "jfrt" + log_type: "artifactory" + output: + logstash: + hosts: ["{{ .Values.filebeat.logstashUrl }}"] +## Allows to add additional kubernetes resources +## Use --- as a separator between multiple resources +## For an example, refer - https://github.com/jfrog/log-analytics-prometheus/blob/master/helm/artifactory-values.yaml +additionalResources: "" +## Adding entries to a Pod's /etc/hosts file +## For an example, refer - https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases +hostAliases: [] +# - ip: "127.0.0.1" +# hostnames: +# - "foo.local" +# - "bar.local" +# - ip: "10.1.2.3" +# hostnames: +# - "foo.remote" +# - "bar.remote" + +## Toggling this feature is seamless and requires helm upgrade +## will enable all microservices to run in different containers in a single pod (by default it is true) +splitServicesToContainers: true +## Specify common probes parameters +probes: + timeoutSeconds: 5 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/ci/default-values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/ci/default-values.yaml new file mode 100644 index 0000000000..86355d3b3c --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/ci/default-values.yaml @@ -0,0 +1,7 @@ +# Leave this file empty to ensure that CI runs builds against the default configuration in values.yaml. +artifactory: + databaseUpgradeReady: true + + # To Fix ct tool --reuse-values - PASSWORDS ERROR: you must provide your current passwords when upgrade the release + postgresql: + postgresqlPassword: password diff --git a/charts/jfrog/artifactory-jcr/107.90.14/logo/jcr-logo.png b/charts/jfrog/artifactory-jcr/107.90.14/logo/jcr-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..b1e312e3219a8e0bb1831976646260683f6d447f GIT binary patch literal 77047 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
3t{2R#w6vF)uYze!GvJ=o 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 f`MkUvo-M@p#hFOE>-xWCTUi literal 0 HcmV?d00001 diff --git a/charts/jfrog/artifactory-jcr/107.90.14/questions.yml b/charts/jfrog/artifactory-jcr/107.90.14/questions.yml new file mode 100644 index 0000000000..9cde428709 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/questions.yml @@ -0,0 +1,271 @@ +questions: +# Advance Settings +- variable: artifactory.artifactory.masterKey + default: "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + description: "Artifactory master key. For security reasons, we strongly recommend you generate your own master key using this command: 'openssl rand -hex 32'" + type: string + label: Artifactory master key + group: "Security Settings" + +# Container Images +- variable: defaultImage + default: true + description: "Use default Docker image" + label: Use Default Image + type: boolean + show_subquestion_if: false + group: "Container Images" + subquestions: + - variable: artifactory.artifactory.image.repository + default: "docker.bintray.io/jfrog/artifactory-jcr" + description: "JFrog Container Registry image name" + type: string + label: JFrog Container Registry Image Name + - variable: artifactory.artifactory.image.version + default: "7.6.3" + description: "JFrog Container Registry image tag" + type: string + label: JFrog Container Registry Image Tag + - variable: artifactory.imagePullSecrets + description: "Image Pull Secret" + type: string + label: Image Pull Secret + +# Services and LoadBalancing Settings +- variable: artifactory.ingress.enabled + default: false + description: "Expose app using Layer 7 Load Balancer - ingress" + type: boolean + label: Expose app using Layer 7 Load Balancer + show_subquestion_if: true + group: "Services and Load Balancing" + required: true + subquestions: + - variable: artifactory.ingress.hosts[0] + default: "xip.io" + description: "Hostname to your artifactory installation" + type: hostname + required: true + label: Hostname + +# Nginx Settings +- variable: artifactory.nginx.enabled + default: true + description: "Enable nginx server" + type: boolean + label: Enable Nginx Server + group: "Services and Load Balancing" + required: true + show_if: "artifactory.ingress.enabled=false" +- variable: artifactory.nginx.service.type + default: "LoadBalancer" + description: "Nginx service type" + type: enum + required: true + label: Nginx Service Type + show_if: "artifactory.nginx.enabled=true&&artifactory.ingress.enabled=false" + group: "Services and Load Balancing" + options: + - "ClusterIP" + - "NodePort" + - "LoadBalancer" +- variable: artifactory.nginx.service.loadBalancerIP + default: "" + description: "Provide Static IP to configure with Nginx" + type: string + label: Config Nginx LoadBalancer IP + show_if: "artifactory.nginx.enabled=true&&artifactory.nginx.service.type=LoadBalancer&&artifactory.ingress.enabled=false" + group: "Services and Load Balancing" +- variable: artifactory.nginx.tlsSecretName + default: "" + description: "Provide SSL Secret name to configure with Nginx" + type: string + label: Config Nginx SSL Secret + show_if: "artifactory.nginx.enabled=true&&artifactory.ingress.enabled=false" + group: "Services and Load Balancing" +- variable: artifactory.nginx.customArtifactoryConfigMap + default: "" + description: "Provide configMap name to configure Nginx with custom `artifactory.conf`" + type: string + label: ConfigMap for Nginx Artifactory Config + show_if: "artifactory.nginx.enabled=true&&artifactory.ingress.enabled=false" + group: "Services and Load Balancing" + +# Database Settings +- variable: artifactory.postgresql.enabled + default: true + description: "Enable PostgreSQL" + type: boolean + required: true + label: Enable PostgreSQL + group: "Database Settings" + show_subquestion_if: true + subquestions: + - variable: artifactory.postgresql.postgresqlPassword + default: "" + description: "PostgreSQL password" + type: password + required: true + label: PostgreSQL Password + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=true" + - variable: artifactory.postgresql.persistence.size + default: 20Gi + description: "PostgreSQL persistent volume size" + type: string + label: PostgreSQL Persistent Volume Size + show_if: "artifactory.postgresql.enabled=true" + - variable: artifactory.postgresql.persistence.storageClass + default: "" + description: "If undefined or null, uses the default StorageClass. Default to null" + type: storageclass + label: Default StorageClass for PostgreSQL + show_if: "artifactory.postgresql.enabled=true" + - variable: artifactory.postgresql.resources.requests.cpu + default: "200m" + description: "PostgreSQL initial cpu request" + type: string + label: PostgreSQL Initial CPU Request + show_if: "artifactory.postgresql.enabled=true" + - variable: artifactory.postgresql.resources.requests.memory + default: "500Mi" + description: "PostgreSQL initial memory request" + type: string + label: PostgreSQL Initial Memory Request + show_if: "artifactory.postgresql.enabled=true" + - variable: artifactory.postgresql.resources.limits.cpu + default: "1" + description: "PostgreSQL cpu limit" + type: string + label: PostgreSQL CPU Limit + show_if: "artifactory.postgresql.enabled=true" + - variable: artifactory.postgresql.resources.limits.memory + default: "1Gi" + description: "PostgreSQL memory limit" + type: string + label: PostgreSQL Memory Limit + show_if: "artifactory.postgresql.enabled=true" +- variable: artifactory.database.type + default: "postgresql" + description: "xternal database type (postgresql, mysql, oracle or mssql)" + type: enum + required: true + label: External Database Type + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=false" + options: + - "postgresql" + - "mysql" + - "oracle" + - "mssql" +- variable: artifactory.database.url + default: "" + description: "External database URL. If you set the url, leave host and port empty" + type: string + label: External Database URL + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=false" +- variable: artifactory.database.host + default: "" + description: "External database hostname" + type: string + label: External Database Hostname + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=false" +- variable: artifactory.database.port + default: "" + description: "External database port" + type: string + label: External Database Port + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=false" +- variable: artifactory.database.user + default: "" + description: "External database username" + type: string + label: External Database Username + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=false" +- variable: artifactory.database.password + default: "" + description: "External database password" + type: password + label: External Database Password + group: "Database Settings" + show_if: "artifactory.postgresql.enabled=false" + +# Advance Settings +- variable: artifactory.advancedOptions + default: false + description: "Show advanced configurations" + label: Show Advanced Configurations + type: boolean + show_subquestion_if: true + group: "Advanced Options" + subquestions: + - variable: artifactory.artifactory.primary.resources.requests.cpu + default: "500m" + description: "Artifactory primary node initial cpu request" + type: string + label: Artifactory Primary Node Initial CPU Request + - variable: artifactory.artifactory.primary.resources.requests.memory + default: "1Gi" + description: "Artifactory primary node initial memory request" + type: string + label: Artifactory Primary Node Initial Memory Request + - variable: artifactory.artifactory.primary.javaOpts.xms + default: "1g" + description: "Artifactory primary node java Xms size" + type: string + label: Artifactory Primary Node Java Xms Size + - variable: artifactory.artifactory.primary.resources.limits.cpu + default: "2" + description: "Artifactory primary node cpu limit" + type: string + label: Artifactory Primary Node CPU Limit + - variable: artifactory.artifactory.primary.resources.limits.memory + default: "4Gi" + description: "Artifactory primary node memory limit" + type: string + label: Artifactory Primary Node Memory Limit + - variable: artifactory.artifactory.primary.javaOpts.xmx + default: "4g" + description: "Artifactory primary node java Xmx size" + type: string + label: Artifactory Primary Node Java Xmx Size + - variable: artifactory.artifactory.node.resources.requests.cpu + default: "500m" + description: "Artifactory member node initial cpu request" + type: string + label: Artifactory Member Node Initial CPU Request + - variable: artifactory.artifactory.node.resources.requests.memory + default: "2Gi" + description: "Artifactory member node initial memory request" + type: string + label: Artifactory Member Node Initial Memory Request + - variable: artifactory.artifactory.node.javaOpts.xms + default: "1g" + description: "Artifactory member node java Xms size" + type: string + label: Artifactory Member Node Java Xms Size + - variable: artifactory.artifactory.node.resources.limits.cpu + default: "2" + description: "Artifactory member node cpu limit" + type: string + label: Artifactory Member Node CPU Limit + - variable: artifactory.artifactory.node.resources.limits.memory + default: "4Gi" + description: "Artifactory member node memory limit" + type: string + label: Artifactory Member Node Memory Limit + - variable: artifactory.artifactory.node.javaOpts.xmx + default: "4g" + description: "Artifactory member node java Xmx size" + type: string + label: Artifactory Member Node Java Xmx Size + +# Internal Settings +- variable: installerInfo + default: '\{\"productId\": \"RancherHelm_artifactory-jcr/7.6.3\", \"features\": \[\{\"featureId\": \"Partner/ACC-007246\"\}\]\}' + type: string + group: "Internal Settings (Do not modify)" diff --git a/charts/jfrog/artifactory-jcr/107.90.14/templates/NOTES.txt b/charts/jfrog/artifactory-jcr/107.90.14/templates/NOTES.txt new file mode 100644 index 0000000000..035bf84174 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/templates/NOTES.txt @@ -0,0 +1 @@ +Congratulations. You have just deployed JFrog Container Registry! diff --git a/charts/jfrog/artifactory-jcr/107.90.14/values.yaml b/charts/jfrog/artifactory-jcr/107.90.14/values.yaml new file mode 100644 index 0000000000..a96b4f7d22 --- /dev/null +++ b/charts/jfrog/artifactory-jcr/107.90.14/values.yaml @@ -0,0 +1,75 @@ +# Default values for artifactory-jcr. +# This is a YAML-formatted file. + +# Beware when changing values here. You should know what you are doing! +# Access the values with {{ .Values.key.subkey }} + +# This chart is based on the main artifactory chart with some customizations. +# See all supported configuration keys in https://github.com/jfrog/charts/tree/master/stable/artifactory + +## All values are under the 'artifactory' sub chart. +artifactory: + ## Artifactory + ## See full list of supported Artifactory options and documentation in artifactory chart: https://github.com/jfrog/charts/tree/master/stable/artifactory + artifactory: + ## Default tag is from the artifactory sub-chart in the requirements.yaml + image: + registry: releases-docker.jfrog.io + repository: jfrog/artifactory-jcr + # tag: + ## Uncomment the following resources definitions or pass them from command line + ## to control the cpu and memory resources allocated by the Kubernetes cluster + resources: {} + # requests: + # memory: "1Gi" + # cpu: "500m" + # limits: + # memory: "4Gi" + # cpu: "1" + ## The following Java options are passed to the java process running Artifactory. + ## You should set them according to the resources set above. + ## IMPORTANT: Make sure resources.limits.memory is at least 1G more than Xmx. + javaOpts: {} + # xms: "1g" + # xmx: "3g" + # other: "" + installer: + platform: jcr-helm + installerInfo: '{"productId":"Helm_artifactory-jcr/{{ .Chart.Version }}","features":[{"featureId":"Platform/{{ printf "%s-%s" "kubernetes" .Capabilities.KubeVersion.Version }}"},{"featureId":"Database/{{ .Values.database.type }}"},{"featureId":"PostgreSQL_Enabled/{{ .Values.postgresql.enabled }}"},{"featureId":"Nginx_Enabled/{{ .Values.nginx.enabled }}"},{"featureId":"ArtifactoryPersistence_Type/{{ .Values.artifactory.persistence.type }}"},{"featureId":"SplitServicesToContainers_Enabled/{{ .Values.splitServicesToContainers }}"},{"featureId":"UnifiedSecretInstallation_Enabled/{{ .Values.artifactory.unifiedSecretInstallation }}"},{"featureId":"Filebeat_Enabled/{{ .Values.filebeat.enabled }}"},{"featureId":"ReplicaCount/{{ .Values.artifactory.replicaCount }}"}]}' + ## Nginx + ## See full list of supported Nginx options and documentation in artifactory chart: https://github.com/jfrog/charts/tree/master/stable/artifactory + nginx: + enabled: true + tlsSecretName: "" + service: + type: LoadBalancer + ## Ingress + ## See full list of supported Ingress options and documentation in artifactory chart: https://github.com/jfrog/charts/tree/master/stable/artifactory + ingress: + enabled: false + tls: + ## PostgreSQL + ## See list of supported postgresql options and documentation in artifactory chart: https://github.com/jfrog/charts/tree/master/stable/artifactory + ## Configuration values for the PostgreSQL dependency sub-chart + ## ref: https://github.com/bitnami/charts/blob/master/bitnami/postgresql/README.md + postgresql: + enabled: true + ## This key is required for upgrades to protect old PostgreSQL chart's breaking changes. + databaseUpgradeReady: "yes" + ## If NOT using the PostgreSQL in this chart (artifactory.postgresql.enabled=false), + ## specify custom database details here or leave empty and Artifactory will use embedded derby. + ## See full list of database options and documentation in artifactory chart: https://github.com/jfrog/charts/tree/master/stable/artifactory + # database: + jfconnect: + enabled: false + federation: + enabled: false +## Enable the PostgreSQL sub chart +postgresql: + enabled: true +router: + image: + tag: 7.118.3 +initContainers: + image: + tag: 9.4.949.1716471857 diff --git a/charts/kasten/k10/7.0.1101/Chart.lock b/charts/kasten/k10/7.0.1101/Chart.lock new file mode 100644 index 0000000000..3e85f0fa3f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/Chart.lock @@ -0,0 +1,9 @@ +dependencies: +- name: grafana + repository: "" + version: 8.5.0 +- name: prometheus + repository: "" + version: 25.24.1 +digest: sha256:916a3d0c3da91cf0344b61267d86bf7f95531cae2582fe57f1f28f7f04db67f7 +generated: "2024-10-08T19:50:05.986249602Z" diff --git a/charts/kasten/k10/7.0.1101/Chart.yaml b/charts/kasten/k10/7.0.1101/Chart.yaml new file mode 100644 index 0000000000..f1f6c8c9bd --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.11 +dependencies: +- condition: grafana.enabled + name: grafana + repository: "" + version: 8.5.0 +- condition: prometheus.server.enabled + name: prometheus + repository: "" + version: 25.24.1 +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.1101 diff --git a/charts/kasten/k10/7.0.1101/README.md b/charts/kasten/k10/7.0.1101/README.md new file mode 100644 index 0000000000..62659cf7e7 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/README.md @@ -0,0 +1,342 @@ +# 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.executor.workerCount` | Specifies count of running executor workers | 8 +`services.executor.maxConcurrentRestoreCsiSnapshots` | Limit of concurrent restore CSI snapshots operations per each restore action | 3 +`services.executor.maxConcurrentRestoreGenericVolumeSnapshots` | Limit of concurrent restore generic volume snapshots operations per each restore action | 3 +`services.executor.maxConcurrentRestoreWorkloads` | Limit of concurrent restore workloads operations per each restore action | 3 +`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]` | Resource requests and limits for Generic Volume Snapshot restore pods | `{}` +`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.concurrentSnapConversions` | Limit of concurrent snapshots to convert during export | `3` +`limiter.genericVolumeSnapshots` | Limit of concurrent generic volume snapshot create operations | `10` +`limiter.genericVolumeCopies` | Limit of concurrent generic volume snapshot copy operations | `10` +`limiter.genericVolumeRestores` | Limit of concurrent generic volume snapshot restore operations | `10` +`limiter.csiSnapshots` | Limit of concurrent CSI snapshot create operations | `10` +`limiter.providerSnapshots` | Limit of concurrent cloud provider create operations | `10` +`limiter.imageCopies` | Limit of concurrent image copy operations | `10` +`cluster.domainName` | Specifies the domain name of the cluster | `""` +`kanister.backupTimeout` | Specifies timeout to set on Kanister backup operations | `45` +`kanister.restoreTimeout` | Specifies timeout to set on Kanister restore operations | `600` +`kanister.deleteTimeout` | Specifies timeout to set on Kanister delete operations | `45` +`kanister.hookTimeout` | Specifies timeout to set on Kanister pre-hook and post-hook operations | `20` +`kanister.checkRepoTimeout` | Specifies timeout to set on Kanister checkRepo operations | `20` +`kanister.statsTimeout` | Specifies timeout to set on Kanister stats operations | `20` +`kanister.efsPostRestoreTimeout` | Specifies timeout to set on Kanister efsPostRestore operations | `45` +`kanister.podReadyWaitTimeout` | Specifies a timeout to wait for Kanister pods to reach the ready state during K10 operations | `15` +`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"]` +`kanisterPodMetricSidecar.enabled` | Enable the sidecar container to gather metrics from ephemeral pods | `true` +`kanisterPodMetricSidecar.metricLifetime` | Check periodically for metrics that should be removed | `2m` +`kanisterPodMetricSidecar.pushGatewayInterval` | Set the interval for sending metrics into the Prometheus | `30s` +`kanisterPodMetricSidecar.resources.[requests\|limits].[cpu\|memory]` | Resource requests and limits for the Kanister pod metric sidecar | `{}` +`maxJobWaitDuration` | Set a maximum duration of waiting for child jobs. If the execution of the subordinate jobs exceeds this value, the parent job will be canceled. If no value is set, a default of 10 hours will be used | `None` +`forceRootInKanisterHooks` | Forces Kanister Execution Hooks to run with root privileges | `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.1101/app-readme.md b/charts/kasten/k10/7.0.1101/app-readme.md new file mode 100644 index 0000000000..1b221891be --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/.helmignore b/charts/kasten/k10/7.0.1101/charts/grafana/.helmignore new file mode 100644 index 0000000000..8cade1318f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/Chart.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/Chart.yaml new file mode 100644 index 0000000000..e334ca5cb5 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1.5 +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.0 diff --git a/charts/kasten/k10/7.0.1101/charts/grafana/README.md b/charts/kasten/k10/7.0.1101/charts/grafana/README.md new file mode 100644 index 0000000000..2c9609e122 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.26.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.1101/charts/grafana/ci/default-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/default-values.yaml new file mode 100644 index 0000000000..fc2ba605ad --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-affinity-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-affinity-values.yaml new file mode 100644 index 0000000000..f5b9b53e73 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-dashboard-json-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-dashboard-json-values.yaml new file mode 100644 index 0000000000..e0c4e41687 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-dashboard-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-dashboard-values.yaml new file mode 100644 index 0000000000..7b662c5fd4 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-extraconfigmapmounts-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-extraconfigmapmounts-values.yaml new file mode 100644 index 0000000000..5cc44a056a --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-image-renderer-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-image-renderer-values.yaml new file mode 100644 index 0000000000..06c0bda13e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-nondefault-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-nondefault-values.yaml new file mode 100644 index 0000000000..fb5c179409 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/ci/with-persistence.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-persistence.yaml new file mode 100644 index 0000000000..b92ca02c9e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-persistence.yaml @@ -0,0 +1,3 @@ +persistence: + type: pvc + enabled: true diff --git a/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/ci/with-sidecars-envvaluefrom-values.yaml new file mode 100644 index 0000000000..a6935e56d0 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/dashboards/custom-dashboard.json b/charts/kasten/k10/7.0.1101/charts/grafana/dashboards/custom-dashboard.json new file mode 100644 index 0000000000..9e26dfeeb6 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/grafana/dashboards/custom-dashboard.json @@ -0,0 +1 @@ +{} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1101/charts/grafana/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/charts/grafana/templates/NOTES.txt new file mode 100644 index 0000000000..a40f666a47 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/_config.tpl b/charts/kasten/k10/7.0.1101/charts/grafana/templates/_config.tpl new file mode 100644 index 0000000000..b866217f2e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/grafana/templates/_config.tpl @@ -0,0 +1,172 @@ +{{/* + 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 "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 "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.1101/charts/grafana/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/charts/grafana/templates/_helpers.tpl new file mode 100644 index 0000000000..2a68cb6f85 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/_pod.tpl b/charts/kasten/k10/7.0.1101/charts/grafana/templates/_pod.tpl new file mode 100644 index 0000000000..8b33db75ba --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/grafana/templates/_pod.tpl @@ -0,0 +1,1320 @@ +{{- 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" +{{- 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" +{{- 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" +{{- 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.1101/charts/grafana/templates/clusterrole.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/clusterrole.yaml new file mode 100644 index 0000000000..3af4b62b63 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..bda9431a2c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/configSecret.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/configSecret.yaml new file mode 100644 index 0000000000..55574b9bbc --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/configmap-dashboard-provider.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/configmap-dashboard-provider.yaml new file mode 100644 index 0000000000..b412c4d1f0 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/configmap.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/configmap.yaml new file mode 100644 index 0000000000..0a2edf47e3 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/dashboards-json-configmap.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/dashboards-json-configmap.yaml new file mode 100644 index 0000000000..df0ed0d8c5 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/deployment.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/deployment.yaml new file mode 100644 index 0000000000..c981362ad4 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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 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 | 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.1101/charts/grafana/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/extra-manifests.yaml new file mode 100644 index 0000000000..a9bb3b6ba8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/grafana/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraObjects }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1101/charts/grafana/templates/headless-service.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/headless-service.yaml new file mode 100644 index 0000000000..3028589d32 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/hpa.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/hpa.yaml new file mode 100644 index 0000000000..46bbcb49a2 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/image-renderer-deployment.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-deployment.yaml new file mode 100644 index 0000000000..7722ede50d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/image-renderer-hpa.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-hpa.yaml new file mode 100644 index 0000000000..b0f0059b79 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/image-renderer-network-policy.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-network-policy.yaml new file mode 100644 index 0000000000..bcbd24976c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/image-renderer-service.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-service.yaml new file mode 100644 index 0000000000..f8da127cf8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/image-renderer-servicemonitor.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/image-renderer-servicemonitor.yaml new file mode 100644 index 0000000000..5d9f09d266 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/ingress.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/ingress.yaml new file mode 100644 index 0000000000..b2ffd81095 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/networkpolicy.yaml new file mode 100644 index 0000000000..4cd3ed6976 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/poddisruptionbudget.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/poddisruptionbudget.yaml new file mode 100644 index 0000000000..05251214ac --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/podsecuritypolicy.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..eed7af95b5 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/pvc.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/pvc.yaml new file mode 100644 index 0000000000..d1c4b2de27 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/role.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/role.yaml new file mode 100644 index 0000000000..4b5edd978c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/rolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/rolebinding.yaml new file mode 100644 index 0000000000..58f77c6b0b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/secret-env.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/secret-env.yaml new file mode 100644 index 0000000000..eb14aac707 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/secret.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/secret.yaml new file mode 100644 index 0000000000..fd2ca50f4b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/service.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/service.yaml new file mode 100644 index 0000000000..022328c114 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/serviceaccount.yaml new file mode 100644 index 0000000000..ffca0717ae --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/servicemonitor.yaml new file mode 100644 index 0000000000..0359013520 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/statefulset.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/statefulset.yaml new file mode 100644 index 0000000000..7546c18876 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/tests/test-configmap.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-configmap.yaml new file mode 100644 index 0000000000..1e81bee90b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/tests/test-podsecuritypolicy.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-podsecuritypolicy.yaml new file mode 100644 index 0000000000..c13a3bf668 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/tests/test-role.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-role.yaml new file mode 100644 index 0000000000..75dddfdd3d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/tests/test-rolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-rolebinding.yaml new file mode 100644 index 0000000000..c0d2d39efa --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/tests/test-serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test-serviceaccount.yaml new file mode 100644 index 0000000000..7af8982723 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/templates/tests/test.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/templates/tests/test.yaml new file mode 100644 index 0000000000..2484a96da0 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/grafana/values.yaml b/charts/kasten/k10/7.0.1101/charts/grafana/values.yaml new file mode 100644 index 0000000000..51e94e01ff --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/grafana/values.yaml @@ -0,0 +1,1386 @@ +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.27.4 + 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 alert 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 dashboard 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 + # 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 + # 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 + # 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.1101/charts/prometheus/.helmignore b/charts/kasten/k10/7.0.1101/charts/prometheus/.helmignore new file mode 100644 index 0000000000..825c007791 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/Chart.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/Chart.yaml new file mode 100644 index 0000000000..ca07363f90 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.53.1 +dependencies: +- condition: alertmanager.enabled + name: alertmanager + repository: https://prometheus-community.github.io/helm-charts + version: 1.11.* +- condition: kube-state-metrics.enabled + name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 5.21.* +- condition: prometheus-node-exporter.enabled + name: prometheus-node-exporter + repository: https://prometheus-community.github.io/helm-charts + version: 4.37.* +- condition: prometheus-pushgateway.enabled + name: prometheus-pushgateway + repository: https://prometheus-community.github.io/helm-charts + version: 2.14.* +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.24.1 diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/OWNERS b/charts/kasten/k10/7.0.1101/charts/prometheus/OWNERS new file mode 100644 index 0000000000..0cfd95021c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/OWNERS @@ -0,0 +1,6 @@ +approvers: +- mgoodness +- gianrubio +reviewers: +- mgoodness +- gianrubio diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/README.md b/charts/kasten/k10/7.0.1101/charts/prometheus/README.md new file mode 100644 index 0000000000..2cb744ce8f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/.helmignore b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/.helmignore new file mode 100644 index 0000000000..7653e97e66 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/Chart.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/Chart.yaml new file mode 100644 index 0000000000..dbdb2ae477 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.11.0 diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/README.md b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/README.md new file mode 100644 index 0000000000..d3f4df73a2 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml new file mode 100644 index 0000000000..cba5de8e29 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/ci/config-reload-values.yaml @@ -0,0 +1,2 @@ +configmapReload: + enabled: true diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/NOTES.txt new file mode 100644 index 0000000000..46ea5bee59 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/_helpers.tpl new file mode 100644 index 0000000000..827b6ee9f7 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/configmap.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/configmap.yaml new file mode 100644 index 0000000000..9e5882dc86 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/ingress.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/ingress.yaml new file mode 100644 index 0000000000..e729a8ad3b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/ingressperreplica.yaml new file mode 100644 index 0000000000..6f5a023500 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/pdb.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/pdb.yaml new file mode 100644 index 0000000000..103e9ecde1 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/serviceaccount.yaml new file mode 100644 index 0000000000..bc9ccaaff9 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/serviceperreplica.yaml new file mode 100644 index 0000000000..faa75b3ba2 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/services.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/services.yaml new file mode 100644 index 0000000000..eefb9ce161 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/templates/statefulset.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/statefulset.yaml new file mode 100644 index 0000000000..2bdafc86db --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/statefulset.yaml @@ -0,0 +1,251 @@ +{{- $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.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.1101/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/templates/tests/test-connection.yaml new file mode 100644 index 0000000000..410eba5bd8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/alertmanager/values.schema.json b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.schema.json new file mode 100644 index 0000000000..48c6e9a631 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.schema.json @@ -0,0 +1,923 @@ +{ + "$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" + } + } + } + } +} diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.yaml new file mode 100644 index 0000000000..f8a9d243b3 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/alertmanager/values.yaml @@ -0,0 +1,379 @@ +# 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" diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/.helmignore b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/Chart.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/Chart.yaml new file mode 100644 index 0000000000..9dea659498 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.12.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.21.0 diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/README.md b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/README.md new file mode 100644 index 0000000000..843be89e69 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/NOTES.txt new file mode 100644 index 0000000000..3589c24ec3 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/_helpers.tpl new file mode 100644 index 0000000000..a4358c87a1 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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 }} +{{ 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.1101/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml new file mode 100644 index 0000000000..025cd47a88 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..cf9f628d04 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/crs-configmap.yaml new file mode 100644 index 0000000000..d38a75a51d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml new file mode 100644 index 0000000000..5d32f82fa7 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/deployment.yaml @@ -0,0 +1,313 @@ +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 }} + {{- if .Values.podAnnotations }} + annotations: +{{ toYaml .Values.podAnnotations | indent 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 }} + 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: /healthz + 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: / + 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 }} + {{- if .Values.nodeSelector }} + nodeSelector: +{{ toYaml .Values.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: +{{ toYaml .Values.tolerations | 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.1101/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/extra-manifests.yaml new file mode 100644 index 0000000000..567f7bf329 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/kubeconfig-secret.yaml new file mode 100644 index 0000000000..6af0084502 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/networkpolicy.yaml new file mode 100644 index 0000000000..309b38ec54 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/pdb.yaml new file mode 100644 index 0000000000..3771b511de --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..8905e113e8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrole.yaml new file mode 100644 index 0000000000..654e4a3d57 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml new file mode 100644 index 0000000000..5b62a18bdf --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/rbac-configmap.yaml new file mode 100644 index 0000000000..671dc9d660 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/role.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/role.yaml new file mode 100644 index 0000000000..d33687f2d1 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/rolebinding.yaml new file mode 100644 index 0000000000..330651b73f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/service.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/service.yaml new file mode 100644 index 0000000000..90c235148f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/serviceaccount.yaml new file mode 100644 index 0000000000..c302bc7ca0 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/servicemonitor.yaml new file mode 100644 index 0000000000..99d7fa9246 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-role.yaml new file mode 100644 index 0000000000..489de147c1 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml new file mode 100644 index 0000000000..73b37a4f64 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml new file mode 100644 index 0000000000..f46305b517 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/kube-state-metrics/values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/values.yaml new file mode 100644 index 0000000000..2a9781138a --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/kube-state-metrics/values.yaml @@ -0,0 +1,522 @@ +# 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: {} + +## 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 + +## 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.1101/charts/prometheus/charts/prometheus-node-exporter/.helmignore b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/Chart.yaml new file mode 100644 index 0000000000..5af1f445e8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1 +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.37.0 diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/README.md b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/README.md new file mode 100644 index 0000000000..ef83844102 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/ci/port-values.yaml new file mode 100644 index 0000000000..dbfb4b67ff --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/NOTES.txt new file mode 100644 index 0000000000..db8584def8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/_helpers.tpl new file mode 100644 index 0000000000..8e84832cbf --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.podLabels }} +{{ 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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrole.yaml new file mode 100644 index 0000000000..c256dba73d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..653305ad9e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml new file mode 100644 index 0000000000..23896a230a --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/daemonset.yaml @@ -0,0 +1,311 @@ +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 }} + 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 }} + {{- 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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/endpoints.yaml new file mode 100644 index 0000000000..45eeb8d966 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/extra-manifests.yaml new file mode 100644 index 0000000000..2b21b71062 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml new file mode 100644 index 0000000000..825722729d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/networkpolicy.yaml @@ -0,0 +1,23 @@ +{{- 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: + - ports: + - port: {{ .Values.service.port }} + policyTypes: + - Egress + - Ingress + podSelector: + matchLabels: + {{- include "prometheus-node-exporter.selectorLabels" . | nindent 6 }} +{{- end }} diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/podmonitor.yaml new file mode 100644 index 0000000000..f88da6a34e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrole.yaml new file mode 100644 index 0000000000..8957317243 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp-clusterrolebinding.yaml new file mode 100644 index 0000000000..333370173b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/psp.yaml new file mode 100644 index 0000000000..4896c84daa --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/rbac-configmap.yaml new file mode 100644 index 0000000000..814e110337 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml new file mode 100644 index 0000000000..8308b7b2b3 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/service.yaml @@ -0,0 +1,35 @@ +{{- 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.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.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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/serviceaccount.yaml new file mode 100644 index 0000000000..462b0cda4b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/servicemonitor.yaml new file mode 100644 index 0000000000..0d7a42eaec --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/templates/verticalpodautoscaler.yaml new file mode 100644 index 0000000000..2c2705f872 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-node-exporter/values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/values.yaml new file mode 100644 index 0000000000..1323567359 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-node-exporter/values.yaml @@ -0,0 +1,533 @@ +# 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: + enabled: true + type: ClusterIP + clusterIP: "" + port: 9100 + targetPort: 9100 + nodePort: + portName: metrics + listenOnAllInterfaces: true + annotations: + prometheus.io/scrape: "true" + ipDualStack: + enabled: false + ipFamilies: ["IPv6", "IPv4"] + ipFamilyPolicy: "PreferDualStack" + externalTrafficPolicy: "" + +# Set a NetworkPolicy with: +# ingress only on service.port +# no egress permitted +networkPolicy: + enabled: false + +# 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 + +# 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 be added to node exporter pods +podLabels: {} + +# 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.1101/charts/prometheus/charts/prometheus-pushgateway/.helmignore b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/.helmignore new file mode 100644 index 0000000000..e90c9f6d23 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/Chart.yaml new file mode 100644 index 0000000000..3d4e84621a --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.9.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.14.0 diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/README.md b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/README.md new file mode 100644 index 0000000000..cc6645fdf9 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/NOTES.txt new file mode 100644 index 0000000000..263b1d8d49 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/_helpers.tpl new file mode 100644 index 0000000000..dcd42ff36b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/deployment.yaml new file mode 100644 index 0000000000..5d3fafde7d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/extra-manifests.yaml new file mode 100644 index 0000000000..bafee95185 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/ingress.yaml new file mode 100644 index 0000000000..237ac4a121 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/networkpolicy.yaml new file mode 100644 index 0000000000..d3b8019e3f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/pdb.yaml new file mode 100644 index 0000000000..6051133c68 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/pushgateway-pvc.yaml new file mode 100644 index 0000000000..d2a85f4242 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/secret.yaml new file mode 100644 index 0000000000..a8142d1389 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/service.yaml new file mode 100644 index 0000000000..15029f7e30 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/serviceaccount.yaml new file mode 100644 index 0000000000..88f1470480 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/servicemonitor.yaml new file mode 100644 index 0000000000..ae173199b3 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/templates/statefulset.yaml new file mode 100644 index 0000000000..8d486a306f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/charts/prometheus-pushgateway/values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/charts/prometheus-pushgateway/values.yaml new file mode 100644 index 0000000000..85f267fdb8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/NOTES.txt new file mode 100644 index 0000000000..fc03c2a5b6 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/_helpers.tpl new file mode 100644 index 0000000000..3d8078f022 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/clusterrole.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/clusterrole.yaml new file mode 100644 index 0000000000..25e3cec45d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/clusterrolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..28f4bda776 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/cm.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/cm.yaml new file mode 100644 index 0000000000..8713bd1ea2 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/deploy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/deploy.yaml new file mode 100644 index 0000000000..9c786ee425 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/extra-manifests.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/extra-manifests.yaml new file mode 100644 index 0000000000..2b21b71062 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraManifests }} +--- +{{ tpl . $ }} +{{ end }} diff --git a/charts/kasten/k10/7.0.1101/charts/prometheus/templates/headless-svc.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/headless-svc.yaml new file mode 100644 index 0000000000..df9db99147 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/ingress.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/ingress.yaml new file mode 100644 index 0000000000..84341a9c2c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/network-policy.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/network-policy.yaml new file mode 100644 index 0000000000..3254ffc04f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/pdb.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/pdb.yaml new file mode 100644 index 0000000000..7ffe673071 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/psp.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/psp.yaml new file mode 100644 index 0000000000..5776e25410 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/pvc.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/pvc.yaml new file mode 100644 index 0000000000..a9dc4fce08 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/rolebinding.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/rolebinding.yaml new file mode 100644 index 0000000000..721b38816a --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/service.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/service.yaml new file mode 100644 index 0000000000..069f3270d8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/serviceaccount.yaml new file mode 100644 index 0000000000..6d5ab0c7d9 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/sts.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/sts.yaml new file mode 100644 index 0000000000..6200555df1 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/templates/vpa.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/templates/vpa.yaml new file mode 100644 index 0000000000..cd07ad8b7d --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/values.schema.json b/charts/kasten/k10/7.0.1101/charts/prometheus/values.schema.json new file mode 100644 index 0000000000..b2d8af26c3 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/charts/prometheus/values.yaml b/charts/kasten/k10/7.0.1101/charts/prometheus/values.yaml new file mode 100644 index 0000000000..d940566662 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/charts/prometheus/values.yaml @@ -0,0 +1,1309 @@ +# 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.74.0 + # 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: {} + +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: {} + # 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.1101/config.json b/charts/kasten/k10/7.0.1101/config.json new file mode 100644 index 0000000000..e69de29bb2 diff --git a/charts/kasten/k10/7.0.1101/eula.txt b/charts/kasten/k10/7.0.1101/eula.txt new file mode 100644 index 0000000000..19f9fc0763 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/files/favicon.png b/charts/kasten/k10/7.0.1101/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.1101/files/kasten-logo.svg b/charts/kasten/k10/7.0.1101/files/kasten-logo.svg new file mode 100644 index 0000000000..0d0ef14eee --- /dev/null +++ b/charts/kasten/k10/7.0.1101/files/kasten-logo.svg @@ -0,0 +1,24 @@ + + + + + + diff --git a/charts/kasten/k10/7.0.1101/files/styles.css b/charts/kasten/k10/7.0.1101/files/styles.css new file mode 100644 index 0000000000..2d92057119 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/grafana/dashboards/default/default.json b/charts/kasten/k10/7.0.1101/grafana/dashboards/default/default.json new file mode 100644 index 0000000000..163461a0ff --- /dev/null +++ b/charts/kasten/k10/7.0.1101/grafana/dashboards/default/default.json @@ -0,0 +1,5855 @@ +{ + "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" + } + ], + "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": 2 +} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1101/license b/charts/kasten/k10/7.0.1101/license new file mode 100644 index 0000000000..fb23dbb826 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/license @@ -0,0 +1 @@ +Y3VzdG9tZXJOYW1lOiBzdGFydGVyLWxpY2Vuc2UKZGF0ZUVuZDogJzIxMDAtMDEtMDFUMDA6MDA6MDAuMDAwWicKZGF0ZVN0YXJ0OiAnMjAyMC0wMS0wMVQwMDowMDowMC4wMDBaJwpmZWF0dXJlczogbnVsbAppZDogc3RhcnRlci00ZjE4NDJjMC0wNzQ1LTQxYTUtYWFhNy1hMDFkNzQ4YjFjMzAKcHJvZHVjdDogSzEwCnJlc3RyaWN0aW9uczoKICBub2RlczogJzEwJwpzZXJ2aWNlQWNjb3VudEtleTogbnVsbAp2ZXJzaW9uOiB2MS4wLjAKc2lnbmF0dXJlOiBqT1N5NDNQZG5ZMFVCZitValhOdU1oUEFSb1J2ZkpzWElQWnhBWFNCaGpKbUwxNlNodi8vVzgyV2NMeGZJM25NZTA0TThtRU03eThPcnArQks1ekxpeFd3clpncmZSbTBEaWlELyttRjR5U3l1Rko0QW1neHV6NDhQTmdnU1VyWUM3S1FVcFYxSEJZV1ZaNm9udEJDeE1rVWtkaDVqdzZJdWMzN3lDaktIYy92bWZaenBzTVhybmxUdGhha2RjVVk0azNyVHJDa3VDcnFUMkpjM1o1amFGalZSZW1Zd1NBVXpkRldNazdQdkp3eHVFdE5rNitPV0pCVERQbnNYdldKdjdNc3NneDBJTmdtNUlJWDRVeEVhQWI4QXpTNkMyQ21XQzlhWURFTDg1aEFpeWhONXUwU0tQczA3ZXB0R1VHYmc3cWtPUVN0d0NhcDFKUURvbDVDT0E9PQo= diff --git a/charts/kasten/k10/7.0.1101/questions.yaml b/charts/kasten/k10/7.0.1101/questions.yaml new file mode 100644 index 0000000000..713fcb1162 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/NOTES.txt b/charts/kasten/k10/7.0.1101/templates/NOTES.txt new file mode 100644 index 0000000000..7c2354ce02 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/NOTES.txt @@ -0,0 +1,106 @@ +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 }} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1101/templates/_definitions.tpl b/charts/kasten/k10/7.0.1101/templates/_definitions.tpl new file mode 100644 index 0000000000..4e26d6df4f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/_definitions.tpl @@ -0,0 +1,238 @@ +{{/* 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" -}}admin 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" -}}admin aggregatedapis auth bloblifecyclemanager controllermanager crypto dashboardbff events executor garbagecollector repositories gateway state vbrintegrationapi{{- end -}} +{{- define "k10.colocatedServices" -}} +admin: + port: 8001 + primary: state +bloblifecyclemanager: + port: 8001 + primary: crypto +events: + port: 8002 + primary: state +garbagecollector: + port: 8002 + primary: crypto +repositories: + port: 8003 + primary: crypto +vbrintegrationapi: + port: 8001 + primary: dashboardbff +{{- end -}} +{{- define "k10.colocatedServiceLookup" -}} +crypto: +- garbagecollector +- repositories +- bloblifecyclemanager +dashboardbff: +- vbrintegrationapi +state: +- admin +- events +{{- end -}} +{{- define "k10.aggregatedAPIs" -}}actions apps repositories vault{{- 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.defaultConcurrentSnapshotConversions" -}}3{{- end -}} +{{- define "k10.defaultConcurrentWorkloadSnapshots" -}}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.defaultK10LimiterGenericVolumeSnapshots" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterGenericVolumeCopies" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterGenericVolumeRestores" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterCsiSnapshots" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterImageCopies" -}}10{{- end -}} +{{- define "k10.defaultK10LimiterProviderSnapshots" -}}10{{- end -}} +{{- define "k10.defaultK10GCDaemonPeriod" -}}21600{{- end -}} +{{- define "k10.defaultK10GCKeepMaxActions" -}}1000{{- end -}} +{{- define "k10.defaultK10GCActionsEnabled" -}}false{{- end -}} +{{- define "k10.defaultK10ExecutorWorkerCount" -}}8{{- end -}} +{{- define "k10.defaultK10ExecutorMaxConcurrentRestoreCsiSnapshots" -}}3{{- end -}} +{{- define "k10.defaultK10ExecutorMaxConcurrentRestoreGenericVolumeSnapshots" -}}3{{- end -}} +{{- define "k10.defaultK10ExecutorMaxConcurrentRestoreWorkloads" -}}3{{- end -}} +{{- define "k10.defaultAssumeRoleDuration" -}}60m{{- end -}} +{{- define "k10.defaultKanisterBackupTimeout" -}}45{{- end -}} +{{- define "k10.defaultKanisterRestoreTimeout" -}}600{{- end -}} +{{- define "k10.defaultKanisterDeleteTimeout" -}}45{{- end -}} +{{- define "k10.defaultKanisterHookTimeout" -}}20{{- end -}} +{{- define "k10.defaultKanisterCheckRepoTimeout" -}}20{{- end -}} +{{- define "k10.defaultKanisterStatsTimeout" -}}20{{- end -}} +{{- define "k10.defaultKanisterEFSPostRestoreTimeout" -}}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 +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 + events-svc: + requests: + cpu: 3m + memory: 500Mi + garbagecollector-svc: + requests: + cpu: 3m + memory: 100Mi +dashboardbff-svc: + dashboardbff-svc: + requests: + cpu: 8m + memory: 40Mi + repositories-svc: + requests: + cpu: 10m + memory: 40Mi +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 +kanister-svc: + kanister-svc: + requests: + cpu: 1m + memory: 30Mi +logging-svc: + logging-svc: + requests: + cpu: 2m + memory: 40Mi +metering-svc: + metering-svc: + requests: + cpu: 2m + memory: 30Mi +state-svc: + admin-svc: + requests: + cpu: 2m + memory: 160Mi + 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.111.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.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.1101/templates/_grafana.tpl b/charts/kasten/k10/7.0.1101/templates/_grafana.tpl new file mode 100644 index 0000000000..53ade64a0c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/_helpers.tpl b/charts/kasten/k10/7.0.1101/templates/_helpers.tpl new file mode 100644 index 0000000000..5be5dd5f70 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/_helpers.tpl @@ -0,0 +1,1524 @@ +{{/* Returns a string of the disabled K10 services */}} +{{- define "get.disabledServices" -}} + {{/* Append services to this list based on helm values */}} + {{- $disabledServices := list -}} + + {{- if .Values.reporting -}} + {{- if eq .Values.reporting.pdfReports false -}} + {{- $disabledServices = append $disabledServices "admin" -}} + {{- end -}} + {{- end -}} + + {{- 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 -}} + +{{/* 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 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 -}} + +{{/* +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.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 "kanisterPodMetricSidecarResources" }} +{{- if .Values.kanisterPodMetricSidecar.resources.requests.memory }} +KanisterPodMetricSidecarMemoryRequest: {{ .Values.kanisterPodMetricSidecar.resources.requests.memory | quote }} +{{- end }} +{{- if .Values.kanisterPodMetricSidecar.resources.requests.cpu }} +KanisterPodMetricSidecarCPURequest: {{ .Values.kanisterPodMetricSidecar.resources.requests.cpu | quote }} +{{- end }} +{{- if .Values.kanisterPodMetricSidecar.resources.limits.memory }} +KanisterPodMetricSidecarMemoryLimit: {{ .Values.kanisterPodMetricSidecar.resources.limits.memory | quote }} +{{- end }} +{{- if .Values.kanisterPodMetricSidecar.resources.limits.cpu }} +KanisterPodMetricSidecarCPULimit: {{ .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 the admin image is turned on */}} +{{- define "k10.fail.ironbankPdfReports" -}} + {{- if and (include "ironbank.enabled" .) (.Values.reporting.pdfReports) -}} + {{- fail "global.ironbank.enabled and reporting.pdfReports 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.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 -}} + +{{/* Fail if FIPS is enabled and PDF reporting is turned on */}} +{{- define "k10.fail.fipsPDFReports" -}} + {{- if and (.Values.fips.enabled) (.Values.reporting.pdfReports) -}} + {{- fail "fips.enabled and reporting.pdfReports 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 -}} + diff --git a/charts/kasten/k10/7.0.1101/templates/_k10_container.tpl b/charts/kasten/k10/7.0.1101/templates/_k10_container.tpl new file mode 100644 index 0000000000..6e2434912c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/_k10_container.tpl @@ -0,0 +1,1121 @@ +{{- 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 or (eq (include "check.azureMSIWithClientID" .) "true") (eq (include "check.azureClientSecretCreds" .) "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 }} +{{- 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" . }} +{{- 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 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: CONCURRENT_SNAP_CONVERSIONS + valueFrom: + configMapKeyRef: + name: k10-config + key: concurrentSnapConversions + - name: CONCURRENT_WORKLOAD_SNAPSHOTS + valueFrom: + configMapKeyRef: + name: k10-config + key: concurrentWorkloadSnapshots + - 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_SNAPSHOTS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterGenericVolumeSnapshots + - name: K10_LIMITER_GENERIC_VOLUME_COPIES + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterGenericVolumeCopies + - name: K10_LIMITER_GENERIC_VOLUME_RESTORES + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterGenericVolumeRestores + - name: K10_LIMITER_CSI_SNAPSHOTS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterCsiSnapshots + - name: K10_LIMITER_PROVIDER_SNAPSHOTS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterProviderSnapshots + - name: K10_LIMITER_IMAGE_COPIES + valueFrom: + configMapKeyRef: + name: k10-config + key: K10LimiterImageCopies + - 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: KANISTER_POD_READY_WAIT_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodReadyWaitTimeout + + - name: K10_KANISTER_POD_METRICS_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarEnabled + - name: PUSHGATEWAY_METRICS_INTERVAL + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodPushgatewayMetricsInterval + {{- if .Values.kanisterPodMetricSidecar.resources.requests.memory }} + - name: K10_KANISTER_POD_METRIC_SIDECAR_MEMORY_REQUEST + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarMemoryRequest + {{- end }} + {{- if .Values.kanisterPodMetricSidecar.resources.requests.cpu }} + - name: K10_KANISTER_POD_METRIC_SIDECAR_CPU_REQUEST + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarCPURequest + {{- end }} + {{- if .Values.kanisterPodMetricSidecar.resources.limits.memory }} + - name: K10_KANISTER_POD_METRIC_SIDECAR_MEMORY_LIMIT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarMemoryLimit + {{- end }} + {{- if .Values.kanisterPodMetricSidecar.resources.limits.cpu }} + - name: K10_KANISTER_POD_METRIC_SIDECAR_CPU_LIMIT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarCPULimit + {{- end }} + {{- if .Values.scc.create }} + - name: {{ include "k10.sccNameEnvVar" . }} + value: {{ .Release.Name }}-scc + {{- 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_EXECUTOR_WORKER_COUNT + valueFrom: + configMapKeyRef: + name: k10-config + key: K10ExecutorWorkerCount + - name: K10_EXECUTOR_MAX_CONCURRENT_RESTORE_CSI_SNAPSHOTS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10ExecutorMaxConcurrentRestoreCsiSnapshots + - name: K10_EXECUTOR_MAX_CONCURRENT_RESTORE_GENERIC_VOLUME_SNAPSHOTS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10ExecutorMaxConcurrentRestoreGenericVolumeSnapshots + - name: K10_EXECUTOR_MAX_CONCURRENT_RESTORE_WORKLOADS + valueFrom: + configMapKeyRef: + name: k10-config + key: K10ExecutorMaxConcurrentRestoreWorkloads + - name: KANISTER_BACKUP_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterBackupTimeout + - name: KANISTER_RESTORE_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterRestoreTimeout + - name: KANISTER_DELETE_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterDeleteTimeout + - name: KANISTER_HOOK_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterHookTimeout + - name: KANISTER_CHECKREPO_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterCheckRepoTimeout + - name: KANISTER_STATS_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterStatsTimeout + - name: KANISTER_EFSPOSTRESTORE_TIMEOUT + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterEFSPostRestoreTimeout + - name: KANISTER_MANAGED_DATA_SERVICES_BLUEPRINTS_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterManagedDataServicesBlueprintsEnabled +{{- if .Values.maxJobWaitDuration }} + - name: K10_JOB_MAX_WAIT_DURATION + valueFrom: + configMapKeyRef: + name: k10-config + key: k10JobMaxWaitDuration +{{- end }} + - name: K10_FORCE_ROOT_IN_KANISTER_HOOKS + valueFrom: + configMapKeyRef: + name: k10-config + key: k10ForceRootInKanisterHooks +{{- 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.1101/templates/_k10_image_tag.tpl b/charts/kasten/k10/7.0.1101/templates/_k10_image_tag.tpl new file mode 100644 index 0000000000..7a3df147bb --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/_k10_image_tag.tpl @@ -0,0 +1 @@ +{{- define "k10.imageTag" -}}7.0.11{{- end -}} \ No newline at end of file diff --git a/charts/kasten/k10/7.0.1101/templates/_k10_metering.tpl b/charts/kasten/k10/7.0.1101/templates/_k10_metering.tpl new file mode 100644 index 0000000000..d611839418 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/_k10_metering.tpl @@ -0,0 +1,356 @@ +{{/* 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.globalPodAnnotations" . | nindent 8 }} + checksum/config: {{ include (print .Template.BasePath "/k10-config.yaml") . | sha256sum }} + checksum/secret: {{ include (print .Template.BasePath "/secrets.yaml") . | sha256sum }} + labels: + {{- include "k10.globalPodLabels" . | nindent 8 }} + {{- include "helm.labels" . | nindent 8 }} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + component: {{ $service }} + run: {{ $service }}-svc + 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 .Values.kanisterPodMetricSidecar.enabled }} + - name: K10_KANISTER_POD_METRICS_ENABLED + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarEnabled + - name: K10_PROMETHEUS_PUSHGATEWAY_METRIC_LIFETIME + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodMetricSidecarMetricLifetime + - name: PUSHGATEWAY_METRICS_INTERVAL + valueFrom: + configMapKeyRef: + name: k10-config + key: KanisterPodPushgatewayMetricsInterval +{{- 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.1101/templates/_k10_serviceimage.tpl b/charts/kasten/k10/7.0.1101/templates/_k10_serviceimage.tpl new file mode 100644 index 0000000000..9a333d92ce --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/_k10_template.tpl b/charts/kasten/k10/7.0.1101/templates/_k10_template.tpl new file mode 100644 index 0000000000..7eec215fbb --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/_k10_template.tpl @@ -0,0 +1,234 @@ +{{/* 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: + {{- include "k10.globalPodAnnotations" . | nindent 8 }} + checksum/config: {{ include (print .Template.BasePath "/k10-config.yaml") . | sha256sum }} + checksum/secret: {{ include (print .Template.BasePath "/secrets.yaml") . | sha256sum }} + checksum/frontend-nginx-config: {{ include (print .Template.BasePath "/frontend-nginx-configmap.yaml") . | sha256sum }} +{{- if .Values.auth.ldap.restartPod }} + rollme: {{ randAlphaNum 5 | quote }} +{{- end}} +{{- if .Values.scc.create }} + openshift.io/required-scc: {{ .Release.Name }}-scc +{{- end }} + labels: + {{- include "k10.globalPodLabels" . | nindent 8 }} + {{- include "helm.labels" . | nindent 8 }} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + component: {{ $service }} + run: {{ $deploymentName }} + 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.1101/templates/_prometheus.tpl b/charts/kasten/k10/7.0.1101/templates/_prometheus.tpl new file mode 100644 index 0000000000..a49a8363f6 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/aggregatedaudit-policy.yaml b/charts/kasten/k10/7.0.1101/templates/aggregatedaudit-policy.yaml new file mode 100644 index 0000000000..ef7f03c6c2 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/api-tls-secrets.yaml b/charts/kasten/k10/7.0.1101/templates/api-tls-secrets.yaml new file mode 100644 index 0000000000..386d3b9999 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/apiservice.yaml b/charts/kasten/k10/7.0.1101/templates/apiservice.yaml new file mode 100644 index 0000000000..1811df48a8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/daemonsets.yaml b/charts/kasten/k10/7.0.1101/templates/daemonsets.yaml new file mode 100644 index 0000000000..b8c50b505f --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/deployments.yaml b/charts/kasten/k10/7.0.1101/templates/deployments.yaml new file mode 100644 index 0000000000..b0a8ab9a20 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/deployments.yaml @@ -0,0 +1,31 @@ +{{/* +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)) -}} + {{ $replicas := get $.Values (printf "%sReplicas" $k10_service) | default 1 }} + {{ $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.1101/templates/fluentbit-configmap.yaml b/charts/kasten/k10/7.0.1101/templates/fluentbit-configmap.yaml new file mode 100644 index 0000000000..71cecb9668 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/frontend-nginx-configmap.yaml b/charts/kasten/k10/7.0.1101/templates/frontend-nginx-configmap.yaml new file mode 100644 index 0000000000..93d17b3a17 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/gateway-ext.yaml b/charts/kasten/k10/7.0.1101/templates/gateway-ext.yaml new file mode 100644 index 0000000000..00da4c27b2 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/gateway.yaml b/charts/kasten/k10/7.0.1101/templates/gateway.yaml new file mode 100644 index 0000000000..93792a10d6 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/gateway.yaml @@ -0,0 +1,253 @@ +{{- $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.globalPodAnnotations" . | nindent 8 }} + checksum/config: {{ include (print .Template.BasePath "/k10-config.yaml") . | sha256sum }} + checksum/secret: {{ include (print .Template.BasePath "/secrets.yaml") . | sha256sum }} + labels: + {{- include "k10.globalPodLabels" . | nindent 8 }} + {{- include "helm.labels" . | nindent 8 }} + {{- include "k10.azMarketPlace.billingIdentifier" . | nindent 8 }} + service: gateway + component: gateway +{{- 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.1101/templates/grafana-scc.yaml b/charts/kasten/k10/7.0.1101/templates/grafana-scc.yaml new file mode 100644 index 0000000000..014d1be46c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/ingress.yaml b/charts/kasten/k10/7.0.1101/templates/ingress.yaml new file mode 100644 index 0000000000..df347477c6 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/k10-config.yaml b/charts/kasten/k10/7.0.1101/templates/k10-config.yaml new file mode 100644 index 0000000000..a6c52552e9 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/k10-config.yaml @@ -0,0 +1,271 @@ +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" . }} + concurrentSnapConversions: {{ default (include "k10.defaultConcurrentSnapshotConversions" .) .Values.limiter.concurrentSnapConversions | quote }} + concurrentWorkloadSnapshots: {{ include "k10.defaultConcurrentWorkloadSnapshots" . | quote }} + 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 }} + KanisterBackupTimeout: {{ default (include "k10.defaultKanisterBackupTimeout" .) .Values.kanister.backupTimeout | quote }} + KanisterRestoreTimeout: {{ default (include "k10.defaultKanisterRestoreTimeout" .) .Values.kanister.restoreTimeout | quote }} + KanisterDeleteTimeout: {{ default (include "k10.defaultKanisterDeleteTimeout" .) .Values.kanister.deleteTimeout | quote }} + KanisterHookTimeout: {{ default (include "k10.defaultKanisterHookTimeout" .) .Values.kanister.hookTimeout | quote }} + KanisterCheckRepoTimeout: {{ default (include "k10.defaultKanisterCheckRepoTimeout" .) .Values.kanister.checkRepoTimeout | quote }} + KanisterStatsTimeout: {{ default (include "k10.defaultKanisterStatsTimeout" .) .Values.kanister.statsTimeout | quote }} + KanisterEFSPostRestoreTimeout: {{ default (include "k10.defaultKanisterEFSPostRestoreTimeout" .) .Values.kanister.efsPostRestoreTimeout | quote }} + KanisterManagedDataServicesBlueprintsEnabled: {{ .Values.kanister.managedDataServicesBlueprintsEnabled | quote }} + KanisterPodReadyWaitTimeout: {{ .Values.kanister.podReadyWaitTimeout | quote }} + KanisterPodMetricSidecarEnabled: {{ .Values.kanisterPodMetricSidecar.enabled | quote }} + KanisterPodMetricSidecarMetricLifetime: {{ .Values.kanisterPodMetricSidecar.metricLifetime | quote }} + KanisterPodPushgatewayMetricsInterval: {{ .Values.kanisterPodMetricSidecar.pushGatewayInterval | quote }} +{{- include "kanisterPodMetricSidecarResources" . | indent 2 }} + KanisterToolsImage: {{ include "get.kanisterToolsImage" . | quote }} + K10MutatingWebhookTLSCertDir: "/etc/ssl/certs/webhook" + + K10LimiterGenericVolumeSnapshots: {{ default (include "k10.defaultK10LimiterGenericVolumeSnapshots" .) .Values.limiter.genericVolumeSnapshots | quote }} + K10LimiterGenericVolumeCopies: {{ default (include "k10.defaultK10LimiterGenericVolumeCopies" .) .Values.limiter.genericVolumeCopies | quote }} + K10LimiterGenericVolumeRestores: {{ default (include "k10.defaultK10LimiterGenericVolumeRestores" .) .Values.limiter.genericVolumeRestores | quote }} + K10LimiterCsiSnapshots: {{ default (include "k10.defaultK10LimiterCsiSnapshots" .) .Values.limiter.csiSnapshots | quote }} + K10LimiterProviderSnapshots: {{ default (include "k10.defaultK10LimiterProviderSnapshots" .) .Values.limiter.providerSnapshots | quote }} + K10LimiterImageCopies: {{ default (include "k10.defaultK10LimiterImageCopies" .) .Values.limiter.imageCopies | quote }} + K10ExecutorWorkerCount: {{ default (include "k10.defaultK10ExecutorWorkerCount" .) .Values.services.executor.workerCount | quote }} + K10ExecutorMaxConcurrentRestoreCsiSnapshots: {{ default (include "k10.defaultK10ExecutorMaxConcurrentRestoreCsiSnapshots" .) .Values.services.executor.maxConcurrentRestoreCsiSnapshots | quote }} + K10ExecutorMaxConcurrentRestoreGenericVolumeSnapshots: {{ default (include "k10.defaultK10ExecutorMaxConcurrentRestoreGenericVolumeSnapshots" .) .Values.services.executor.maxConcurrentRestoreGenericVolumeSnapshots | quote }} + K10ExecutorMaxConcurrentRestoreWorkloads: {{ default (include "k10.defaultK10ExecutorMaxConcurrentRestoreWorkloads" .) .Values.services.executor.maxConcurrentRestoreWorkloads | quote }} + + 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 }} + + k10JobMaxWaitDuration: {{ .Values.maxJobWaitDuration | quote }} + + quickDisasterRecoveryEnabled: {{ .Values.kastenDisasterRecovery.quickMode.enabled | quote }} + + k10ForceRootInKanisterHooks: {{ .Values.forceRootInKanisterHooks | quote }} + + 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.1101/templates/k10-eula.yaml b/charts/kasten/k10/7.0.1101/templates/k10-eula.yaml new file mode 100644 index 0000000000..21e251d6c8 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/k10-scc.yaml b/charts/kasten/k10/7.0.1101/templates/k10-scc.yaml new file mode 100644 index 0000000000..12a449f6fa --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/kopia-tls-certs.yaml b/charts/kasten/k10/7.0.1101/templates/kopia-tls-certs.yaml new file mode 100644 index 0000000000..ac0635f51c --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/license.yaml b/charts/kasten/k10/7.0.1101/templates/license.yaml new file mode 100644 index 0000000000..f409fb7e51 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/mc.yaml b/charts/kasten/k10/7.0.1101/templates/mc.yaml new file mode 100644 index 0000000000..2c23f94ae5 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/mutatingwebhook.yaml b/charts/kasten/k10/7.0.1101/templates/mutatingwebhook.yaml new file mode 100644 index 0000000000..729df58656 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/networkpolicy.yaml b/charts/kasten/k10/7.0.1101/templates/networkpolicy.yaml new file mode 100644 index 0000000000..c026de5147 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/networkpolicy.yaml @@ -0,0 +1,282 @@ +{{- $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 +--- +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 .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.1101/templates/ocp-ca-cert-extract-hook.yaml b/charts/kasten/k10/7.0.1101/templates/ocp-ca-cert-extract-hook.yaml new file mode 100644 index 0000000000..ac1523a9cf --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/ocp-ca-cert-extract-hook.yaml @@ -0,0 +1,195 @@ +{{- if (include "k10.ocpcacertsautoextraction" .) -}} +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 }}"] + backoffLimit: 0 +{{ end }} diff --git a/charts/kasten/k10/7.0.1101/templates/prometheus-configmap.yaml b/charts/kasten/k10/7.0.1101/templates/prometheus-configmap.yaml new file mode 100644 index 0000000000..227d19ae25 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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 .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.1101/templates/prometheus-scc.yaml b/charts/kasten/k10/7.0.1101/templates/prometheus-scc.yaml new file mode 100644 index 0000000000..4d039ef00e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/prometheus-service.yaml b/charts/kasten/k10/7.0.1101/templates/prometheus-service.yaml new file mode 100644 index 0000000000..a5a2281710 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/rbac.yaml b/charts/kasten/k10/7.0.1101/templates/rbac.yaml new file mode 100644 index 0000000000..ec68013e98 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/rhmarketplace.tpl b/charts/kasten/k10/7.0.1101/templates/rhmarketplace.tpl new file mode 100644 index 0000000000..e64022641e --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/rhmarketplace.tpl @@ -0,0 +1,8 @@ +{{/* +This file is used to fail the helm deployment if certain values are set which are +not compatible with an Operator deployment. +*/}} + +{{- if and (.Values.global.rhMarketPlace) (.Values.reporting.pdfReports) -}} + {{- fail "reporting.pdfReports cannot be enabled for the K10 Red Hat Marketplace Operator" -}} +{{- end -}} diff --git a/charts/kasten/k10/7.0.1101/templates/route.yaml b/charts/kasten/k10/7.0.1101/templates/route.yaml new file mode 100644 index 0000000000..1ecd244bed --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/secrets.yaml b/charts/kasten/k10/7.0.1101/templates/secrets.yaml new file mode 100644 index 0000000000..0a040e2c0b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/secure_deployment.tpl b/charts/kasten/k10/7.0.1101/templates/secure_deployment.tpl new file mode 100644 index 0000000000..b8e5a66428 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/secure_deployment.tpl @@ -0,0 +1,19 @@ +{{/* +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.ironbankPdfReports" . -}} +{{- include "k10.fail.ironbankPrometheus" . -}} +{{- include "k10.fail.ironbankRHMarketplace" . -}} + +{{/* FIPS */}} +{{- include "k10.fail.fipsGrafana" . -}} +{{- include "k10.fail.fipsPrometheus" . -}} +{{- include "k10.fail.fipsPDFReports" . -}} diff --git a/charts/kasten/k10/7.0.1101/templates/serviceaccount.yaml b/charts/kasten/k10/7.0.1101/templates/serviceaccount.yaml new file mode 100644 index 0000000000..b4e61c8922 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/serviceaccount.yaml @@ -0,0 +1,44 @@ +{{- 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 }} + 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.1101/templates/v0services.yaml b/charts/kasten/k10/7.0.1101/templates/v0services.yaml new file mode 100644 index 0000000000..8a744cfe36 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/workloadIdentityFederation.tpl b/charts/kasten/k10/7.0.1101/templates/workloadIdentityFederation.tpl new file mode 100644 index 0000000000..75296e98b0 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/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.1101/templates/{values}/grafana/values/grafana_values.tpl b/charts/kasten/k10/7.0.1101/templates/{values}/grafana/values/grafana_values.tpl new file mode 100644 index 0000000000..d6ac74f63b --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/{values}/grafana/values/grafana_values.tpl @@ -0,0 +1,275 @@ +{{/* + 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) -}} +{{- $_ := 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.1101/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl b/charts/kasten/k10/7.0.1101/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl new file mode 100644 index 0000000000..c40dd7a079 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/templates/{values}/prometheus/charts/{charts}/values/prometheus_values.tpl @@ -0,0 +1,183 @@ +{{/* + 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) -}} +{{- $_ := 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.1101/triallicense b/charts/kasten/k10/7.0.1101/triallicense new file mode 100644 index 0000000000..cfe6dd46bf --- /dev/null +++ b/charts/kasten/k10/7.0.1101/triallicense @@ -0,0 +1 @@ +Y3VzdG9tZXJOYW1lOiB0cmlhbHN0YXJ0ZXItbGljZW5zZQpkYXRlRW5kOiAnMjEwMC0wMS0wMVQwMDowMDowMC4wMDBaJwpkYXRlU3RhcnQ6ICcyMDIwLTAxLTAxVDAwOjAwOjAwLjAwMFonCmZlYXR1cmVzOgogIHRyaWFsOiBudWxsCmlkOiB0cmlhbC0wOWY4MzE5Zi0xODBmLTRhOTAtOTE3My1kOTJiNzZmMTgzNWUKcHJvZHVjdDogSzEwCnJlc3RyaWN0aW9uczoKICBub2RlczogNTAwCnNlcnZpY2VBY2NvdW50S2V5OiBudWxsCnZlcnNpb246IHYxLjAuMApzaWduYXR1cmU6IEYxbnVLUFV5STJtbDJGMmV1VHdGOXNZRTZMVU5rQ3ZiR2tTV1ZkT0ZqdERCb1B6SjUyVWFsVkFmRjVmQUxpcm5BcVhkcERnYi9YcnpxSEYrTE0xS2pEMVdXUFd0ZUdXNFc1anBPSFN0T296Y0c5M0pUUHF5M2l6TVk3RmczZVFLYTZzWDhBdnFwOXArWXVBMWNwTENlQ2dsR2dnOTVzSUFmYmRMMTBmV2d2RmR6QUt4dUZLN2psRzVtbG1CRVF5R0hrYWdoZFIrVGxzeUNTNEFkbXVBOEZodVUwZnRBdXN0b1M3R2JKd1BuTFI3STFZY1Q4OW8wU2xRZEJ2Yjg2QzdKbm1OdnY0aHhiSUo5TTJvWGJPSnQ4ZnBNcjhNWFR6YWRMTWJzSndhZ3VBVHlNUWF2cExHNXRPb0U2ZE1uMVlFVDZLdWZiYy9NdThVRDVYYXlDYTdkZz09Cg== diff --git a/charts/kasten/k10/7.0.1101/values.schema.json b/charts/kasten/k10/7.0.1101/values.schema.json new file mode 100644 index 0000000000..8ebb1424d5 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/values.schema.json @@ -0,0 +1,2874 @@ +{ + "$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)'" + }, + "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": 3, + "title": "Number of executor service pod replicas", + "description": "Set number of executor service pod replicas for better performance" + }, + "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" + } + } + }, + "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", + "title": "Metric sidecar for ephemeral pods", + "description": "Sidecar container for gathering metrics from ephemeral pods", + "properties": { + "enabled": { + "type": "boolean", + "default": true, + "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" + ] + } + } + } + } + } + } + }, + "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": "Resource configuration for Generic Volume Snapshot restore pods", + "properties": { + "resources": { + "type": "object", + "title": "Generic Volume Snapshot restore pod resource config", + "description": "Configure resource request and limits by Generic Volume Snapshot restore pods", + "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": "Generic Volume Snapshot resource limits configuration", + "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": 8, + "title": "Executor workers count", + "description": "Count of running executor workers" + }, + "maxConcurrentRestoreCsiSnapshots": { + "type": "integer", + "default": 3, + "title": "Concurrent restore CSI snapshots operations", + "description": "Limit of concurrent restore CSI snapshots operations per each restore action" + }, + "maxConcurrentRestoreGenericVolumeSnapshots": { + "type": "integer", + "default": 3, + "title": "Concurrent restore generic volume snapshots operations", + "description": "Limit of concurrent restore generic volume snapshots operations per each restore action" + }, + "maxConcurrentRestoreWorkloads": { + "type": "integer", + "default": 3, + "title": "Concurrent restore workloads operations", + "description": "Limit of concurrent restore workloads operations per each restore action" + } + } + }, + "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": 3, + "title": "Concurrent snapshot conversions", + "description": "Limit of concurrent snapshots to convert during export " + }, + "genericVolumeSnapshots": { + "type": "integer", + "default": 10, + "title": "Concurrent generic volume snapshot creation", + "description": "Limit of concurrent generic volume snapshot create operations" + }, + "genericVolumeCopies": { + "type": "integer", + "default": 10, + "title": "Concurrent generic volume snapshot copy", + "description": "Limit of concurrent generic volume snapshot copy operations" + }, + "genericVolumeRestores": { + "type": "integer", + "default": 10, + "title": "Concurrent generic volume snapshot restore", + "description": "Limit of concurrent generic volume snapshot restore operations" + }, + "csiSnapshots": { + "type": "integer", + "default": 10, + "title": "Concurrent CSI snapshot create", + "description": "Limit of concurrent CSI snapshot create operations" + }, + "providerSnapshots": { + "type": "integer", + "default": 10, + "title": "Concurrent cloud provider create", + "description": "Limit of concurrent cloud provider create operations" + }, + "imageCopies": { + "type": "integer", + "default": 10, + "title": "Concurrent image copy", + "description": "Limit of concurrent image copy operations" + } + } + }, + "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": 45, + "title": "Timeout on Kanister backup operations", + "description": "Timeout on Kanister backup operations in mins" + }, + "restoreTimeout": { + "type": "integer", + "default": 600, + "title": "Timeout for Kanister restore operations", + "description": "Timeout for Kanister restore operations in mins" + }, + "deleteTimeout": { + "type": "integer", + "default": 45, + "title": "Timeout for Kanister delete operations", + "description": "Timeout for Kanister delete operations in mins" + }, + "hookTimeout": { + "type": "integer", + "default": 20, + "title": "Timeout for Kanister pre-hook and post-hook operations", + "description": "Timeout for Kanister pre-hook and post-hook operations in minutes" + }, + "checkRepoTimeout": { + "type": "integer", + "default": 20, + "title": "Timeout for Kanister checkRepo operations", + "description": "Specify timeout to set on Kanister checkRepo operations in minutes" + }, + "statsTimeout": { + "type": "integer", + "default": 20, + "title": "Timeout for Kanister stats operations", + "description": "Timeout for Kanister stats operations in minutes" + }, + "efsPostRestoreTimeout": { + "type": "integer", + "default": 45, + "title": "Timeout for Kanister efsPostRestore operations", + "description": "Timeout for Kanister efsPostRestore operations in minutes" + }, + "podReadyWaitTimeout": { + "type": "integer", + "default": 15, + "title": "Timeout for Kanister tooling pods to be ready", + "description": "Timeout for Kanister tooling pods to be ready during operations in minutes" + }, + "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" + } + } + }, + "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" + } + } + }, + "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" + }, + "reporting": { + "type": "object", + "properties": { + "pdfReports": { + "title": "Enable PDF reports", + "description": "Enable download of PDF reports in the Dashboard", + "type": "boolean", + "default": false + } + } + }, + "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": "", + "title": "Maximum duration for jobs in minutes", + "description": "Set a maximum duration of waiting for child jobs. If the execution of the subordinate jobs exceeds this value, the parent job will be canceled. If no value is set, a default of 10 hours will be used" + }, + "forceRootInKanisterHooks": { + "type": "boolean", + "default": true, + "title": "Run Kanister Hooks as root", + "description": "Forces Kanister Execution Hooks to run with root privileges" + }, + "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.1101/values.yaml b/charts/kasten/k10/7.0.1101/values.yaml new file mode 100644 index 0000000000..3c6b2fc162 --- /dev/null +++ b/charts/kasten/k10/7.0.1101/values.yaml @@ -0,0 +1,545 @@ +# 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: '' + 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: '' +executorReplicas: 3 +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. + +kanisterPodMetricSidecar: + 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 + workerCount: 8 + maxConcurrentRestoreCsiSnapshots: 3 + maxConcurrentRestoreGenericVolumeSnapshots: 3 + maxConcurrentRestoreWorkloads: 3 + 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: + concurrentSnapConversions: 3 + genericVolumeSnapshots: 10 + genericVolumeCopies: 10 + genericVolumeRestores: 10 + csiSnapshots: 10 + providerSnapshots: 10 + imageCopies: 10 + +gateway: + insecureDisableSSLVerify: false + exposeAdminPort: true + service: + externalPort: 80 + resources: + requests: + memory: 300Mi + cpu: 200m + limits: + memory: 1Gi + cpu: 1000m + +kanister: + backupTimeout: 45 + restoreTimeout: 600 + deleteTimeout: 45 + hookTimeout: 20 + checkRepoTimeout: 20 + statsTimeout: 20 + efsPostRestoreTimeout: 45 + podReadyWaitTimeout: 15 + managedDataServicesBlueprintsEnabled: true + +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 + +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" + +reporting: + pdfReports: false + +logging: + internal: true + # Used to set an external fluentbit endpoint. 'logging.internal' must be set to false. + fluentbit_endpoint: "" + +maxJobWaitDuration: "" + +forceRootInKanisterHooks: 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/new-relic/nri-bundle/5.0.94/.helmignore b/charts/new-relic/nri-bundle/5.0.94/.helmignore new file mode 100644 index 0000000000..50af031725 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/new-relic/nri-bundle/5.0.94/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/Chart.lock new file mode 100644 index 0000000000..3f62955d6b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/Chart.lock @@ -0,0 +1,39 @@ +dependencies: +- name: newrelic-infrastructure + repository: https://newrelic.github.io/nri-kubernetes + version: 3.34.6 +- name: nri-prometheus + repository: https://newrelic.github.io/nri-prometheus + version: 2.1.19 +- name: newrelic-prometheus-agent + repository: https://newrelic.github.io/newrelic-prometheus-configurator + version: 1.14.4 +- name: nri-metadata-injection + repository: https://newrelic.github.io/k8s-metadata-injection + version: 4.21.2 +- name: newrelic-k8s-metrics-adapter + repository: https://newrelic.github.io/newrelic-k8s-metrics-adapter + version: 1.11.4 +- name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 5.12.1 +- name: nri-kube-events + repository: https://newrelic.github.io/nri-kube-events + version: 3.10.8 +- name: newrelic-logging + repository: https://newrelic.github.io/helm-charts + version: 1.23.0 +- name: newrelic-pixie + repository: https://newrelic.github.io/helm-charts + version: 2.1.4 +- name: k8s-agents-operator + repository: https://newrelic.github.io/k8s-agents-operator + version: 0.13.0 +- name: pixie-operator-chart + repository: https://pixie-operator-charts.storage.googleapis.com + version: 0.1.6 +- name: newrelic-infra-operator + repository: https://newrelic.github.io/newrelic-infra-operator + version: 2.11.4 +digest: sha256:3134efd351e6aa2f701d8bd3a1205dfa0b807136ba5c63ba79e5c87da45a4b4c +generated: "2024-10-07T13:42:59.811951434Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/Chart.yaml new file mode 100644 index 0000000000..9229dd2b0f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/Chart.yaml @@ -0,0 +1,85 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: New Relic + catalog.cattle.io/release-name: nri-bundle +apiVersion: v2 +dependencies: +- condition: infrastructure.enabled,newrelic-infrastructure.enabled + name: newrelic-infrastructure + repository: https://newrelic.github.io/nri-kubernetes + version: 3.34.6 +- condition: prometheus.enabled,nri-prometheus.enabled + name: nri-prometheus + repository: https://newrelic.github.io/nri-prometheus + version: 2.1.19 +- condition: newrelic-prometheus-agent.enabled + name: newrelic-prometheus-agent + repository: https://newrelic.github.io/newrelic-prometheus-configurator + version: 1.14.4 +- condition: webhook.enabled,nri-metadata-injection.enabled + name: nri-metadata-injection + repository: https://newrelic.github.io/k8s-metadata-injection + version: 4.21.2 +- condition: metrics-adapter.enabled,newrelic-k8s-metrics-adapter.enabled + name: newrelic-k8s-metrics-adapter + repository: https://newrelic.github.io/newrelic-k8s-metrics-adapter + version: 1.11.4 +- condition: ksm.enabled,kube-state-metrics.enabled + name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 5.12.1 +- condition: kubeEvents.enabled,nri-kube-events.enabled + name: nri-kube-events + repository: https://newrelic.github.io/nri-kube-events + version: 3.10.8 +- condition: logging.enabled,newrelic-logging.enabled + name: newrelic-logging + repository: https://newrelic.github.io/helm-charts + version: 1.23.0 +- condition: newrelic-pixie.enabled + name: newrelic-pixie + repository: https://newrelic.github.io/helm-charts + version: 2.1.4 +- condition: k8s-agents-operator.enabled + name: k8s-agents-operator + repository: https://newrelic.github.io/k8s-agents-operator + version: 0.13.0 +- alias: pixie-chart + condition: pixie-chart.enabled + name: pixie-operator-chart + repository: https://pixie-operator-charts.storage.googleapis.com + version: 0.1.6 +- condition: newrelic-infra-operator.enabled + name: newrelic-infra-operator + repository: https://newrelic.github.io/newrelic-infra-operator + version: 2.11.4 +description: Groups together the individual charts for the New Relic Kubernetes solution + for a more comfortable deployment. +home: https://github.com/newrelic/helm-charts +icon: file://assets/icons/nri-bundle.svg +keywords: +- infrastructure +- newrelic +- monitoring +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: nri-bundle +sources: +- https://github.com/newrelic/nri-bundle/ +- https://github.com/newrelic/nri-bundle/tree/master/charts/nri-bundle +- https://github.com/newrelic/nri-kubernetes/tree/master/charts/newrelic-infrastructure +- https://github.com/newrelic/nri-prometheus/tree/master/charts/nri-prometheus +- https://github.com/newrelic/newrelic-prometheus-configurator/tree/master/charts/newrelic-prometheus-agent +- https://github.com/newrelic/k8s-metadata-injection/tree/master/charts/nri-metadata-injection +- https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/master/charts/newrelic-k8s-metrics-adapter +- https://github.com/newrelic/nri-kube-events/tree/master/charts/nri-kube-events +- https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-logging +- https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-pixie +- https://github.com/newrelic/newrelic-infra-operator/tree/master/charts/newrelic-infra-operator +- https://github.com/newrelic/k8s-agents-operator/tree/master/charts/k8s-agents-operator +version: 5.0.94 diff --git a/charts/new-relic/nri-bundle/5.0.94/README.md b/charts/new-relic/nri-bundle/5.0.94/README.md new file mode 100644 index 0000000000..3fcc97d2b7 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/README.md @@ -0,0 +1,200 @@ +# nri-bundle + +Groups together the individual charts for the New Relic Kubernetes solution for a more comfortable deployment. + +**Homepage:** + +## Bundled charts + +This chart does not deploy anything by itself but has many charts as dependencies. This allows you to easily install and upgrade the New Relic +Kubernetes Integration using only one chart. + +In case you need more information about each component this chart installs, or you are an advanced user that want to install each component separately, +here is a list of components that this chart installs and where you can find more information about them: + +| Component | Installed by default? | Description | +|------------------------------|-----------------------|-------------| +| [newrelic-infrastructure](https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure) | Yes | Sends metrics about nodes, cluster objects (e.g. Deployments, Pods), and the control plane to New Relic. | +| [nri-metadata-injection](https://github.com/newrelic/k8s-metadata-injection/tree/main/charts/nri-metadata-injection) | Yes | Enriches New Relic-instrumented applications (APM) with Kubernetes information. | +| [kube-state-metrics](https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics) | | Required for `newrelic-infrastructure` to gather cluster-level metrics. | +| [nri-kube-events](https://github.com/newrelic/nri-kube-events/tree/main/charts/nri-kube-events) | | Reports Kubernetes events to New Relic. | +| [newrelic-infra-operator](https://github.com/newrelic/newrelic-infra-operator/tree/main/charts/newrelic-infra-operator) | | (Beta) Used with Fargate or serverless environments to inject `newrelic-infrastructure` as a sidecar instead of the usual DaemonSet. | +| [newrelic-k8s-metrics-adapter](https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/main/charts/newrelic-k8s-metrics-adapter) | | (Beta) Provides a source of data for Horizontal Pod Autoscalers (HPA) based on a NRQL query from New Relic. | +| [newrelic-logging](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-logging) | | Sends logs for Kubernetes components and workloads running on the cluster to New Relic. | +| [nri-prometheus](https://github.com/newrelic/nri-prometheus/tree/main/charts/nri-prometheus) | | Sends metrics from applications exposing Prometheus metrics to New Relic. | +| [newrelic-prometheus-configurator](https://github.com/newrelic/newrelic-prometheus-configurator/tree/master/charts/newrelic-prometheus-agent) | | Configures instances of Prometheus in Agent mode to send metrics to the New Relic Prometheus endpoint. | +| [newrelic-pixie](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-pixie) | | Connects to the Pixie API and enables the New Relic plugin in Pixie. The plugin allows you to export data from Pixie to New Relic for long-term data retention. | +| [Pixie](https://docs.pixielabs.ai/installing-pixie/install-schemes/helm/#3.-deploy) | | Is an open source observability tool for Kubernetes applications that uses eBPF to automatically capture telemetry data without the need for manual instrumentation. | +| [k8s-agents-operator](https://github.com/newrelic/k8s-agents-operator/tree/main/charts/k8s-agents-operator) | | (Preview) Streamlines full-stack observability for Kubernetes environments by automating APM instrumentation alongside Kubernetes agent deployment. | + +## Configure components + +It is possible to configure settings for the individual charts this chart groups by specifying values for them under a key using the name of the chart, +as specified in [helm documentation](https://helm.sh/docs/chart_template_guide/subcharts_and_globals). + +For example, by adding the following to the `values.yml` file: + +```yaml +# Configuration settings for the newrelic-infrastructure chart +newrelic-infrastructure: + # Any key defined in the values.yml file for the newrelic-infrastructure chart can be configured here: + # https://github.com/newrelic/nri-kubernetes/blob/main/charts/newrelic-infrastructure/values.yaml + + verboseLog: false + + resources: + limits: + memory: 512M +``` + +It is possible to override any entry of the [`newrelic-infrastructure`](https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure) +chart, as defined in their [`values.yml` file](https://github.com/newrelic/nri-kubernetes/blob/main/charts/newrelic-infrastructure/values.yaml). + +The same approach can be followed to update any of the subcharts. + +After making these changes to the `values.yml` file, or a custom values file, make sure to apply them using: + +``` +$ helm upgrade --reuse-values -f values.yaml [RELEASE] newrelic/nri-bundle +``` + +Where `[RELEASE]` is the name of the helm release, e.g. `newrelic-bundle`. + +## Monitor on host integrations + +If you wish to monitor services running on Kubernetes you can provide integrations +configuration under `integrations_config` that it will passed down to the `newrelic-infrastructure` chart. + +You just need to create a new entry where the "name" is the filename of the configuration file and the data is the content of +the integration configuration. The name must end in ".yaml" as this will be the +filename generated and the Infrastructure agent only looks for YAML files. + +The data part is the actual integration configuration as described in the spec here: +https://docs.newrelic.com/docs/integrations/integrations-sdk/file-specifications/integration-configuration-file-specifications-agent-v180 + +In the following example you can see how to monitor a Redis integration with autodiscovery + +```yaml +newrelic-infrastructure: + integrations: + nri-redis-sampleapp: + discovery: + command: + exec: /var/db/newrelic-infra/nri-discovery-kubernetes --tls --port 10250 + match: + label.app: sampleapp + integrations: + - name: nri-redis + env: + # using the discovered IP as the hostname address + HOSTNAME: ${discovery.ip} + PORT: 6379 + labels: + env: test +``` + +## Bring your own KSM + +New Relic Kubernetes Integration requires an instance of kube-state-metrics (KSM) to be running in the cluster, which this chart pulls as a dependency. If you are already running or want to run your own KSM instance, you will need to make some small adjustments as described below. + +### Bring your own KSM + +If you already have one KSM instance running, you can point `nri-kubernetes` to your instance: + +```yaml +kube-state-metrics: + # Disable bundled KSM. + enabled: false +newrelic-infrastructure: + ksm: + config: + # Selector for your pre-installed KSM Service. You may need to adjust this to fit your existing installation. + selector: "app.kubernetes.io/name=kube-state-metrics" + # Alternatively, you can specify a fixed URL where KSM is available. Doing so will bypass autodiscovery. + #staticUrl: http://ksm.ksm.svc.cluster.local:8080/metrics +``` + +### Run KSM alongside a different version + +If you need to run a different instance of KSM in your cluster, you can still run a separate instance for the Kubernetes Integration to work as intended: + +```yaml +kube-state-metrics: + # Enable bundled KSM. + enabled: true + prometheusScrape: false + customLabels: + # Label unique to this KSM instance. + newrelic.com/custom-ksm: "true" +newrelic-infrastructure: + ksm: + config: + # Use label above as a selector. + selector: "newrelic.com/custom-ksm=true" +``` + +For more information on supported KSM version visit the [requirements documentation](https://docs.newrelic.com/docs/kubernetes-pixie/kubernetes-integration/get-started/kubernetes-integration-compatibility-requirements#reqs) + +## Values managed globally + +Some of the subchart implement the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +At the time of writing this document, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this library and +honors global options as described below. + +Note, the value table below is automatically generated from `values.yaml` by `helm-docs`. If you need to add new fields or update existing fields, please update the `values.yaml` and then run `helm-docs` to update this value table. + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| global | object | See [`values.yaml`](values.yaml) | change the behaviour globally to all the supported helm charts. See [user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md) for further information. | +| global.affinity | object | `{}` | Sets pod/node affinities | +| global.cluster | string | `""` | The cluster name for the Kubernetes cluster. | +| global.containerSecurityContext | object | `{}` | Sets security context (at container level) | +| global.customAttributes | object | `{}` | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.customSecretLicenseKey | string | `""` | Key in the Secret object where the license key is stored | +| global.customSecretName | string | `""` | Name of the Secret object where the license key is stored | +| global.dnsConfig | object | `{}` | Sets pod's dnsConfig | +| global.fargate | bool | false | Must be set to `true` when deploying in an EKS Fargate environment | +| global.hostNetwork | bool | false | Sets pod's hostNetwork | +| global.images.pullSecrets | list | `[]` | Set secrets to be able to fetch images | +| global.images.registry | string | `""` | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.insightsKey | string | `""` | The license key for your New Relic Account. This will be preferred configuration option if both `insightsKey` and `customSecret` are specified. | +| global.labels | object | `{}` | Additional labels for chart objects | +| global.licenseKey | string | `""` | The license key for your New Relic Account. This will be preferred configuration option if both `licenseKey` and `customSecret` are specified. | +| global.lowDataMode | bool | false | Reduces number of metrics sent in order to reduce costs | +| global.nodeSelector | object | `{}` | Sets pod's node selector | +| global.nrStaging | bool | false | Send the metrics to the staging backend. Requires a valid staging license key | +| global.podLabels | object | `{}` | Additional labels for chart pods | +| global.podSecurityContext | object | `{}` | Sets security context (at pod level) | +| global.priorityClassName | string | `""` | Sets pod's priorityClassName | +| global.privileged | bool | false | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | | +| global.proxy | string | `""` | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.serviceAccount.annotations | object | `{}` | Add these annotations to the service account we create | +| global.serviceAccount.create | string | `nil` | Configures if the service account should be created or not | +| global.serviceAccount.name | string | `nil` | Change the name of the service account. This is honored if you disable on this chart the creation of the service account so you can use your own | +| global.tolerations | list | `[]` | Sets pod's tolerations to node taints | +| global.verboseLog | bool | false | Sets the debug logs to this integration or all integrations if it is set globally | +| k8s-agents-operator.enabled | bool | `false` | Install the [`k8s-agents-operator` chart](https://github.com/newrelic/k8s-agents-operator/tree/main/charts/k8s-agents-operator) | +| kube-state-metrics.enabled | bool | `false` | Install the [`kube-state-metrics` chart](https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics) from the stable helm charts repository. This is mandatory if `infrastructure.enabled` is set to `true` and the user does not provide its own instance of KSM version >=1.8 and <=2.0. Note, kube-state-metrics v2+ disables labels/annotations metrics by default. You can enable the target labels/annotations metrics to be monitored by using the metricLabelsAllowlist/metricAnnotationsAllowList options described [here](https://github.com/prometheus-community/helm-charts/blob/159cd8e4fb89b8b107dcc100287504bb91bf30e0/charts/kube-state-metrics/values.yaml#L274) in your Kubernetes clusters. | +| newrelic-infra-operator.enabled | bool | `false` | Install the [`newrelic-infra-operator` chart](https://github.com/newrelic/newrelic-infra-operator/tree/main/charts/newrelic-infra-operator) (Beta) | +| newrelic-infrastructure.enabled | bool | `true` | Install the [`newrelic-infrastructure` chart](https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure) | +| newrelic-k8s-metrics-adapter.enabled | bool | `false` | Install the [`newrelic-k8s-metrics-adapter.` chart](https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/main/charts/newrelic-k8s-metrics-adapter) (Beta) | +| newrelic-logging.enabled | bool | `false` | Install the [`newrelic-logging` chart](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-logging) | +| newrelic-pixie.enabled | bool | `false` | Install the [`newrelic-pixie`](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-pixie) | +| newrelic-prometheus-agent.enabled | bool | `false` | Install the [`newrelic-prometheus-agent` chart](https://github.com/newrelic/newrelic-prometheus-configurator/tree/main/charts/newrelic-prometheus-agent) | +| nri-kube-events.enabled | bool | `false` | Install the [`nri-kube-events` chart](https://github.com/newrelic/nri-kube-events/tree/main/charts/nri-kube-events) | +| nri-metadata-injection.enabled | bool | `true` | Install the [`nri-metadata-injection` chart](https://github.com/newrelic/k8s-metadata-injection/tree/main/charts/nri-metadata-injection) | +| nri-prometheus.enabled | bool | `false` | Install the [`nri-prometheus` chart](https://github.com/newrelic/nri-prometheus/tree/main/charts/nri-prometheus) | +| pixie-chart.enabled | bool | `false` | Install the [`pixie-chart` chart](https://docs.pixielabs.ai/installing-pixie/install-schemes/helm/#3.-deploy) | + +## Maintainers + +* [juanjjaramillo](https://github.com/juanjjaramillo) +* [csongnr](https://github.com/csongnr) +* [dbudziwojskiNR](https://github.com/dbudziwojskiNR) diff --git a/charts/new-relic/nri-bundle/5.0.94/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/README.md.gotmpl new file mode 100644 index 0000000000..269c4925a2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/README.md.gotmpl @@ -0,0 +1,166 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +## Bundled charts + +This chart does not deploy anything by itself but has many charts as dependencies. This allows you to easily install and upgrade the New Relic +Kubernetes Integration using only one chart. + +In case you need more information about each component this chart installs, or you are an advanced user that want to install each component separately, +here is a list of components that this chart installs and where you can find more information about them: + +| Component | Installed by default? | Description | +|------------------------------|-----------------------|-------------| +| [newrelic-infrastructure](https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure) | Yes | Sends metrics about nodes, cluster objects (e.g. Deployments, Pods), and the control plane to New Relic. | +| [nri-metadata-injection](https://github.com/newrelic/k8s-metadata-injection/tree/main/charts/nri-metadata-injection) | Yes | Enriches New Relic-instrumented applications (APM) with Kubernetes information. | +| [kube-state-metrics](https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics) | | Required for `newrelic-infrastructure` to gather cluster-level metrics. | +| [nri-kube-events](https://github.com/newrelic/nri-kube-events/tree/main/charts/nri-kube-events) | | Reports Kubernetes events to New Relic. | +| [newrelic-infra-operator](https://github.com/newrelic/newrelic-infra-operator/tree/main/charts/newrelic-infra-operator) | | (Beta) Used with Fargate or serverless environments to inject `newrelic-infrastructure` as a sidecar instead of the usual DaemonSet. | +| [newrelic-k8s-metrics-adapter](https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/main/charts/newrelic-k8s-metrics-adapter) | | (Beta) Provides a source of data for Horizontal Pod Autoscalers (HPA) based on a NRQL query from New Relic. | +| [newrelic-logging](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-logging) | | Sends logs for Kubernetes components and workloads running on the cluster to New Relic. | +| [nri-prometheus](https://github.com/newrelic/nri-prometheus/tree/main/charts/nri-prometheus) | | Sends metrics from applications exposing Prometheus metrics to New Relic. | +| [newrelic-prometheus-configurator](https://github.com/newrelic/newrelic-prometheus-configurator/tree/master/charts/newrelic-prometheus-agent) | | Configures instances of Prometheus in Agent mode to send metrics to the New Relic Prometheus endpoint. | +| [newrelic-pixie](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-pixie) | | Connects to the Pixie API and enables the New Relic plugin in Pixie. The plugin allows you to export data from Pixie to New Relic for long-term data retention. | +| [Pixie](https://docs.pixielabs.ai/installing-pixie/install-schemes/helm/#3.-deploy) | | Is an open source observability tool for Kubernetes applications that uses eBPF to automatically capture telemetry data without the need for manual instrumentation. | +| [k8s-agents-operator](https://github.com/newrelic/k8s-agents-operator/tree/main/charts/k8s-agents-operator) | | (Preview) Streamlines full-stack observability for Kubernetes environments by automating APM instrumentation alongside Kubernetes agent deployment. | + +## Configure components + +It is possible to configure settings for the individual charts this chart groups by specifying values for them under a key using the name of the chart, +as specified in [helm documentation](https://helm.sh/docs/chart_template_guide/subcharts_and_globals). + +For example, by adding the following to the `values.yml` file: + +```yaml +# Configuration settings for the newrelic-infrastructure chart +newrelic-infrastructure: + # Any key defined in the values.yml file for the newrelic-infrastructure chart can be configured here: + # https://github.com/newrelic/nri-kubernetes/blob/main/charts/newrelic-infrastructure/values.yaml + + verboseLog: false + + resources: + limits: + memory: 512M +``` + +It is possible to override any entry of the [`newrelic-infrastructure`](https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure) +chart, as defined in their [`values.yml` file](https://github.com/newrelic/nri-kubernetes/blob/main/charts/newrelic-infrastructure/values.yaml). + +The same approach can be followed to update any of the subcharts. + +After making these changes to the `values.yml` file, or a custom values file, make sure to apply them using: + +``` +$ helm upgrade --reuse-values -f values.yaml [RELEASE] newrelic/nri-bundle +``` + +Where `[RELEASE]` is the name of the helm release, e.g. `newrelic-bundle`. + + +## Monitor on host integrations + +If you wish to monitor services running on Kubernetes you can provide integrations +configuration under `integrations_config` that it will passed down to the `newrelic-infrastructure` chart. + +You just need to create a new entry where the "name" is the filename of the configuration file and the data is the content of +the integration configuration. The name must end in ".yaml" as this will be the +filename generated and the Infrastructure agent only looks for YAML files. + +The data part is the actual integration configuration as described in the spec here: +https://docs.newrelic.com/docs/integrations/integrations-sdk/file-specifications/integration-configuration-file-specifications-agent-v180 + +In the following example you can see how to monitor a Redis integration with autodiscovery + +```yaml +newrelic-infrastructure: + integrations: + nri-redis-sampleapp: + discovery: + command: + exec: /var/db/newrelic-infra/nri-discovery-kubernetes --tls --port 10250 + match: + label.app: sampleapp + integrations: + - name: nri-redis + env: + # using the discovered IP as the hostname address + HOSTNAME: ${discovery.ip} + PORT: 6379 + labels: + env: test +``` + +## Bring your own KSM + +New Relic Kubernetes Integration requires an instance of kube-state-metrics (KSM) to be running in the cluster, which this chart pulls as a dependency. If you are already running or want to run your own KSM instance, you will need to make some small adjustments as described below. + +### Bring your own KSM + +If you already have one KSM instance running, you can point `nri-kubernetes` to your instance: + +```yaml +kube-state-metrics: + # Disable bundled KSM. + enabled: false +newrelic-infrastructure: + ksm: + config: + # Selector for your pre-installed KSM Service. You may need to adjust this to fit your existing installation. + selector: "app.kubernetes.io/name=kube-state-metrics" + # Alternatively, you can specify a fixed URL where KSM is available. Doing so will bypass autodiscovery. + #staticUrl: http://ksm.ksm.svc.cluster.local:8080/metrics +``` + +### Run KSM alongside a different version + +If you need to run a different instance of KSM in your cluster, you can still run a separate instance for the Kubernetes Integration to work as intended: + +```yaml +kube-state-metrics: + # Enable bundled KSM. + enabled: true + prometheusScrape: false + customLabels: + # Label unique to this KSM instance. + newrelic.com/custom-ksm: "true" +newrelic-infrastructure: + ksm: + config: + # Use label above as a selector. + selector: "newrelic.com/custom-ksm=true" +``` + +For more information on supported KSM version visit the [requirements documentation](https://docs.newrelic.com/docs/kubernetes-pixie/kubernetes-integration/get-started/kubernetes-integration-compatibility-requirements#reqs) + +## Values managed globally + +Some of the subchart implement the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +At the time of writing this document, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this library and +honors global options as described below. + +Note, the value table below is automatically generated from `values.yaml` by `helm-docs`. If you need to add new fields or update existing fields, please update the `values.yaml` and then run `helm-docs` to update this value table. + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/app-readme.md b/charts/new-relic/nri-bundle/5.0.94/app-readme.md new file mode 100644 index 0000000000..61e5507873 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/app-readme.md @@ -0,0 +1,5 @@ +# New Relic Kubernetes Integration + +New Relic's Kubernetes integration gives you full observability into the health and performance of your environment, no matter whether you run Kubernetes on-premises or in the cloud. With our [cluster explorer](https://docs.newrelic.com/docs/integrations/kubernetes-integration/cluster-explorer/kubernetes-cluster-explorer), you can cut through layers of complexity to see how your cluster is performing, from the heights of the control plane down to applications running on a single pod. + +You can see the power of the Kubernetes integration in the [cluster explorer](https://docs.newrelic.com/docs/integrations/kubernetes-integration/cluster-explorer/kubernetes-cluster-explorer), where the full picture of a cluster is made available on a single screen: nodes and pods are visualized according to their health and performance, with pending and alerting nodes in the innermost circles. [Predefined alert conditions](https://docs.newrelic.com/docs/integrations/kubernetes-integration/kubernetes-events/kubernetes-integration-predefined-alert-policy) help you troubleshoot issues right from the start. Clicking each node reveals its status and how each app is performing. \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/.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/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/Chart.yaml new file mode 100644 index 0000000000..13b99710bd --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/Chart.yaml @@ -0,0 +1,16 @@ +apiVersion: v2 +appVersion: 0.13.0 +description: A Helm chart for the Kubernetes Agents Operator +home: https://github.com/newrelic/k8s-agents-operator/blob/main/charts/k8s-agents-operator/README.md +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: k8s-agents-operator +sources: +- https://github.com/newrelic/k8s-agents-operator +type: application +version: 0.13.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md new file mode 100644 index 0000000000..5c1da2d0a7 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md @@ -0,0 +1,204 @@ +# k8s-agents-operator + +![Version: 0.13.0](https://img.shields.io/badge/Version-0.13.0-informational?style=flat-square) ![Type: application](https://img.shields.io/badge/Type-application-informational?style=flat-square) ![AppVersion: 0.13.0](https://img.shields.io/badge/AppVersion-0.13.0-informational?style=flat-square) + +A Helm chart for the Kubernetes Agents Operator + +**Homepage:** + +## Prerequisites + +[Helm](https://helm.sh) must be installed to use the charts. Please refer to Helm's [documentation](https://helm.sh/docs) to get started. + +## Installation + +### Requirements + +Add the `k8s-agents-operator` Helm chart repository: +```shell +helm repo add k8s-agents-operator https://newrelic.github.io/k8s-agents-operator +``` + +### Instrumentation + +Install the [`k8s-agents-operator`](https://github.com/newrelic/k8s-agents-operator) Helm chart: +```shell +helm upgrade --install k8s-agents-operator k8s-agents-operator/k8s-agents-operator \ + --namespace newrelic \ + --create-namespace \ + --values your-custom-values.yaml +``` + +### Monitored namespaces + +For each namespace you want the operator to be instrumented, create a secret containing a valid New Relic ingest license key: +```shell +kubectl create secret generic newrelic-key-secret \ + --namespace my-monitored-namespace \ + --from-literal=new_relic_license_key= +``` + +Similarly, for each namespace you need to instrument create the `Instrumentation` custom resource, specifying which APM agents you want to instrument. All available APM agent docker images and corresponding tags are listed on DockerHub: +* [Java](https://hub.docker.com/repository/docker/newrelic/newrelic-java-init/general) +* [Node](https://hub.docker.com/repository/docker/newrelic/newrelic-node-init/general) +* [Python](https://hub.docker.com/repository/docker/newrelic/newrelic-python-init/general) +* [.NET](https://hub.docker.com/repository/docker/newrelic/newrelic-dotnet-init/general) +* [Ruby](https://hub.docker.com/repository/docker/newrelic/newrelic-ruby-init/general) + +```yaml +apiVersion: newrelic.com/v1alpha1 +kind: Instrumentation +metadata: + labels: + app.kubernetes.io/name: instrumentation + app.kubernetes.io/created-by: k8s-agents-operator + name: newrelic-instrumentation +spec: + java: + image: newrelic/newrelic-java-init:latest + # env: + # Example New Relic agent supported environment variables + # - name: NEW_RELIC_LABELS + # value: "environment:auto-injection" + # Example overriding the appName configuration + # - name: NEW_RELIC_POD_NAME + # valueFrom: + # fieldRef: + # fieldPath: metadata.name + # - name: NEW_RELIC_APP_NAME + # value: "$(NEW_RELIC_LABELS)-$(NEW_RELIC_POD_NAME)" + nodejs: + image: newrelic/newrelic-node-init:latest + python: + image: newrelic/newrelic-python-init:latest + dotnet: + image: newrelic/newrelic-dotnet-init:latest + ruby: + image: newrelic/newrelic-ruby-init:latest +``` +In the example above, we show how you can configure the agent settings globally using environment variables. See each agent's configuration documentation for available configuration options: +* [Java](https://docs.newrelic.com/docs/apm/agents/java-agent/configuration/java-agent-configuration-config-file/) +* [Node](https://docs.newrelic.com/docs/apm/agents/nodejs-agent/installation-configuration/nodejs-agent-configuration/) +* [Python](https://docs.newrelic.com/docs/apm/agents/python-agent/configuration/python-agent-configuration/) +* [.NET](https://docs.newrelic.com/docs/apm/agents/net-agent/configuration/net-agent-configuration/) +* [Ruby](https://docs.newrelic.com/docs/apm/agents/ruby-agent/configuration/ruby-agent-configuration/) + +Global agent settings can be overridden in your deployment manifest if a different configuration is required. + +### Annotations + +The `k8s-agents-operator` looks for language-specific annotations when your pods are being scheduled to know which applications you want to monitor. + +Below are the currently supported annotations: +```yaml +instrumentation.newrelic.com/inject-java: "true" +instrumentation.newrelic.com/inject-nodejs: "true" +instrumentation.newrelic.com/inject-python: "true" +instrumentation.newrelic.com/inject-dotnet: "true" +instrumentation.newrelic.com/inject-ruby: "true" +``` + +Example deployment with annotation to instrument the Java agent: +```yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: spring-petclinic +spec: + selector: + matchLabels: + app: spring-petclinic + replicas: 1 + template: + metadata: + labels: + app: spring-petclinic + annotations: + instrumentation.newrelic.com/inject-java: "true" + spec: + containers: + - name: spring-petclinic + image: ghcr.io/pavolloffay/spring-petclinic:latest + ports: + - containerPort: 8080 + env: + - name: NEW_RELIC_APP_NAME + value: spring-petclinic-demo +``` + +### cert-manager + +The K8s Agents Operator supports the use of [`cert-manager`](https://github.com/cert-manager/cert-manager) if preferred. + +Install the [`cert-manager`](https://github.com/cert-manager/cert-manager) Helm chart: +```shell +helm install cert-manager jetstack/cert-manager \ + --namespace cert-manager \ + --create-namespace \ + --set crds.enabled=true +``` + +In your `values.yaml` file, set `admissionWebhooks.autoGenerateCert.enabled: false` and `admissionWebhooks.certManager.enabled: true`. Then install the chart as normal. + +## Available Chart Releases + +To see the available charts: +```shell +helm search repo k8s-agents-operator +``` + +If you want to see a list of all available charts and releases, check [index.yaml](https://newrelic.github.io/k8s-agents-operator/index.yaml). + +## Source Code + +* + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| admissionWebhooks | object | `{"autoGenerateCert":{"certPeriodDays":365,"enabled":true,"recreate":true},"caFile":"","certFile":"","certManager":{"enabled":false},"create":true,"keyFile":""}` | Admission webhooks make sure only requests with correctly formatted rules will get into the Operator | +| admissionWebhooks.autoGenerateCert.certPeriodDays | int | `365` | Cert validity period time in days. | +| admissionWebhooks.autoGenerateCert.enabled | bool | `true` | If true and certManager.enabled is false, Helm will automatically create a self-signed cert and secret for you. | +| admissionWebhooks.autoGenerateCert.recreate | bool | `true` | If set to true, new webhook key/certificate is generated on helm upgrade. | +| admissionWebhooks.caFile | string | `""` | Path to the CA cert. | +| admissionWebhooks.certFile | string | `""` | Path to your own PEM-encoded certificate. | +| admissionWebhooks.certManager.enabled | bool | `false` | If true and autoGenerateCert.enabled is false, cert-manager will create a self-signed cert and secret for you. | +| admissionWebhooks.keyFile | string | `""` | Path to your own PEM-encoded private key. | +| controllerManager.kubeRbacProxy.image.repository | string | `"gcr.io/kubebuilder/kube-rbac-proxy"` | | +| controllerManager.kubeRbacProxy.image.tag | string | `"v0.14.0"` | | +| controllerManager.kubeRbacProxy.resources.limits.cpu | string | `"500m"` | | +| controllerManager.kubeRbacProxy.resources.limits.memory | string | `"128Mi"` | | +| controllerManager.kubeRbacProxy.resources.requests.cpu | string | `"5m"` | | +| controllerManager.kubeRbacProxy.resources.requests.memory | string | `"64Mi"` | | +| controllerManager.manager.image.pullPolicy | string | `nil` | | +| controllerManager.manager.image.repository | string | `"newrelic/k8s-agents-operator"` | | +| controllerManager.manager.image.tag | string | `nil` | | +| controllerManager.manager.leaderElection | object | `{"enabled":true}` | Enable leader election mechanism for protecting against split brain if multiple operator pods/replicas are started | +| controllerManager.manager.resources.requests.cpu | string | `"100m"` | | +| controllerManager.manager.resources.requests.memory | string | `"64Mi"` | | +| controllerManager.manager.serviceAccount.create | bool | `true` | | +| controllerManager.replicas | int | `1` | | +| kubernetesClusterDomain | string | `"cluster.local"` | | +| licenseKey | string | `""` | This set this license key to use. Can be configured also with `global.licenseKey` | +| metricsService.ports[0].name | string | `"https"` | | +| metricsService.ports[0].port | int | `8443` | | +| metricsService.ports[0].protocol | string | `"TCP"` | | +| metricsService.ports[0].targetPort | string | `"https"` | | +| metricsService.type | string | `"ClusterIP"` | | +| securityContext | object | `{"fsGroup":65532,"runAsGroup":65532,"runAsNonRoot":true,"runAsUser":65532}` | SecurityContext holds pod-level security attributes and common container settings | +| webhookService.ports[0].port | int | `443` | | +| webhookService.ports[0].protocol | string | `"TCP"` | | +| webhookService.ports[0].targetPort | int | `9443` | | +| webhookService.type | string | `"ClusterIP"` | | + +## Maintainers + +| Name | Email | Url | +| ---- | ------ | --- | +| juanjjaramillo | | | +| csongnr | | | +| dbudziwojskiNR | | | + +---------------------------------------------- +Autogenerated from chart metadata using [helm-docs v1.13.1](https://github.com/norwoodj/helm-docs/releases/v1.13.1) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md.gotmpl new file mode 100644 index 0000000000..5e583b703e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/README.md.gotmpl @@ -0,0 +1,162 @@ +{{ template "chart.header" . }} + +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.badgesSection" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +## Prerequisites + +[Helm](https://helm.sh) must be installed to use the charts. Please refer to Helm's [documentation](https://helm.sh/docs) to get started. + +## Installation + +### Requirements + +Add the `k8s-agents-operator` Helm chart repository: +```shell +helm repo add k8s-agents-operator https://newrelic.github.io/k8s-agents-operator +``` + +### Instrumentation + +Install the [`k8s-agents-operator`](https://github.com/newrelic/k8s-agents-operator) Helm chart: +```shell +helm upgrade --install k8s-agents-operator k8s-agents-operator/k8s-agents-operator \ + --namespace newrelic \ + --create-namespace \ + --values your-custom-values.yaml +``` + +### Monitored namespaces + +For each namespace you want the operator to be instrumented, create a secret containing a valid New Relic ingest license key: +```shell +kubectl create secret generic newrelic-key-secret \ + --namespace my-monitored-namespace \ + --from-literal=new_relic_license_key= +``` + +Similarly, for each namespace you need to instrument create the `Instrumentation` custom resource, specifying which APM agents you want to instrument. All available APM agent docker images and corresponding tags are listed on DockerHub: +* [Java](https://hub.docker.com/repository/docker/newrelic/newrelic-java-init/general) +* [Node](https://hub.docker.com/repository/docker/newrelic/newrelic-node-init/general) +* [Python](https://hub.docker.com/repository/docker/newrelic/newrelic-python-init/general) +* [.NET](https://hub.docker.com/repository/docker/newrelic/newrelic-dotnet-init/general) +* [Ruby](https://hub.docker.com/repository/docker/newrelic/newrelic-ruby-init/general) + +```yaml +apiVersion: newrelic.com/v1alpha1 +kind: Instrumentation +metadata: + labels: + app.kubernetes.io/name: instrumentation + app.kubernetes.io/created-by: k8s-agents-operator + name: newrelic-instrumentation +spec: + java: + image: newrelic/newrelic-java-init:latest + # env: + # Example New Relic agent supported environment variables + # - name: NEW_RELIC_LABELS + # value: "environment:auto-injection" + # Example overriding the appName configuration + # - name: NEW_RELIC_POD_NAME + # valueFrom: + # fieldRef: + # fieldPath: metadata.name + # - name: NEW_RELIC_APP_NAME + # value: "$(NEW_RELIC_LABELS)-$(NEW_RELIC_POD_NAME)" + nodejs: + image: newrelic/newrelic-node-init:latest + python: + image: newrelic/newrelic-python-init:latest + dotnet: + image: newrelic/newrelic-dotnet-init:latest + ruby: + image: newrelic/newrelic-ruby-init:latest +``` +In the example above, we show how you can configure the agent settings globally using environment variables. See each agent's configuration documentation for available configuration options: +* [Java](https://docs.newrelic.com/docs/apm/agents/java-agent/configuration/java-agent-configuration-config-file/) +* [Node](https://docs.newrelic.com/docs/apm/agents/nodejs-agent/installation-configuration/nodejs-agent-configuration/) +* [Python](https://docs.newrelic.com/docs/apm/agents/python-agent/configuration/python-agent-configuration/) +* [.NET](https://docs.newrelic.com/docs/apm/agents/net-agent/configuration/net-agent-configuration/) +* [Ruby](https://docs.newrelic.com/docs/apm/agents/ruby-agent/configuration/ruby-agent-configuration/) + +Global agent settings can be overridden in your deployment manifest if a different configuration is required. + +### Annotations + +The `k8s-agents-operator` looks for language-specific annotations when your pods are being scheduled to know which applications you want to monitor. + +Below are the currently supported annotations: +```yaml +instrumentation.newrelic.com/inject-java: "true" +instrumentation.newrelic.com/inject-nodejs: "true" +instrumentation.newrelic.com/inject-python: "true" +instrumentation.newrelic.com/inject-dotnet: "true" +instrumentation.newrelic.com/inject-ruby: "true" +``` + +Example deployment with annotation to instrument the Java agent: +```yaml +apiVersion: apps/v1 +kind: Deployment +metadata: + name: spring-petclinic +spec: + selector: + matchLabels: + app: spring-petclinic + replicas: 1 + template: + metadata: + labels: + app: spring-petclinic + annotations: + instrumentation.newrelic.com/inject-java: "true" + spec: + containers: + - name: spring-petclinic + image: ghcr.io/pavolloffay/spring-petclinic:latest + ports: + - containerPort: 8080 + env: + - name: NEW_RELIC_APP_NAME + value: spring-petclinic-demo +``` + +### cert-manager + +The K8s Agents Operator supports the use of [`cert-manager`](https://github.com/cert-manager/cert-manager) if preferred. + +Install the [`cert-manager`](https://github.com/cert-manager/cert-manager) Helm chart: +```shell +helm install cert-manager jetstack/cert-manager \ + --namespace cert-manager \ + --create-namespace \ + --set crds.enabled=true +``` + +In your `values.yaml` file, set `admissionWebhooks.autoGenerateCert.enabled: false` and `admissionWebhooks.certManager.enabled: true`. Then install the chart as normal. + +## Available Chart Releases + +To see the available charts: +```shell +helm search repo k8s-agents-operator +``` + +If you want to see a list of all available charts and releases, check [index.yaml](https://newrelic.github.io/k8s-agents-operator/index.yaml). + +{{ template "chart.sourcesSection" . }} + +{{ template "chart.requirementsSection" . }} + +{{ template "chart.valuesSection" . }} + +{{ template "chart.maintainersSection" . }} + +{{ template "helm-docs.versionFooter" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/NOTES.txt new file mode 100644 index 0000000000..b330f475d0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/NOTES.txt @@ -0,0 +1,36 @@ +This project is currently in preview. +Issues and contributions should be reported to the project's GitHub. +{{- if (include "k8s-agents-operator.areValuesValid" .) }} +===================================== + + ******** + **************** + ********** **********, + &&&**** ****/((( + &&&&&&& (((((( + &&&&&&&&&& (((((( + &&&&&&&& (((((( + &&&&& (((((( + &&&&& (((((((( + &&&&& .(((((((((( + &&&&&(((((((( + &&&(((, + +Your deployment of the New Relic Agent Operator is complete. +You can check on the progress of this by running the following command: + +kubectl get deployments -o wide -w --namespace {{ .Release.Namespace }} {{ template "k8s-agents-operator.fullname" . }} + +WARNING: This deployment will be incomplete until you configure your Instrumentation custom resource definition. +===================================== + +Please visit https://github.com/newrelic/k8s-agents-operator for instructions on how to create & configure the +Instrumentation custom resource definition required by the Operator. +{{- else }} + +############################################################################## +#### ERROR: You did not set a license key. #### +############################################################################## + +This deployment will be incomplete until you get your ingest license key from New Relic. +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/_helpers.tpl new file mode 100644 index 0000000000..f6ed6d04ed --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/_helpers.tpl @@ -0,0 +1,121 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "k8s-agents-operator.name" -}} +{{- .Chart.Name | 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 "k8s-agents-operator.fullname" -}} +{{- printf "%s-%s" .Release.Name .Chart.Name | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "k8s-agents-operator.chart" -}} +{{- printf "%s" .Chart.Name | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "k8s-agents-operator.labels" -}} +helm.sh/chart: {{ include "k8s-agents-operator.chart" . }} +{{ include "k8s-agents-operator.selectorLabels" . }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "k8s-agents-operator.selectorLabels" -}} +app.kubernetes.io/name: {{ include "k8s-agents-operator.chart" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "k8s-agents-operator.serviceAccountName" -}} +{{- if .Values.controllerManager.manager.serviceAccount.create }} +{{- default (include "k8s-agents-operator.name" .) .Values.controllerManager.manager.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.controllerManager.manager.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +Return the licenseKey +*/}} +{{- define "k8s-agents-operator.licenseKey" -}} +{{- if .Values.global}} + {{- if .Values.global.licenseKey }} + {{- .Values.global.licenseKey -}} + {{- else -}} + {{- .Values.licenseKey | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.licenseKey | default "" -}} +{{- end -}} +{{- end -}} + +{{/* +Returns if the template should render, it checks if the required values are set. +*/}} +{{- define "k8s-agents-operator.areValuesValid" -}} +{{- $licenseKey := include "k8s-agents-operator.licenseKey" . -}} +{{- and (or $licenseKey)}} +{{- end -}} + +{{/* +Controller manager service certificate's secret. +*/}} +{{- define "k8s-agents-operator.certificateSecret" -}} +{{- printf "%s-controller-manager-service-cert" (include "k8s-agents-operator.fullname" .) | trunc 63 | trimSuffix "-" -}} +{{- end }} + +{{/* +Return certificate and CA for Webhooks. +It handles variants when a cert has to be generated by Helm, +a cert is loaded from an existing secret or is provided via `.Values` +*/}} +{{- define "k8s-agents-operator.webhookCert" -}} +{{- $caCert := "" }} +{{- $clientCert := "" }} +{{- $clientKey := "" }} +{{- if .Values.admissionWebhooks.autoGenerateCert.enabled }} + {{- $prevSecret := (lookup "v1" "Secret" .Release.Namespace (include "k8s-agents-operator.certificateSecret" . )) }} + {{- if and (not .Values.admissionWebhooks.autoGenerateCert.recreate) $prevSecret }} + {{- $clientCert = index $prevSecret "data" "tls.crt" }} + {{- $clientKey = index $prevSecret "data" "tls.key" }} + {{- $caCert = index $prevSecret "data" "ca.crt" }} + {{- if not $caCert }} + {{- $prevHook := (lookup "admissionregistration.k8s.io/v1" "MutatingWebhookConfiguration" .Release.Namespace (print (include "k8s-agents-operator.fullname" . ) "-mutation")) }} + {{- if not (eq (toString $prevHook) "") }} + {{- $caCert = (first $prevHook.webhooks).clientConfig.caBundle }} + {{- end }} + {{- end }} + {{- else }} + {{- $certValidity := int .Values.admissionWebhooks.autoGenerateCert.certPeriodDays | default 365 }} + {{- $ca := genCA "k8s-agents-operator-operator-ca" $certValidity }} + {{- $domain1 := printf "%s-webhook-service.%s.svc" (include "k8s-agents-operator.fullname" .) $.Release.Namespace }} + {{- $domain2 := printf "%s-webhook-service.%s.svc.%s" (include "k8s-agents-operator.fullname" .) $.Release.Namespace $.Values.kubernetesClusterDomain }} + {{- $domains := list $domain1 $domain2 }} + {{- $cert := genSignedCert (include "k8s-agents-operator.fullname" .) nil $domains $certValidity $ca }} + {{- $clientCert = b64enc $cert.Cert }} + {{- $clientKey = b64enc $cert.Key }} + {{- $caCert = b64enc $ca.Cert }} + {{- end }} +{{- else }} + {{- $clientCert = .Files.Get .Values.admissionWebhooks.certFile | b64enc }} + {{- $clientKey = .Files.Get .Values.admissionWebhooks.keyFile | b64enc }} + {{- $caCert = .Files.Get .Values.admissionWebhooks.caFile | b64enc }} +{{- end }} +{{- $result := dict "clientCert" $clientCert "clientKey" $clientKey "caCert" $caCert }} +{{- $result | toYaml }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/certmanager.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/certmanager.yaml new file mode 100644 index 0000000000..ccef9f25bc --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/certmanager.yaml @@ -0,0 +1,30 @@ +{{- if and .Values.admissionWebhooks.create .Values.admissionWebhooks.certManager.enabled }} +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-serving-cert + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +spec: + dnsNames: + - '{{ template "k8s-agents-operator.fullname" . }}-webhook-service.{{ .Release.Namespace }}.svc' + - '{{ template "k8s-agents-operator.fullname" . }}-webhook-service.{{ .Release.Namespace }}.svc.{{ .Values.kubernetesClusterDomain }}' + issuerRef: + kind: Issuer + name: '{{ template "k8s-agents-operator.fullname" . }}-selfsigned-issuer' + secretName: {{ template "k8s-agents-operator.certificateSecret" . }} + subject: + organizationalUnits: + - k8s-agents-operator +--- +apiVersion: cert-manager.io/v1 +kind: Issuer +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-selfsigned-issuer + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +spec: + selfSigned: {} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/deployment.yaml new file mode 100644 index 0000000000..8cccd259e3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/deployment.yaml @@ -0,0 +1,91 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ template "k8s-agents-operator.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ template "k8s-agents-operator.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + control-plane: controller-manager + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +spec: + replicas: {{ .Values.controllerManager.replicas }} + selector: + matchLabels: + control-plane: controller-manager + {{- include "k8s-agents-operator.labels" . | nindent 6 }} + template: + metadata: + labels: + control-plane: controller-manager + {{- include "k8s-agents-operator.labels" . | nindent 8 }} + spec: + containers: + - args: + - --metrics-addr=127.0.0.1:8080 + {{- if .Values.controllerManager.manager.leaderElection.enabled }} + - --enable-leader-election + {{- end }} + - --zap-log-level=info + - --zap-time-encoding=rfc3339nano + env: + - name: KUBERNETES_CLUSTER_DOMAIN + value: {{ quote .Values.kubernetesClusterDomain }} + - name: ENABLE_WEBHOOKS + value: "true" + image: {{ .Values.controllerManager.manager.image.repository }}:{{ .Values.controllerManager.manager.image.tag | default .Chart.AppVersion }} + imagePullPolicy: {{ .Values.controllerManager.manager.image.pullPolicy | default "Always" }} + livenessProbe: + httpGet: + path: /healthz + port: 8081 + initialDelaySeconds: 15 + periodSeconds: 20 + name: manager + ports: + - containerPort: 9443 + name: webhook-server + protocol: TCP + readinessProbe: + httpGet: + path: /readyz + port: 8081 + initialDelaySeconds: 5 + periodSeconds: 10 + resources: {{- toYaml .Values.controllerManager.manager.resources | nindent 10 }} + volumeMounts: + - mountPath: /tmp/k8s-webhook-server/serving-certs + name: cert + readOnly: true + - args: + - --secure-listen-address=0.0.0.0:8443 + - --upstream=http://127.0.0.1:8080/ + - --logtostderr=true + - --v=0 + env: + - name: KUBERNETES_CLUSTER_DOMAIN + value: {{ quote .Values.kubernetesClusterDomain }} + image: {{ .Values.controllerManager.kubeRbacProxy.image.repository }}:{{ .Values.controllerManager.kubeRbacProxy.image.tag | default .Chart.AppVersion }} + name: kube-rbac-proxy + ports: + - containerPort: 8443 + name: https + protocol: TCP + resources: {{- toYaml .Values.controllerManager.kubeRbacProxy.resources | nindent 10 }} + serviceAccountName: {{ template "k8s-agents-operator.serviceAccountName" . }} + terminationGracePeriodSeconds: 10 + {{- if or .Values.admissionWebhooks.create (include "k8s-agents-operator.certificateSecret" . ) }} + volumes: + - name: cert + secret: + defaultMode: 420 + secretName: {{ template "k8s-agents-operator.certificateSecret" . }} + {{- end }} + securityContext: + {{- toYaml .Values.securityContext | nindent 8 }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/instrumentation-crd.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/instrumentation-crd.yaml new file mode 100644 index 0000000000..ae81414fb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/instrumentation-crd.yaml @@ -0,0 +1,1150 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: instrumentations.newrelic.com + annotations: + controller-gen.kubebuilder.io/version: v0.11.3 + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +spec: + group: newrelic.com + names: + kind: Instrumentation + listKind: InstrumentationList + plural: instrumentations + shortNames: + - nragent + - nragents + singular: instrumentation + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: Instrumentation is the Schema for the instrumentations 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: InstrumentationSpec defines the desired state of Instrumentation + properties: + dotnet: + description: DotNet defines configuration for dotnet auto-instrumentation. + properties: + env: + description: Env defines DotNet specific env vars. If the former + var had been defined, then the other vars would be ignored. + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with DotNet agent and + auto-instrumentation. + type: string + type: object + env: + description: 'Env defines common env vars. There are four layers for + env vars'' definitions and the precedence order is: `original container + env vars` > `language specific env vars` > `common env vars` > `instrument + spec configs'' vars`. If the former var had been defined, then the + other vars would be ignored.' + items: + description: EnvVar represents an environment variable present in + a Container. + properties: + name: + description: Name of the environment variable. Must be a C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded using + the previously defined environment variables in the container + and any service environment variables. If a variable cannot + be resolved, the reference in the input string will be unchanged. + Double $$ are reduced to a single $, which allows for escaping + the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the + string literal "$(VAR_NAME)". Escaped references will never + be expanded, regardless of whether the variable exists or + not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. Cannot + be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`, + spec.nodeName, spec.serviceAccountName, status.hostIP, + status.podIP, status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath is + written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified + API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the exposed + resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret or its key must + be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + exporter: + description: Exporter defines exporter configuration. + properties: + endpoint: + description: Endpoint is address of the collector with OTLP endpoint. + type: string + type: object + go: + description: Go defines configuration for Go auto-instrumentation. + When using Go auto-instrumentation you must provide a value for + the OTEL_GO_AUTO_TARGET_EXE env var via the Instrumentation env + vars or via the instrumentation.opentelemetry.io/otel-go-auto-target-exe + pod annotation. Failure to set this value causes instrumentation + injection to abort, leaving the original pod unchanged. + properties: + env: + description: 'Env defines Go specific env vars. There are four + layers for env vars'' definitions and the precedence order is: + `original container env vars` > `language specific env vars` + > `common env vars` > `instrument spec configs'' vars`. If the + former var had been defined, then the other vars would be ignored.' + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with Go SDK and auto-instrumentation. + type: string + resourceRequirements: + description: Resources describes the compute resource requirements. + properties: + claims: + description: "Claims lists the names of resources, defined + in spec.resourceClaims, that are used by this container. + \n This is an alpha field and requires enabling the DynamicResourceAllocation + feature gate. \n This field is immutable. It can only be + set for containers." + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + properties: + name: + description: Name must match the name of one entry in + pod.spec.resourceClaims of the Pod where this field + is used. It makes that resource available inside a + container. + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: 'Limits describes the maximum amount of compute + resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + 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 + volumeLimitSize: + anyOf: + - type: integer + - type: string + description: VolumeSizeLimit defines size limit for volume used + for auto-instrumentation. The default size is 200Mi. + 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 + java: + description: Java defines configuration for java auto-instrumentation. + properties: + env: + description: Env defines java specific env vars. If the former + var had been defined, then the other vars would be ignored. + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with javaagent auto-instrumentation + JAR. + type: string + type: object + nodejs: + description: NodeJS defines configuration for nodejs auto-instrumentation. + properties: + env: + description: Env defines nodejs specific env vars. If the former + var had been defined, then the other vars would be ignored. + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with NodeJS agent and + auto-instrumentation. + type: string + type: object + php: + description: Php defines configuration for php auto-instrumentation. + properties: + env: + description: Env defines Php specific env vars. If the former + var had been defined, then the other vars would be ignored. + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with Php agent and auto-instrumentation. + type: string + type: object + propagators: + description: Propagators defines inter-process context propagation + configuration. Values in this list will be set in the OTEL_PROPAGATORS + env var. Enum=tracecontext;none + items: + description: Propagator represents the propagation type. + enum: + - tracecontext + - none + type: string + type: array + python: + description: Python defines configuration for python auto-instrumentation. + properties: + env: + description: Env defines python specific env vars. If the former + var had been defined, then the other vars would be ignored. + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with Python agent and + auto-instrumentation. + type: string + type: object + ruby: + description: Ruby defines configuration for ruby auto-instrumentation. + properties: + env: + description: Env defines Ruby specific env vars. If the former + var had been defined, then the other vars would be ignored. + items: + description: EnvVar represents an environment variable present + in a Container. + properties: + name: + description: Name of the environment variable. Must be a + C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded + using the previously defined environment variables in + the container and any service environment variables. If + a variable cannot be resolved, the reference in the input + string will be unchanged. Double $$ are reduced to a single + $, which allows for escaping the $(VAR_NAME) syntax: i.e. + "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". + Escaped references will never be expanded, regardless + of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. + Cannot be used if value is not empty. + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the ConfigMap or its + key must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, + metadata.namespace, `metadata.labels['''']`, + `metadata.annotations['''']`, spec.nodeName, + spec.serviceAccountName, status.hostIP, status.podIP, + status.podIPs.' + properties: + apiVersion: + description: Version of the schema the FieldPath + is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the + specified API version. + type: string + required: + - fieldPath + type: object + x-kubernetes-map-type: atomic + resourceFieldRef: + description: 'Selects a resource of the container: only + resources limits and requests (limits.cpu, limits.memory, + limits.ephemeral-storage, requests.cpu, requests.memory + and requests.ephemeral-storage) are currently supported.' + properties: + containerName: + description: 'Container name: required for volumes, + optional for env vars' + type: string + divisor: + anyOf: + - type: integer + - type: string + description: Specifies the output format of the + exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + required: + - resource + type: object + x-kubernetes-map-type: atomic + secretKeyRef: + description: Selects a key of a secret in the pod's + namespace + properties: + key: + description: The key of the secret to select from. Must + be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names + TODO: Add other useful fields. apiVersion, kind, + uid?' + type: string + optional: + description: Specify whether the Secret or its key + must be defined + type: boolean + required: + - key + type: object + x-kubernetes-map-type: atomic + type: object + required: + - name + type: object + type: array + image: + description: Image is a container image with Ruby agent and + auto-instrumentation. + type: string + type: object + resource: + description: Resource defines the configuration for the resource attributes, + as defined by the OpenTelemetry specification. + properties: + addK8sUIDAttributes: + description: AddK8sUIDAttributes defines whether K8s UID attributes + should be collected (e.g. k8s.deployment.uid). + type: boolean + resourceAttributes: + additionalProperties: + type: string + description: 'Attributes defines attributes that are added to + the resource. For example environment: dev' + type: object + type: object + sampler: + description: Sampler defines sampling configuration. + properties: + argument: + description: Argument defines sampler argument. The value depends + on the sampler type. For instance for parentbased_traceidratio + sampler type it is a number in range [0..1] e.g. 0.25. The value + will be set in the OTEL_TRACES_SAMPLER_ARG env var. + type: string + type: + description: Type defines sampler type. The value will be set + in the OTEL_TRACES_SAMPLER env var. The value can be for instance + parentbased_always_on, parentbased_always_off, parentbased_traceidratio... + enum: + - always_on + - always_off + - traceidratio + - parentbased_always_on + - parentbased_always_off + - parentbased_traceidratio + type: string + type: object + type: object + status: + description: InstrumentationStatus defines the observed state of Instrumentation + type: object + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: null + storedVersions: null \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/leader-election-rbac.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/leader-election-rbac.yaml new file mode 100644 index 0000000000..5a42149205 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/leader-election-rbac.yaml @@ -0,0 +1,51 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-leader-election-role + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +rules: +- apiGroups: + - "" + resources: + - configmaps + verbs: + - get + - list + - watch + - create + - update + - patch + - delete +- apiGroups: + - "" + resources: + - configmaps/status + verbs: + - get + - update + - patch +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-leader-election-rolebinding + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: '{{ template "k8s-agents-operator.fullname" . }}-leader-election-role' +subjects: +- kind: ServiceAccount + name: '{{ template "k8s-agents-operator.serviceAccountName" . }}' + namespace: '{{ .Release.Namespace }}' diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/manager-rbac.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/manager-rbac.yaml new file mode 100644 index 0000000000..062b5821b7 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/manager-rbac.yaml @@ -0,0 +1,77 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-manager-role + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +rules: +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch +- apiGroups: + - "" + resources: + - namespaces + verbs: + - list + - watch +- apiGroups: + - apps + resources: + - replicasets + verbs: + - get + - list + - watch +- apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - create + - get + - list + - update +- apiGroups: + - newrelic.com + resources: + - instrumentations + verbs: + - get + - list + - patch + - update + - watch +- apiGroups: + - route.openshift.io + resources: + - routes + - routes/custom-host + verbs: + - create + - delete + - get + - list + - patch + - update + - watch +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-manager-rolebinding + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: '{{ template "k8s-agents-operator.fullname" . }}-manager-role' +subjects: +- kind: ServiceAccount + name: '{{ template "k8s-agents-operator.serviceAccountName" . }}' + namespace: '{{ .Release.Namespace }}' diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/newrelic_license_secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/newrelic_license_secret.yaml new file mode 100644 index 0000000000..00734c4cb3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/newrelic_license_secret.yaml @@ -0,0 +1,14 @@ +{{- $licenseKey := include "k8s-agents-operator.licenseKey" . -}} +{{- if $licenseKey }} +apiVersion: v1 +kind: Secret +metadata: + name: "newrelic-key-secret" + namespace: {{ .Release.Namespace }} + labels: + app.kubernetes.io/name: {{ include "k8s-agents-operator.chart" . }} + app.kubernetes.io/instance: {{ .Release.Name }} +type: Opaque +data: + new_relic_license_key: {{ $licenseKey | b64enc }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/proxy-rbac.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/proxy-rbac.yaml new file mode 100644 index 0000000000..2a9ea8667b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/proxy-rbac.yaml @@ -0,0 +1,35 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-proxy-role + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +rules: +- apiGroups: + - authentication.k8s.io + resources: + - tokenreviews + verbs: + - create +- apiGroups: + - authorization.k8s.io + resources: + - subjectaccessreviews + verbs: + - create +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-proxy-rolebinding + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: '{{ template "k8s-agents-operator.fullname" . }}-proxy-role' +subjects: +- kind: ServiceAccount + name: '{{ template "k8s-agents-operator.serviceAccountName" . }}' + namespace: '{{ .Release.Namespace }}' diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/reader-rbac.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/reader-rbac.yaml new file mode 100644 index 0000000000..4aa1d446e1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/reader-rbac.yaml @@ -0,0 +1,11 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-metrics-reader + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +rules: +- nonResourceURLs: + - /metrics + verbs: + - get diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/service.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/service.yaml new file mode 100644 index 0000000000..cfdc9d2b12 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/service.yaml @@ -0,0 +1,15 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "k8s-agents-operator.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + control-plane: controller-manager + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +spec: + type: {{ .Values.metricsService.type }} + selector: + control-plane: controller-manager + {{- include "k8s-agents-operator.labels" . | nindent 4 }} + ports: + {{- .Values.metricsService.ports | toYaml | nindent 2 -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-configuration.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-configuration.yaml new file mode 100644 index 0000000000..152ec79dc9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-configuration.yaml @@ -0,0 +1,134 @@ +{{- $tls := fromYaml (include "k8s-agents-operator.webhookCert" .) }} +{{- if .Values.admissionWebhooks.autoGenerateCert.enabled }} +apiVersion: v1 +kind: Secret +type: kubernetes.io/tls +metadata: + name: {{ template "k8s-agents-operator.certificateSecret" . }} + annotations: + "helm.sh/hook": "pre-install,pre-upgrade" + "helm.sh/hook-delete-policy": "before-hook-creation" + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} + app.kubernetes.io/component: webhook + namespace: {{ .Release.Namespace }} +data: + tls.crt: {{ $tls.clientCert }} + tls.key: {{ $tls.clientKey }} + ca.crt: {{ $tls.caCert }} +{{- end }} +--- +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-mutation + {{- if .Values.admissionWebhooks.certManager.enabled }} + annotations: + cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ template "k8s-agents-operator.fullname" . }}-serving-cert + {{- end }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +webhooks: +- admissionReviewVersions: + - v1 + clientConfig: + {{- if .Values.admissionWebhooks.autoGenerateCert.enabled }} + caBundle: {{ $tls.caCert }} + {{- end }} + service: + name: '{{ template "k8s-agents-operator.fullname" . }}-webhook-service' + namespace: '{{ .Release.Namespace }}' + path: /mutate-newrelic-com-v1alpha1-instrumentation + failurePolicy: Fail + name: instrumentation.kb.io + rules: + - apiGroups: + - newrelic.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - instrumentations + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + {{- if .Values.admissionWebhooks.autoGenerateCert.enabled }} + caBundle: {{ $tls.caCert }} + {{- end }} + service: + name: '{{ template "k8s-agents-operator.fullname" . }}-webhook-service' + namespace: '{{ .Release.Namespace }}' + path: /mutate-v1-pod + failurePolicy: Ignore + name: mpod.kb.io + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + resources: + - pods + sideEffects: None +--- +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-validation + {{- if .Values.admissionWebhooks.certManager.enabled }} + annotations: + cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ template "k8s-agents-operator.fullname" . }}-serving-cert + {{- end }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +webhooks: +- admissionReviewVersions: + - v1 + clientConfig: + {{- if .Values.admissionWebhooks.autoGenerateCert.enabled }} + caBundle: {{ $tls.caCert }} + {{- end }} + service: + name: '{{ template "k8s-agents-operator.fullname" . }}-webhook-service' + namespace: '{{ .Release.Namespace }}' + path: /validate-newrelic-com-v1alpha1-instrumentation + failurePolicy: Fail + name: vinstrumentationcreateupdate.kb.io + rules: + - apiGroups: + - newrelic.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - instrumentations + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + {{- if .Values.admissionWebhooks.autoGenerateCert.enabled }} + caBundle: {{ $tls.caCert }} + {{- end }} + service: + name: '{{ template "k8s-agents-operator.fullname" . }}-webhook-service' + namespace: '{{ .Release.Namespace }}' + path: /validate-newrelic-com-v1alpha1-instrumentation + failurePolicy: Ignore + name: vinstrumentationdelete.kb.io + rules: + - apiGroups: + - newrelic.com + apiVersions: + - v1alpha1 + operations: + - DELETE + resources: + - instrumentations + sideEffects: None diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-service.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-service.yaml new file mode 100644 index 0000000000..e5598e8f72 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/templates/webhook-service.yaml @@ -0,0 +1,14 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ template "k8s-agents-operator.fullname" . }}-webhook-service + namespace: {{ .Release.Namespace }} + labels: + {{- include "k8s-agents-operator.labels" . | nindent 4 }} +spec: + type: {{ .Values.webhookService.type }} + selector: + control-plane: controller-manager + {{- include "k8s-agents-operator.labels" . | nindent 4 }} + ports: + {{- .Values.webhookService.ports | toYaml | nindent 2 -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/cert_manager_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/cert_manager_test.yaml new file mode 100644 index 0000000000..1de2019212 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/cert_manager_test.yaml @@ -0,0 +1,85 @@ +suite: cert-manager +templates: + - templates/certmanager.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: creates cert-manager resources if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - hasDocuments: + count: 2 + - it: creates Issuer if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - equal: + path: kind + value: Issuer + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-selfsigned-issuer + - exists: + path: spec.selfSigned + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-selfsigned-issuer + - it: creates Certificate in default domain if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - equal: + path: kind + value: Certificate + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-serving-cert + - equal: + path: spec.dnsNames + value: + - my-release-k8s-agents-operator-webhook-service.my-namespace.svc + - my-release-k8s-agents-operator-webhook-service.my-namespace.svc.cluster.local + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-serving-cert + - it: creates Certificate in custom domain if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + kubernetesClusterDomain: kubey.test + asserts: + - equal: + path: kind + value: Certificate + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-serving-cert + - equal: + path: spec.dnsNames + value: + - my-release-k8s-agents-operator-webhook-service.my-namespace.svc + - my-release-k8s-agents-operator-webhook-service.my-namespace.svc.kubey.test + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-serving-cert diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/webhook_ssl_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/webhook_ssl_test.yaml new file mode 100644 index 0000000000..9343a43a43 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/tests/webhook_ssl_test.yaml @@ -0,0 +1,176 @@ +suite: webhook ssl +templates: + - templates/webhook-configuration.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: creates ssl certificate secret by default + set: + licenseKey: us-whatever + asserts: + - hasDocuments: + count: 3 + - containsDocument: + kind: Secret + apiVersion: v1 + name: my-release-k8s-agents-operator-controller-manager-service-cert + namespace: my-namespace + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-controller-manager-service-cert + - exists: + path: data["tls.crt"] + template: templates/webhook-configuration.yaml + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-controller-manager-service-cert + - exists: + path: data["tls.key"] + template: templates/webhook-configuration.yaml + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-controller-manager-service-cert + - exists: + path: data["ca.crt"] + template: templates/webhook-configuration.yaml + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-controller-manager-service-cert + - it: does not inject cert-manager annotations into MutatingWebhook by default + set: + licenseKey: us-whatever + asserts: + - notExists: + path: metadata.annotations["cert-manager.io/inject-ca-from"] + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - it: does not inject cert-manager annotations into ValidatingWebhook by default + set: + licenseKey: us-whatever + asserts: + - notExists: + path: metadata.annotations["cert-manager.io/inject-ca-from"] + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-validation + - it: does inject caBundle into MutatingWebhook clientConfigs by default + set: + licenseKey: us-whatever + asserts: + - lengthEqual: + path: webhooks + count: 2 + - exists: + path: webhooks[0].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - exists: + path: webhooks[1].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - it: does inject caBundle into ValidatingWebhook clientConfigs by default + set: + licenseKey: us-whatever + asserts: + - lengthEqual: + path: webhooks + count: 2 + - exists: + path: webhooks[0].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - exists: + path: webhooks[1].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-validation + - it: does not creates ssl certificate secret if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - hasDocuments: + count: 2 + - it: injects cert-manager annotations into MutatingWebhook if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - equal: + path: metadata.annotations["cert-manager.io/inject-ca-from"] + value: my-namespace/my-release-k8s-agents-operator-serving-cert + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - it: injects cert-manager annotations into ValidatingWebhook if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - equal: + path: metadata.annotations["cert-manager.io/inject-ca-from"] + value: my-namespace/my-release-k8s-agents-operator-serving-cert + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-validation + - it: does not inject caBundle into MutatingWebhook clientConfigs if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - lengthEqual: + path: webhooks + count: 2 + - notExists: + path: webhooks[0].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - notExists: + path: webhooks[1].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - it: does not inject caBundle into ValidatingWebhook clientConfigs if cert-manager enabled and auto cert disabled + set: + licenseKey: us-whatever + admissionWebhooks: + autoGenerateCert: + enabled: false + certManager: + enabled: true + asserts: + - lengthEqual: + path: webhooks + count: 2 + - notExists: + path: webhooks[0].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-mutation + - notExists: + path: webhooks[1].clientConfig.caBundle + documentSelector: + path: metadata.name + value: my-release-k8s-agents-operator-validation diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/values.yaml new file mode 100644 index 0000000000..f28979778a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/k8s-agents-operator/values.yaml @@ -0,0 +1,93 @@ +# Default values for k8s-agents-operator. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +# -- This set this license key to use. Can be configured also with `global.licenseKey` +licenseKey: "" + +controllerManager: + replicas: 1 + + kubeRbacProxy: + image: + repository: gcr.io/kubebuilder/kube-rbac-proxy + tag: v0.14.0 + resources: + limits: + cpu: 500m + memory: 128Mi + requests: + cpu: 5m + memory: 64Mi + + manager: + image: + repository: newrelic/k8s-agents-operator + tag: + pullPolicy: + resources: + requests: + cpu: 100m + memory: 64Mi + serviceAccount: + create: true + # -- Source: https://docs.openshift.com/container-platform/4.10/operators/operator_sdk/osdk-leader-election.html + # -- Enable leader election mechanism for protecting against split brain if multiple operator pods/replicas are started + leaderElection: + enabled: true + +kubernetesClusterDomain: cluster.local + +metricsService: + ports: + - name: https + port: 8443 + protocol: TCP + targetPort: https + type: ClusterIP + +webhookService: + ports: + - port: 443 + protocol: TCP + targetPort: 9443 + type: ClusterIP + +# -- Source: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/ +# -- SecurityContext holds pod-level security attributes and common container settings +securityContext: + runAsGroup: 65532 + runAsNonRoot: true + runAsUser: 65532 + fsGroup: 65532 + +# -- Admission webhooks make sure only requests with correctly formatted rules will get into the Operator +admissionWebhooks: + create: true + + ## TLS Certificate Option 1: Use Helm to automatically generate self-signed certificate. + ## certManager must be disabled and autoGenerateCert must be enabled. + autoGenerateCert: + # -- If true and certManager.enabled is false, Helm will automatically create a self-signed cert and secret for you. + enabled: true + # -- If set to true, new webhook key/certificate is generated on helm upgrade. + recreate: true + # -- Cert validity period time in days. + certPeriodDays: 365 + + ## TLS Certificate Option 2: Use certManager to generate self-signed certificate. + certManager: + # -- If true and autoGenerateCert.enabled is false, cert-manager will create a self-signed cert and secret for you. + enabled: false + + ## TLS Certificate Option 3: Use your own self-signed certificate. + ## certManager and autoGenerateCert must be disabled and certFile, keyFile, and caFile must be set. + ## The chart reads the contents of the file paths with the helm .Files.Get function. + ## Refer to this doc https://helm.sh/docs/chart_template_guide/accessing_files/ to understand + ## limitations of file paths accessible to the chart. + # -- Path to your own PEM-encoded certificate. + certFile: "" + # -- Path to your own PEM-encoded private key. + keyFile: "" + # -- Path to the CA cert. + caFile: "" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/.helmignore new file mode 100644 index 0000000000..f0c1319444 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/Chart.yaml new file mode 100644 index 0000000000..a86cd07e97 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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.10.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.12.1 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/README.md new file mode 100644 index 0000000000..843be89e69 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/NOTES.txt new file mode 100644 index 0000000000..3589c24ec3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/_helpers.tpl new file mode 100644 index 0000000000..a4358c87a1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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 }} +{{ 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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/ciliumnetworkpolicy.yaml new file mode 100644 index 0000000000..025cd47a88 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..cf9f628d04 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/crs-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/crs-configmap.yaml new file mode 100644 index 0000000000..d38a75a51d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/deployment.yaml new file mode 100644 index 0000000000..31aa610181 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/deployment.yaml @@ -0,0 +1,279 @@ +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 }} + 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 }} + {{- 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: {{- omit .Values.securityContext "enabled" | toYaml | nindent 8 }} + {{- end }} + {{- if .Values.priorityClassName }} + priorityClassName: {{ .Values.priorityClassName }} + {{- end }} + containers: + {{- $httpPort := 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={{ $httpPort }} + {{- 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 }} + {{- if .Values.customResourceState.enabled }} + - --custom-resource-state-config-file=/etc/customresourcestate/config.yaml + {{- end }} + {{- 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 }} + livenessProbe: + httpGet: + path: /healthz + port: {{ $httpPort }} + initialDelaySeconds: 5 + timeoutSeconds: 5 + readinessProbe: + httpGet: + path: / + port: {{ $httpPort }} + 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.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:{{ $httpPort }}/ + - --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 }} +{{- 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 }} + {{- if .Values.nodeSelector }} + nodeSelector: +{{ toYaml .Values.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: +{{ toYaml .Values.tolerations | 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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/extra-manifests.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/extra-manifests.yaml new file mode 100644 index 0000000000..567f7bf329 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/extra-manifests.yaml @@ -0,0 +1,4 @@ +{{ range .Values.extraManifests }} +--- +{{ tpl (toYaml .) $ }} +{{ end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/kubeconfig-secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/kubeconfig-secret.yaml new file mode 100644 index 0000000000..6af0084502 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/networkpolicy.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/networkpolicy.yaml new file mode 100644 index 0000000000..309b38ec54 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/pdb.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/pdb.yaml new file mode 100644 index 0000000000..3771b511de --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/podsecuritypolicy.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..8905e113e8 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/psp-clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/psp-clusterrole.yaml new file mode 100644 index 0000000000..654e4a3d57 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/psp-clusterrolebinding.yaml new file mode 100644 index 0000000000..5b62a18bdf --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/rbac-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/rbac-configmap.yaml new file mode 100644 index 0000000000..671dc9d660 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/role.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/role.yaml new file mode 100644 index 0000000000..d33687f2d1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/rolebinding.yaml new file mode 100644 index 0000000000..330651b73f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/service.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/service.yaml new file mode 100644 index 0000000000..6c486a662a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/service.yaml @@ -0,0 +1,49 @@ +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 }} + {{- 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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/serviceaccount.yaml new file mode 100644 index 0000000000..a7ff4dd3d7 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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: + {{- include "kube-state-metrics.imagePullSecrets" (dict "Values" .Values "imagePullSecrets" .Values.serviceAccount.imagePullSecrets) | indent 2 }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/servicemonitor.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/servicemonitor.yaml new file mode 100644 index 0000000000..79a07a6555 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/servicemonitor.yaml @@ -0,0 +1,114 @@ +{{- 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 }} + {{- with .Values.prometheus.monitor.annotations }} + annotations: + {{- 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 .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.prometheus.monitor.scheme }} + scheme: {{ .Values.prometheus.monitor.scheme }} + {{- end }} + {{- if .Values.prometheus.monitor.tlsConfig }} + tlsConfig: + {{- toYaml .Values.prometheus.monitor.tlsConfig | nindent 8 }} + {{- end }} + {{- if .Values.prometheus.monitor.bearerTokenFile }} + bearerTokenFile: {{ .Values.prometheus.monitor.bearerTokenFile }} + {{- end }} + {{- with .Values.prometheus.monitor.bearerTokenSecret }} + bearerTokenSecret: + {{- toYaml . | 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 }} + {{- if .Values.prometheus.monitor.scheme }} + scheme: {{ .Values.prometheus.monitor.scheme }} + {{- end }} + {{- if .Values.prometheus.monitor.tlsConfig }} + tlsConfig: + {{- toYaml .Values.prometheus.monitor.tlsConfig | nindent 8 }} + {{- end }} + {{- if .Values.prometheus.monitor.bearerTokenFile }} + bearerTokenFile: {{ .Values.prometheus.monitor.bearerTokenFile }} + {{- end }} + {{- with .Values.prometheus.monitor.bearerTokenSecret }} + bearerTokenSecret: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/stsdiscovery-role.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/stsdiscovery-role.yaml new file mode 100644 index 0000000000..489de147c1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/stsdiscovery-rolebinding.yaml new file mode 100644 index 0000000000..73b37a4f64 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/templates/verticalpodautoscaler.yaml new file mode 100644 index 0000000000..f46305b517 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/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/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/values.yaml new file mode 100644 index 0000000000..00eabab6ab --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/kube-state-metrics/values.yaml @@ -0,0 +1,441 @@ +# 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 + +# 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: [] + +service: + port: 8080 + # Default to clusterIP for backward compatibility + type: ClusterIP + 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.14.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: {} + + 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: {} + +prometheus: + monitor: + enabled: false + annotations: {} + additionalLabels: {} + namespace: "" + namespaceSelector: [] + jobLabel: "" + targetLabels: [] + podTargetLabels: [] + interval: "" + ## 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 + scrapeTimeout: "" + proxyUrl: "" + selectorOverride: {} + 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: + 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: {} + +## 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" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/.helmignore new file mode 100644 index 0000000000..f62b5519e5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/.helmignore @@ -0,0 +1 @@ +templates/admission-webhooks/job-patch/README.md diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.lock new file mode 100644 index 0000000000..9efaa36a61 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-08-30T22:48:07.029709954Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.yaml new file mode 100644 index 0000000000..18b406f840 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/Chart.yaml @@ -0,0 +1,35 @@ +apiVersion: v2 +appVersion: 0.19.4 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy the New Relic Infrastructure Kubernetes Operator. +home: https://hub.docker.com/r/newrelic/newrelic-infra-operator +icon: https://newrelic.com/themes/custom/curio/assets/mediakit/new_relic_logo_vertical.svg +keywords: +- infrastructure +- newrelic +- monitoring +maintainers: +- name: alvarocabanas + url: https://github.com/alvarocabanas +- name: carlossscastro + url: https://github.com/carlossscastro +- name: sigilioso + url: https://github.com/sigilioso +- name: gsanchezgavier + url: https://github.com/gsanchezgavier +- name: kang-makes + url: https://github.com/kang-makes +- name: marcsanmi + url: https://github.com/marcsanmi +- name: paologallinaharbur + url: https://github.com/paologallinaharbur +- name: roobre + url: https://github.com/roobre +name: newrelic-infra-operator +sources: +- https://github.com/newrelic/newrelic-infra-operator +- https://github.com/newrelic/newrelic-infra-operator/tree/main/charts/newrelic-infra-operator +version: 2.11.4 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md new file mode 100644 index 0000000000..05e8a8d48b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md @@ -0,0 +1,114 @@ +# newrelic-infra-operator + +A Helm chart to deploy the New Relic Infrastructure Kubernetes Operator. + +**Homepage:** + +## Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add newrelic-infra-operator https://newrelic.github.io/newrelic-infra-operator +helm upgrade --install newrelic-infra-operator/newrelic-infra-operator -f your-custom-values.yaml +``` + +## Source Code + +* +* + +## Usage example + +Make sure you have [added the New Relic chart repository.](../../README.md#install) + +Then, to install this chart, run the following command: + +```sh +helm upgrade --install [release-name] newrelic-infra-operator/newrelic-infra-operator --set cluster=my_cluster_name --set licenseKey [your-license-key] +``` + +When installing on Fargate add as well `--set fargate=true` + +### Configure in which pods the sidecar should be injected + +Policies are available in order to configure in which pods the sidecar should be injected. +Each policy is evaluated independently and if at least one policy matches the operator will inject the sidecar. + +Policies are composed by `namespaceSelector` checking the labels of the Pod namespace, `podSelector` checking +the labels of the Pod and `namespace` checking the namespace name. Each of those, if specified, are ANDed. + +By default, the policies are configured in order to inject the sidecar in each pod belonging to a Fargate profile. + +> Moreover, it is possible to add the label `infra-operator.newrelic.com/disable-injection` to Pods to exclude injection +for a single Pod that otherwise would be selected by the policies. + +Please make sure to configure policies correctly to avoid injecting sidecar for pods running on EC2 nodes +already monitored by the infrastructure DaemonSet. + +### Configure the sidecar with labelsSelectors + +It is also possible to configure `resourceRequirements` and `extraEnvVars` based on the labels of the mutating Pod. + +The current configuration increases the resource requirements for sidecar injected on `KSM` instances. Moreover, +injectes disable the `DISABLE_KUBE_STATE_METRICS` environment variable for Pods not running on `KSM` instances +to decrease the load on the API server. + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| admissionWebhooksPatchJob | object | See `values.yaml` | Image used to create certificates and inject them to the admission webhook | +| admissionWebhooksPatchJob.image.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| admissionWebhooksPatchJob.volumeMounts | list | `[]` | Volume mounts to add to the job, you might want to mount tmp if Pod Security Policies. Enforce a read-only root. | +| admissionWebhooksPatchJob.volumes | list | `[]` | Volumes to add to the job container. | +| affinity | object | `{}` | Sets pod/node affinities. Can be configured also with `global.affinity` | +| certManager.enabled | bool | `false` | Use cert manager for webhook certs | +| cluster | string | `""` | Name of the Kubernetes cluster monitored. Mandatory. Can be configured also with `global.cluster` | +| config | object | See `values.yaml` | Operator configuration | +| config.ignoreMutationErrors | bool | `true` | IgnoreMutationErrors instruments the operator to ignore injection error instead of failing. If set to false errors of the injection could block the creation of pods. | +| config.infraAgentInjection | object | See `values.yaml` | configuration of the sidecar injection webhook | +| config.infraAgentInjection.agentConfig | object | See `values.yaml` | agentConfig contains the configuration for the container agent injected | +| config.infraAgentInjection.agentConfig.configSelectors | list | See `values.yaml` | configSelectors is the way to configure resource requirements and extra envVars of the injected sidecar container. When mutating it will be applied the first configuration having the labelSelector matching with the mutating pod. | +| config.infraAgentInjection.agentConfig.image | object | See `values.yaml` | Image of the infrastructure agent to be injected. | +| containerSecurityContext | object | `{}` | Sets security context (at container level). Can be configured also with `global.containerSecurityContext` | +| customSecretLicenseKey | string | `""` | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` | +| customSecretName | string | `""` | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` | +| dnsConfig | object | `{}` | Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` | +| fullnameOverride | string | `""` | Override the full name of the release | +| hostNetwork | bool | `false` | Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` | +| image | object | See `values.yaml` | Image for the New Relic Infrastructure Operator | +| image.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| licenseKey | string | `""` | This set this license key to use. Can be configured also with `global.licenseKey` | +| nameOverride | string | `""` | Override the name of the chart | +| nodeSelector | object | `{}` | Sets pod's node selector. Can be configured also with `global.nodeSelector` | +| podAnnotations | object | `{}` | Annotations to add to the pod. | +| podSecurityContext | object | `{"fsGroup":1001,"runAsGroup":1001,"runAsUser":1001}` | Sets security context (at pod level). Can be configured also with `global.podSecurityContext` | +| priorityClassName | string | `""` | Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` | +| rbac.pspEnabled | bool | `false` | Whether the chart should create Pod Security Policy objects. | +| replicas | int | `1` | | +| resources | object | `{"limits":{"memory":"80M"},"requests":{"cpu":"100m","memory":"30M"}}` | Resources available for this pod | +| serviceAccount | object | See `values.yaml` | Settings controlling ServiceAccount creation | +| serviceAccount.create | bool | `true` | Specifies whether a ServiceAccount should be created | +| timeoutSeconds | int | `10` | Webhook timeout Ref: https://kubernetes.io/docs/reference/access-authn-authz/extensible-admission-controllers/#timeouts | +| tolerations | list | `[]` | Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` | + +## Maintainers + +* [alvarocabanas](https://github.com/alvarocabanas) +* [carlossscastro](https://github.com/carlossscastro) +* [sigilioso](https://github.com/sigilioso) +* [gsanchezgavier](https://github.com/gsanchezgavier) +* [kang-makes](https://github.com/kang-makes) +* [marcsanmi](https://github.com/marcsanmi) +* [paologallinaharbur](https://github.com/paologallinaharbur) +* [roobre](https://github.com/roobre) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md.gotmpl new file mode 100644 index 0000000000..1ef6033559 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/README.md.gotmpl @@ -0,0 +1,77 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +## Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add newrelic-infra-operator https://newrelic.github.io/newrelic-infra-operator +helm upgrade --install newrelic-infra-operator/newrelic-infra-operator -f your-custom-values.yaml +``` + +{{ template "chart.sourcesSection" . }} + +## Usage example + +Make sure you have [added the New Relic chart repository.](../../README.md#install) + +Then, to install this chart, run the following command: + +```sh +helm upgrade --install [release-name] newrelic-infra-operator/newrelic-infra-operator --set cluster=my_cluster_name --set licenseKey [your-license-key] +``` + +When installing on Fargate add as well `--set fargate=true` + +### Configure in which pods the sidecar should be injected + +Policies are available in order to configure in which pods the sidecar should be injected. +Each policy is evaluated independently and if at least one policy matches the operator will inject the sidecar. + +Policies are composed by `namespaceSelector` checking the labels of the Pod namespace, `podSelector` checking +the labels of the Pod and `namespace` checking the namespace name. Each of those, if specified, are ANDed. + +By default, the policies are configured in order to inject the sidecar in each pod belonging to a Fargate profile. + +> Moreover, it is possible to add the label `infra-operator.newrelic.com/disable-injection` to Pods to exclude injection +for a single Pod that otherwise would be selected by the policies. + +Please make sure to configure policies correctly to avoid injecting sidecar for pods running on EC2 nodes +already monitored by the infrastructure DaemonSet. + +### Configure the sidecar with labelsSelectors + +It is also possible to configure `resourceRequirements` and `extraEnvVars` based on the labels of the mutating Pod. + +The current configuration increases the resource requirements for sidecar injected on `KSM` instances. Moreover, +injectes disable the `DISABLE_KUBE_STATE_METRICS` environment variable for Pods not running on `KSM` instances +to decrease the load on the API server. + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/ci/test-values.yaml new file mode 100644 index 0000000000..3e154e1d46 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/ci/test-values.yaml @@ -0,0 +1,39 @@ +cluster: test-cluster +licenseKey: pleasePassCIThanks +serviceAccount: + name: newrelic-infra-operator-test +image: + repository: e2e/newrelic-infra-operator + tag: test # Defaults to AppVersion + pullPolicy: IfNotPresent + pullSecrets: + - name: test-pull-secret +admissionWebhooksPatchJob: + volumeMounts: + - name: tmp + mountPath: /tmp + volumes: + - name: tmp + emptyDir: +podAnnotations: + test-annotation: test-value +affinity: + podAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - weight: 1 + podAffinityTerm: + topologyKey: topology.kubernetes.io/zone + labelSelector: + matchExpressions: + - key: test-key + operator: In + values: + - test-value +tolerations: +- key: "key1" + operator: "Exists" + effect: "NoSchedule" +nodeSelector: + beta.kubernetes.io/os: linux + +fargate: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/NOTES.txt new file mode 100644 index 0000000000..5b11d2d833 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/NOTES.txt @@ -0,0 +1,4 @@ +Your deployment of the New Relic Infrastructure Operator is complete. +You can check on the progress of this by running the following command: + + kubectl get deployments -o wide -w --namespace {{ .Release.Namespace }} {{ include "newrelic.common.naming.fullname" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/_helpers.tpl new file mode 100644 index 0000000000..8a8858c82b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/_helpers.tpl @@ -0,0 +1,136 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Renders a value that contains template. +Usage: +{{ include "tplvalues.render" ( dict "value" .Values.path.to.the.Value "context" $) }} +*/}} +{{- define "tplvalues.render" -}} + {{- if typeIs "string" .value }} + {{- tpl .value .context }} + {{- else }} + {{- tpl (.value | toYaml) .context }} + {{- end }} +{{- end -}} + +{{- /* +Naming helpers +*/ -}} +{{- define "newrelic-infra-operator.name.admission" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "admission") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.admission" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.admission.serviceAccount" -}} +{{- if include "newrelic.common.serviceAccount.create" . -}} + {{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission") -}} +{{- else -}} + {{- include "newrelic.common.serviceAccount.name" . -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-infra-operator.name.admission-create" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "admission-create") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.admission-create" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission-create") }} +{{- end -}} + +{{- define "newrelic-infra-operator.name.admission-patch" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "admission-patch") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.admission-patch" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission-patch") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.self-signed-issuer" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "self-signed-issuer") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.root-cert" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "root-cert") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.root-issuer" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "root-issuer") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.webhook-cert" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "webhook-cert") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.infra-agent" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "infra-agent") }} +{{- end -}} + +{{- define "newrelic-infra-operator.fullname.config" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "config") }} +{{- end -}} + +{{/* +Returns Infra-agent rules +*/}} +{{- define "newrelic-infra-operator.infra-agent-monitoring-rules" -}} +- apiGroups: [""] + resources: + - "nodes" + - "nodes/metrics" + - "nodes/stats" + - "nodes/proxy" + - "pods" + - "services" + - "namespaces" + verbs: ["get", "list"] +- nonResourceURLs: ["/metrics"] + verbs: ["get"] +{{- end -}} + +{{/* +Returns fargate +*/}} +{{- define "newrelic-infra-operator.fargate" -}} +{{- if .Values.global }} + {{- if .Values.global.fargate }} + {{- .Values.global.fargate -}} + {{- end -}} +{{- else if .Values.fargate }} + {{- .Values.fargate -}} +{{- end -}} +{{- end -}} + +{{/* +Returns fargate configuration for configmap data +*/}} +{{- define "newrelic-infra-operator.fargate-config" -}} +infraAgentInjection: + resourcePrefix: {{ include "newrelic.common.naming.fullname" . }} +{{- if include "newrelic-infra-operator.fargate" . }} +{{- if not .Values.config.infraAgentInjection.policies }} + policies: + - podSelector: + matchExpressions: + - key: "eks.amazonaws.com/fargate-profile" + operator: Exists +{{- end }} + agentConfig: +{{- if not .Values.config.infraAgentInjection.agentConfig.customAttributes }} + customAttributes: + - name: computeType + defaultValue: serverless + - name: fargateProfile + fromLabel: eks.amazonaws.com/fargate-profile +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Returns configmap data +*/}} +{{- define "newrelic-infra-operator.configmap.data" -}} +{{ toYaml (merge (include "newrelic-infra-operator.fargate-config" . | fromYaml) .Values.config) }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrole.yaml new file mode 100644 index 0000000000..44c2b3eba8 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrole.yaml @@ -0,0 +1,27 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic-infra-operator.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - admissionregistration.k8s.io + resources: + - mutatingwebhookconfigurations + verbs: + - get + - update + {{- if .Values.rbac.pspEnabled }} + - apiGroups: ['policy'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ include "newrelic-infra-operator.fullname.admission" . }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrolebinding.yaml new file mode 100644 index 0000000000..902206c220 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic-infra-operator.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic-infra-operator.fullname.admission" . }} +subjects: + - kind: ServiceAccount + name: {{ include "newrelic-infra-operator.fullname.admission.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-createSecret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-createSecret.yaml new file mode 100644 index 0000000000..022e6254e6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-createSecret.yaml @@ -0,0 +1,57 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: batch/v1 +kind: Job +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.admission-create" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission-create" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + template: + metadata: + name: {{ include "newrelic-infra-operator.fullname.admission-create" . }} + labels: + app: {{ include "newrelic-infra-operator.name.admission-create" . }} + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" ( list .Values.admissionWebhooksPatchJob.image.pullSecrets ) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: create + image: {{ include "newrelic.common.images.image" ( dict "defaultRegistry" "registry.k8s.io" "imageRoot" .Values.admissionWebhooksPatchJob.image "context" .) }} + imagePullPolicy: {{ .Values.admissionWebhooksPatchJob.image.pullPolicy }} + args: + - create + - --host={{ include "newrelic.common.naming.fullname" . }},{{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }}.svc + - --namespace={{ .Release.Namespace }} + - --secret-name={{ include "newrelic-infra-operator.fullname.admission" . }} + - --cert-name=tls.crt + - --key-name=tls.key + {{- if .Values.admissionWebhooksPatchJob.image.volumeMounts }} + volumeMounts: + {{- include "tplvalues.render" ( dict "value" .Values.admissionWebhooksPatchJob.image.volumeMounts "context" $ ) | nindent 10 }} + {{- end }} + {{- if .Values.admissionWebhooksPatchJob.image.volumes }} + volumes: + {{- include "tplvalues.render" ( dict "value" .Values.admissionWebhooksPatchJob.image.volumes "context" $ ) | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + serviceAccountName: {{ include "newrelic-infra-operator.fullname.admission.serviceAccount" . }} + securityContext: + runAsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 -}} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-patchWebhook.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-patchWebhook.yaml new file mode 100644 index 0000000000..61e3636784 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/job-patchWebhook.yaml @@ -0,0 +1,57 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: batch/v1 +kind: Job +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.admission-patch" . }} + annotations: + "helm.sh/hook": post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission-patch" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + template: + metadata: + name: {{ include "newrelic-infra-operator.fullname.admission-patch" . }} + labels: + app: {{ include "newrelic-infra-operator.name.admission-patch" . }} + {{- include "newrelic.common.labels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" ( list .Values.admissionWebhooksPatchJob.image.pullSecrets ) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: patch + image: {{ include "newrelic.common.images.image" ( dict "defaultRegistry" "registry.k8s.io" "imageRoot" .Values.admissionWebhooksPatchJob.image "context" .) }} + imagePullPolicy: {{ .Values.admissionWebhooksPatchJob.image.pullPolicy }} + args: + - patch + - --webhook-name={{ include "newrelic.common.naming.fullname" . }} + - --namespace={{ .Release.Namespace }} + - --secret-name={{ include "newrelic-infra-operator.fullname.admission" . }} + - --patch-failure-policy=Ignore + - --patch-validating=false + {{- if .Values.admissionWebhooksPatchJob.image.volumeMounts }} + volumeMounts: + {{- include "tplvalues.render" ( dict "value" .Values.admissionWebhooksPatchJob.image.volumeMounts "context" $ ) | nindent 10 }} + {{- end }} + {{- if .Values.admissionWebhooksPatchJob.image.volumes }} + volumes: + {{- include "tplvalues.render" ( dict "value" .Values.admissionWebhooksPatchJob.image.volumes "context" $ ) | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + serviceAccountName: {{ include "newrelic-infra-operator.fullname.admission.serviceAccount" . }} + securityContext: + runAsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 -}} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/psp.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/psp.yaml new file mode 100644 index 0000000000..64237abb4d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/psp.yaml @@ -0,0 +1,50 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled) (.Values.rbac.pspEnabled)) }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "newrelic-infra-operator.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + privileged: false + # Required to prevent escalations to root. + # allowPrivilegeEscalation: false + # This is redundant with non-root + disallow privilege escalation, + # but we can provide it for defense in depth. + # requiredDropCapabilities: + # - ALL + # Allow core volume types. + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} + hostIPC: false + hostPID: false + 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: + # Forbid adding the root group. + - min: 0 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 0 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/role.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/role.yaml new file mode 100644 index 0000000000..e3213f7c5e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/role.yaml @@ -0,0 +1,21 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - secrets + verbs: + - get + - create +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/rolebinding.yaml new file mode 100644 index 0000000000..67eb792987 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/rolebinding.yaml @@ -0,0 +1,21 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "newrelic-infra-operator.fullname.admission" . }} +subjects: + - kind: ServiceAccount + name: {{ include "newrelic-infra-operator.fullname.admission.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/serviceaccount.yaml new file mode 100644 index 0000000000..18eb7347df --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/job-patch/serviceaccount.yaml @@ -0,0 +1,14 @@ +{{- $createServiceAccount := include "newrelic.common.serviceAccount.create" . -}} +{{- if (and $createServiceAccount (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.admission.serviceAccount" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic-infra-operator.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/mutatingWebhookConfiguration.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/mutatingWebhookConfiguration.yaml new file mode 100644 index 0000000000..efa6052555 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/admission-webhooks/mutatingWebhookConfiguration.yaml @@ -0,0 +1,32 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} +{{- if .Values.certManager.enabled }} + annotations: + certmanager.k8s.io/inject-ca-from: {{ printf "%s/%s-root-cert" .Release.Namespace (include "newrelic.common.naming.fullname" .) | quote }} + cert-manager.io/inject-ca-from: {{ printf "%s/%s-root-cert" .Release.Namespace (include "newrelic.common.naming.fullname" .) | quote }} +{{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +webhooks: +- name: newrelic-infra-operator.newrelic.com + clientConfig: + service: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + path: "/mutate-v1-pod" +{{- if not .Values.certManager.enabled }} + caBundle: "" +{{- end }} + rules: + - operations: ["CREATE"] + apiGroups: [""] + apiVersions: ["v1"] + resources: ["pods"] + failurePolicy: Ignore + timeoutSeconds: {{ .Values.timeoutSeconds }} + sideEffects: NoneOnDryRun + admissionReviewVersions: + - v1 + reinvocationPolicy: IfNeeded diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/cert-manager.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/cert-manager.yaml new file mode 100644 index 0000000000..800dc24530 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/cert-manager.yaml @@ -0,0 +1,52 @@ +{{ if .Values.certManager.enabled }} +--- +# Create a selfsigned Issuer, in order to create a root CA certificate for +# signing webhook serving certificates +apiVersion: cert-manager.io/v1 +kind: Issuer +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.self-signed-issuer" . }} +spec: + selfSigned: {} +--- +# Generate a CA Certificate used to sign certificates for the webhook +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.root-cert" . }} +spec: + secretName: {{ include "newrelic-infra-operator.fullname.root-cert" . }} + duration: 43800h # 5y + issuerRef: + name: {{ include "newrelic-infra-operator.fullname.self-signed-issuer" . }} + commonName: "ca.webhook.nri" + isCA: true +--- +# Create an Issuer that uses the above generated CA certificate to issue certs +apiVersion: cert-manager.io/v1 +kind: Issuer +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.root-issuer" . }} +spec: + ca: + secretName: {{ include "newrelic-infra-operator.fullname.root-cert" . }} +--- +# Finally, generate a serving certificate for the webhook to use +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.webhook-cert" . }} +spec: + secretName: {{ include "newrelic-infra-operator.fullname.admission" . }} + duration: 8760h # 1y + issuerRef: + name: {{ include "newrelic-infra-operator.fullname.root-issuer" . }} + dnsNames: + - {{ include "newrelic.common.naming.fullname" . }} + - {{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }} + - {{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }}.svc +{{ end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrole.yaml new file mode 100644 index 0000000000..cb20e310db --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrole.yaml @@ -0,0 +1,39 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + {{/* Allow creating and updating secrets with license key for infra agent. */ -}} + - apiGroups: [""] + resources: + - "secrets" + verbs: ["get", "update", "patch"] + resourceNames: [ {{ include "newrelic-infra-operator.fullname.config" . | quote }} ] + {{/* resourceNames used above do not support "create" verb. */ -}} + - apiGroups: [""] + resources: + - "secrets" + verbs: ["create"] + {{/* "list" and "watch" are required for controller-runtime caching. */ -}} + - apiGroups: ["rbac.authorization.k8s.io"] + resources: ["clusterrolebindings"] + verbs: ["list", "watch", "get"] + {{/* Our controller needs permission to add the ServiceAccounts from the user to the -infra-agent CRB. */ -}} + - apiGroups: ["rbac.authorization.k8s.io"] + resources: ["clusterrolebindings"] + verbs: ["update"] + resourceNames: [ {{ include "newrelic-infra-operator.fullname.infra-agent" . | quote }} ] + {{- /* Controller must have permissions it will grant to other ServiceAccounts. */ -}} + {{- include "newrelic-infra-operator.infra-agent-monitoring-rules" . | nindent 2 }} +--- +{{/* infra-agent is the ClusterRole to be used by the injected agents to get metrics */}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic-infra-operator.fullname.infra-agent" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + {{- include "newrelic-infra-operator.infra-agent-monitoring-rules" . | nindent 2 }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..1f5f8b89ba --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/clusterrolebinding.yaml @@ -0,0 +1,26 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic.common.naming.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ template "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +--- +{{/* infra-agent is the ClusterRoleBinding to be used by the ServiceAccounts of the injected agents */}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic-infra-operator.fullname.infra-agent" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic-infra-operator.fullname.infra-agent" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/configmap.yaml new file mode 100644 index 0000000000..fdb4a1e3ba --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/configmap.yaml @@ -0,0 +1,9 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-infra-operator.fullname.config" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + operator.yaml: {{- include "newrelic-infra-operator.configmap.data" . | toYaml | nindent 4 }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/deployment.yaml new file mode 100644 index 0000000000..40f3898873 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/deployment.yaml @@ -0,0 +1,92 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + replicas: {{ .Values.replicas }} + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + checksum/secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- if .Values.podAnnotations }} + {{- toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + serviceAccountName: {{ template "newrelic.common.serviceAccount.name" . }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" ( list .Values.image.pullSecrets ) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: {{ include "newrelic.common.naming.name" . }} + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.image "context" .) }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 10 }} + {{- end }} + env: + - name: CLUSTER_NAME + value: {{ include "newrelic.common.cluster" . }} + - name: NRIA_LICENSE_KEY + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + volumeMounts: + - name: config + mountPath: /etc/newrelic/newrelic-infra-operator/ + - name: tls-key-cert-pair + mountPath: /tmp/k8s-webhook-server/serving-certs/ + readinessProbe: + httpGet: + path: /healthz + port: 9440 + initialDelaySeconds: 1 + periodSeconds: 1 + {{- if .Values.resources }} + resources: + {{- toYaml .Values.resources | nindent 10 }} + {{- end }} + volumes: + - name: config + configMap: + name: {{ include "newrelic-infra-operator.fullname.config" . }} + - name: tls-key-cert-pair + secret: + secretName: {{ include "newrelic-infra-operator.fullname.admission" . }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 -}} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 8 -}} + {{- end }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} + {{- if include "newrelic.common.hostNetwork" . }} + dnsPolicy: ClusterFirstWithHostNet + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/secret.yaml new file mode 100644 index 0000000000..f558ee86c9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/secret.yaml @@ -0,0 +1,2 @@ +{{- /* Common library will take care of creating the secret or not. */}} +{{- include "newrelic.common.license.secret" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/service.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/service.yaml new file mode 100644 index 0000000000..04af4d09c6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/service.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +kind: Service +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + ports: + - port: 443 + targetPort: 9443 + selector: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 4 }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/serviceaccount.yaml new file mode 100644 index 0000000000..b1e74523e5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- if include "newrelic.common.serviceAccount.annotations" . }} + annotations: + {{- include "newrelic.common.serviceAccount.annotations" . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/deployment_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/deployment_test.yaml new file mode 100644 index 0000000000..a1ffa88d05 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/deployment_test.yaml @@ -0,0 +1,32 @@ +suite: test cluster environment variable setup +templates: + - templates/deployment.yaml + - templates/configmap.yaml + - templates/secret.yaml +release: + name: my-release + namespace: my-namespac +tests: + - it: has a linux node selector by default + set: + cluster: my-cluster + licenseKey: use-whatever + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + template: templates/deployment.yaml + - it: has a linux node selector and additional selectors + set: + cluster: my-cluster + licenseKey: use-whatever + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue + template: templates/deployment.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_patch_psp_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_patch_psp_test.yaml new file mode 100644 index 0000000000..78f1b1f6af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_patch_psp_test.yaml @@ -0,0 +1,23 @@ +suite: test rendering for PSPs +templates: + - templates/admission-webhooks/job-patch/psp.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: If PSPs are enabled PodSecurityPolicy is rendered + set: + cluster: test-cluster + licenseKey: use-whatever + rbac: + pspEnabled: true + asserts: + - hasDocuments: + count: 1 + - it: If PSPs are disabled PodSecurityPolicy isn't rendered + set: + cluster: test-cluster + licenseKey: use-whatever + asserts: + - hasDocuments: + count: 0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_serviceaccount_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_serviceaccount_test.yaml new file mode 100644 index 0000000000..c6acda2db4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/job_serviceaccount_test.yaml @@ -0,0 +1,64 @@ +suite: test job' serviceAccount +templates: + - templates/admission-webhooks/job-patch/job-createSecret.yaml + - templates/admission-webhooks/job-patch/job-patchWebhook.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: RBAC points to the service account that is created by default + set: + cluster: test-cluster + licenseKey: use-whatever + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: my-release-newrelic-infra-operator-admission + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + licenseKey: use-whatever + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: sa-test + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + licenseKey: use-whatever + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: default + + - it: has a linux node selector by default + set: + cluster: my-cluster + licenseKey: use-whatever + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + + - it: has a linux node selector and additional selectors + set: + cluster: my-cluster + licenseKey: use-whatever + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/rbac_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/rbac_test.yaml new file mode 100644 index 0000000000..03473cb393 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/tests/rbac_test.yaml @@ -0,0 +1,41 @@ +suite: test RBAC creation +templates: + - templates/admission-webhooks/job-patch/rolebinding.yaml + - templates/admission-webhooks/job-patch/clusterrolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: RBAC points to the service account that is created by default + set: + cluster: test-cluster + licenseKey: use-whatever + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: subjects[0].name + value: my-release-newrelic-infra-operator-admission + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + licenseKey: use-whatever + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: subjects[0].name + value: sa-test + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + licenseKey: use-whatever + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: subjects[0].name + value: default diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/values.yaml new file mode 100644 index 0000000000..3dd6fd0554 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infra-operator/values.yaml @@ -0,0 +1,222 @@ +# -- Override the name of the chart +nameOverride: "" +# -- Override the full name of the release +fullnameOverride: "" + +# -- Name of the Kubernetes cluster monitored. Mandatory. Can be configured also with `global.cluster` +cluster: "" +# -- This set this license key to use. Can be configured also with `global.licenseKey` +licenseKey: "" +# -- In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` +customSecretName: "" +# -- In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` +customSecretLicenseKey: "" + +# -- Image for the New Relic Infrastructure Operator +# @default -- See `values.yaml` +image: + repository: newrelic/newrelic-infra-operator + tag: "" + pullPolicy: IfNotPresent + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + +# -- Image used to create certificates and inject them to the admission webhook +# @default -- See `values.yaml` +admissionWebhooksPatchJob: + image: + registry: # Defaults to registry.k8s.io + repository: ingress-nginx/kube-webhook-certgen + tag: v1.3.0 + pullPolicy: IfNotPresent + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + + # -- Volume mounts to add to the job, you might want to mount tmp if Pod Security Policies. + # Enforce a read-only root. + volumeMounts: [] + # - name: tmp + # mountPath: /tmp + # -- Volumes to add to the job container. + volumes: [] + # - name: tmp + # emptyDir: {} + +rbac: + # rbac.pspEnabled -- Whether the chart should create Pod Security Policy objects. + pspEnabled: false + +replicas: 1 + +# -- Resources available for this pod +resources: + limits: + memory: 80M + requests: + cpu: 100m + memory: 30M + +# -- Settings controlling ServiceAccount creation +# @default -- See `values.yaml` +serviceAccount: + # serviceAccount.create -- (bool) Specifies whether a ServiceAccount should be created + # @default -- `true` + create: + # If not set and create is true, a name is generated using the fullname template + name: "" + # Specify any annotations to add to the ServiceAccount + annotations: + +# -- Annotations to add to the pod. +podAnnotations: {} + +# -- Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` +priorityClassName: "" +# -- (bool) Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` +# @default -- `false` +hostNetwork: +# -- Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` +dnsConfig: {} +# -- Sets security context (at pod level). Can be configured also with `global.podSecurityContext` +podSecurityContext: + fsGroup: 1001 + runAsUser: 1001 + runAsGroup: 1001 +# -- Sets security context (at container level). Can be configured also with `global.containerSecurityContext` +containerSecurityContext: {} + +# -- Sets pod/node affinities. Can be configured also with `global.affinity` +affinity: {} +# -- Sets pod's node selector. Can be configured also with `global.nodeSelector` +nodeSelector: {} +# -- Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` +tolerations: [] + +certManager: + # certManager.enabled -- Use cert manager for webhook certs + enabled: false + +# -- Webhook timeout +# Ref: https://kubernetes.io/docs/reference/access-authn-authz/extensible-admission-controllers/#timeouts +timeoutSeconds: 10 + +# -- Operator configuration +# @default -- See `values.yaml` +config: + # -- IgnoreMutationErrors instruments the operator to ignore injection error instead of failing. + # If set to false errors of the injection could block the creation of pods. + ignoreMutationErrors: true + + # -- configuration of the sidecar injection webhook + # @default -- See `values.yaml` + infraAgentInjection: +# policies: +# - podSelector: +# matchExpressions: +# - key: app +# operator: In +# values: [ "nginx-sidecar" ] +# + # All policies are ORed, if one policy matches the sidecar is injected. + # Within a policy PodSelectors, NamespaceSelector and NamespaceName are ANDed, any of these, if not specified, is ignored. + # The following policy is injected if global.fargate=true and matches all pods belonging to any fargate profile. + # policies: + # - podSelector: + # matchExpressions: + # - key: "eks.amazonaws.com/fargate-profile" + # operator: Exists + # Also NamespaceName and NamespaceSelector can be leveraged. + # namespaceName: "my-namespace" + # namespaceSelector: {} + + # -- agentConfig contains the configuration for the container agent injected + # @default -- See `values.yaml` + agentConfig: + # Custom Attributes allows to pass any custom attribute to the injected infra agents. + # The value is computed either from the defaultValue or taken at injected time from Label specified in "fromLabel". + # Either the label should exist or the default should be specified in order to have the injection working. + # customAttributes: + # - name: computeType + # defaultValue: serverless + # - name: fargateProfile + # fromLabel: eks.amazonaws.com/fargate-profile + + # -- Image of the infrastructure agent to be injected. + # @default -- See `values.yaml` + image: + repository: newrelic/infrastructure-k8s + tag: 2.13.15-unprivileged + pullPolicy: IfNotPresent + + # -- configSelectors is the way to configure resource requirements and extra envVars of the injected sidecar container. + # When mutating it will be applied the first configuration having the labelSelector matching with the mutating pod. + # @default -- See `values.yaml` + configSelectors: + - resourceRequirements: # resourceRequirements to apply to the injected sidecar. + limits: + memory: 100M + cpu: 200m + requests: + memory: 50M + cpu: 100m + extraEnvVars: # extraEnvVars to pass to the injected sidecar. + DISABLE_KUBE_STATE_METRICS: "true" + # NRIA_VERBOSE: "1" + labelSelector: + matchExpressions: + - key: "app.kubernetes.io/name" + operator: NotIn + values: ["kube-state-metrics"] + - key: "app" + operator: NotIn + values: ["kube-state-metrics"] + - key: "k8s-app" + operator: NotIn + values: ["kube-state-metrics"] + + - resourceRequirements: + limits: + memory: 300M + cpu: 300m + requests: + memory: 150M + cpu: 150m + labelSelector: + matchLabels: + k8s-app: kube-state-metrics + # extraEnvVars: + # NRIA_VERBOSE: "1" + + - resourceRequirements: + limits: + memory: 300M + cpu: 300m + requests: + memory: 150M + cpu: 150m + labelSelector: + matchLabels: + app: kube-state-metrics + # extraEnvVars: + # NRIA_VERBOSE: "1" + + - resourceRequirements: + limits: + memory: 300M + cpu: 300m + requests: + memory: 150M + cpu: 150m + labelSelector: + matchLabels: + app.kubernetes.io/name: kube-state-metrics + # extraEnvVars: + # NRIA_VERBOSE: "1" + + # pod Security Context of the sidecar injected. + # Notice that ReadOnlyRootFilesystem and AllowPrivilegeEscalation enforced respectively to true and to false. + # podSecurityContext: + # RunAsUser: + # RunAsGroup: diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/.helmignore new file mode 100644 index 0000000000..2bfa6a4d95 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/.helmignore @@ -0,0 +1 @@ +tests/ diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.lock new file mode 100644 index 0000000000..51857821fb --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-08-30T23:46:25.952459233Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.yaml new file mode 100644 index 0000000000..2c211355b5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/Chart.yaml @@ -0,0 +1,26 @@ +apiVersion: v2 +appVersion: 3.29.6 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy the New Relic Kubernetes monitoring solution +home: https://docs.newrelic.com/docs/kubernetes-pixie/kubernetes-integration/get-started/introduction-kubernetes-integration/ +icon: https://newrelic.com/themes/custom/curio/assets/mediakit/NR_logo_Horizontal.svg +keywords: +- infrastructure +- newrelic +- monitoring +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: newrelic-infrastructure +sources: +- https://github.com/newrelic/nri-kubernetes/ +- https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure +- https://github.com/newrelic/infrastructure-agent/ +version: 3.34.6 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md new file mode 100644 index 0000000000..923b6109b1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md @@ -0,0 +1,220 @@ +# newrelic-infrastructure + +A Helm chart to deploy the New Relic Kubernetes monitoring solution + +**Homepage:** + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-kubernetes https://newrelic.github.io/nri-kubernetes +helm upgrade --install newrelic-infrastructure nri-kubernetes/newrelic-infrastructure -f your-custom-values.yaml +``` + +## Source Code + +* +* +* + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Chart particularities + +### Low data mode +There are two mechanisms to reduce the amount of data that this integration sends to New Relic. See this snippet from the `values.yaml` file: +```yaml +common: + config: + interval: 15s + +lowDataMode: false +``` + +The `lowDataMode` toggle is the simplest way to reduce data send to Newrelic. Setting it to `true` changes the default scrape interval from 15 seconds +(the default) to 30 seconds. + +If you need for some reason to fine-tune the number of seconds you can use `common.config.interval` directly. If you take a look at the `values.yaml` +file, the value there is `nil`. If any value is set there, the `lowDataMode` toggle is ignored as this value takes precedence. + +Setting this interval above 40 seconds can make you experience issues with the Kubernetes Cluster Explorer so this chart limits setting the interval +inside the range of 10 to 40 seconds. + +### Affinities and tolerations + +The New Relic common library allows to set affinities, tolerations, and node selectors globally using e.g. `.global.affinity` to ease the configuration +when you use this chart using `nri-bundle`. This chart has an extra level of granularity to the components that it deploys: +control plane, ksm, and kubelet. + +Take this snippet as an example: +```yaml +global: + affinity: {} +affinity: {} + +kubelet: + affinity: {} +ksm: + affinity: {} +controlPlane: + affinity: {} +``` + +The order to set an affinity is to set first any `kubelet.affinity`, `ksm.affinity`, or `controlPlane.affinity`. If these values are empty the chart +fallbacks to `affinity` (at root level), and if that value is empty, the chart fallbacks to `global.affinity`. + +The same procedure applies to `nodeSelector` and `tolerations`. + +On the other hand, some components have affinities and tolerations predefined e.g. to be able to run kubelet pods on nodes that are tainted as master +nodes or to schedule the KSM scraper on the same node of KSM to reduce the inter-node traffic. + +If you are having problems assigning pods to nodes it may be because of this. Take a look at the [`values.yaml`](values.yaml) to see if the pod that is +not having your expected behavior has any predefined value. + +### `hostNetwork` toggle + +In versions below v3, changing the `privileged` mode affected the `hostNetwork`. We changed this behavior and now you can set pods to use `hostNetwork` +using the corresponding [flags from the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md) +(`.global.hostNetwork` and `.hostNetwork`) but the component that scrapes data from the control plane has always set `hostNetwork` enabled by default +(Look in the [`values.yaml`](values.yaml) for `controlPlane.hostNetwork: true`) + +This is because the most common configuration of the control plane components is to be configured to listen only to `localhost`. + +If your cluster security policy does not allow to use `hostNetwork`, you can disable it control plane monitoring by setting `controlPlane.enabled` to +`false.` + +### `privileged` toggle + +The default value for `privileged` [from the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md) is +`false` but in this particular this chart it is set to `true` (Look in the [`values.yaml`](values.yaml) for `privileged: true`) + +This is because when `kubelet` pods need to run in privileged mode to fetch cpu, memory, process, and network metrics of your nodes. + +If your cluster security policy does not allow to have `privileged` in your pod' security context, you can disable it by setting `privileged` to +`false` taking into account that you will lose all the metrics from the host and some metadata from the host that are added to the metrics of the +integrations that you have configured. + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Sets pod/node affinities set almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) | +| cluster | string | `""` | Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster` | +| common | object | See `values.yaml` | Config that applies to all instances of the solution: kubelet, ksm, control plane and sidecars. | +| common.agentConfig | object | `{}` | Config for the Infrastructure agent. Will be used by the forwarder sidecars and the agent running integrations. See: https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ | +| common.config.interval | duration | `15s` (See [Low data mode](README.md#low-data-mode)) | Intervals larger than 40s are not supported and will cause the NR UI to not behave properly. Any non-nil value will override the `lowDataMode` default. | +| common.config.namespaceSelector | object | `{}` | Config for filtering ksm and kubelet metrics by namespace. | +| containerSecurityContext | object | `{}` | Sets security context (at container level). Can be configured also with `global.containerSecurityContext` | +| controlPlane | object | See `values.yaml` | Configuration for the control plane scraper. | +| controlPlane.affinity | object | Deployed only in master nodes. | Affinity for the control plane DaemonSet. | +| controlPlane.agentConfig | object | `{}` | Config for the Infrastructure agent that will forward the metrics to the backend. It will be merged with the configuration in `.common.agentConfig` See: https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ | +| controlPlane.config.apiServer | object | Common settings for most K8s distributions. | API Server monitoring configuration | +| controlPlane.config.apiServer.enabled | bool | `true` | Enable API Server monitoring | +| controlPlane.config.controllerManager | object | Common settings for most K8s distributions. | Controller manager monitoring configuration | +| controlPlane.config.controllerManager.enabled | bool | `true` | Enable controller manager monitoring. | +| controlPlane.config.etcd | object | Common settings for most K8s distributions. | etcd monitoring configuration | +| controlPlane.config.etcd.enabled | bool | `true` | Enable etcd monitoring. Might require manual configuration in some environments. | +| controlPlane.config.retries | int | `3` | Number of retries after timeout expired | +| controlPlane.config.scheduler | object | Common settings for most K8s distributions. | Scheduler monitoring configuration | +| controlPlane.config.scheduler.enabled | bool | `true` | Enable scheduler monitoring. | +| controlPlane.config.timeout | string | `"10s"` | Timeout for the Kubernetes APIs contacted by the integration | +| controlPlane.enabled | bool | `true` | Deploy control plane monitoring component. | +| controlPlane.hostNetwork | bool | `true` | Run Control Plane scraper with `hostNetwork`. `hostNetwork` is required for most control plane configurations, as they only accept connections from localhost. | +| controlPlane.kind | string | `"DaemonSet"` | How to deploy the control plane scraper. If autodiscovery is in use, it should be `DaemonSet`. Advanced users using static endpoints set this to `Deployment` to avoid reporting metrics twice. | +| controlPlane.tolerations | list | Schedules in all tainted nodes | Tolerations for the control plane DaemonSet. | +| customAttributes | object | `{}` | Adds extra attributes to the cluster and all the metrics emitted to the backend. Can be configured also with `global.customAttributes` | +| customSecretLicenseKey | string | `""` | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` | +| customSecretName | string | `""` | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` | +| dnsConfig | object | `{}` | Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` | +| enableProcessMetrics | bool | `false` | Collect detailed metrics from processes running in the host. This defaults to true for accounts created before July 20, 2020. ref: https://docs.newrelic.com/docs/release-notes/infrastructure-release-notes/infrastructure-agent-release-notes/new-relic-infrastructure-agent-1120 | +| fedramp.enabled | bool | `false` | Enables FedRAMP. Can be configured also with `global.fedramp.enabled` | +| fullnameOverride | string | `""` | Override the full name of the release | +| hostNetwork | bool | `false` | Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` | +| images | object | See `values.yaml` | Images used by the chart for the integration and agents. | +| images.agent | object | See `values.yaml` | Image for the New Relic Infrastructure Agent plus integrations. | +| images.forwarder | object | See `values.yaml` | Image for the New Relic Infrastructure Agent sidecar. | +| images.integration | object | See `values.yaml` | Image for the New Relic Kubernetes integration. | +| images.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| integrations | object | `{}` | Config files for other New Relic integrations that should run in this cluster. | +| ksm | object | See `values.yaml` | Configuration for the Deployment that collects state metrics from KSM (kube-state-metrics). | +| ksm.affinity | object | Deployed in the same node as KSM | Affinity for the KSM Deployment. | +| ksm.agentConfig | object | `{}` | Config for the Infrastructure agent that will forward the metrics to the backend. It will be merged with the configuration in `.common.agentConfig` See: https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ | +| ksm.config.retries | int | `3` | Number of retries after timeout expired | +| ksm.config.scheme | string | `"http"` | Scheme to use to connect to kube-state-metrics. Supported values are `http` and `https`. | +| ksm.config.selector | string | `"app.kubernetes.io/name=kube-state-metrics"` | Label selector that will be used to automatically discover an instance of kube-state-metrics running in the cluster. | +| ksm.config.timeout | string | `"10s"` | Timeout for the ksm API contacted by the integration | +| ksm.enabled | bool | `true` | Enable cluster state monitoring. Advanced users only. Setting this to `false` is not supported and will break the New Relic experience. | +| ksm.hostNetwork | bool | Not set | Sets pod's hostNetwork. When set bypasses global/common variable | +| ksm.resources | object | 100m/150M -/850M | Resources for the KSM scraper pod. Keep in mind that sharding is not supported at the moment, so memory usage for this component ramps up quickly on large clusters. | +| ksm.tolerations | list | Schedules in all tainted nodes | Tolerations for the KSM Deployment. | +| kubelet | object | See `values.yaml` | Configuration for the DaemonSet that collects metrics from the Kubelet. | +| kubelet.agentConfig | object | `{}` | Config for the Infrastructure agent that will forward the metrics to the backend and will run the integrations in this cluster. It will be merged with the configuration in `.common.agentConfig`. You can see all the agent configurations in [New Relic docs](https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/) e.g. you can set `passthrough_environment` int the [config file](https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/configure-infrastructure-agent/#config-file) so the agent let use that environment variables to the integrations. | +| kubelet.config.retries | int | `3` | Number of retries after timeout expired | +| kubelet.config.scraperMaxReruns | int | `4` | Max number of scraper rerun when scraper runtime error happens | +| kubelet.config.timeout | string | `"10s"` | Timeout for the kubelet APIs contacted by the integration | +| kubelet.enabled | bool | `true` | Enable kubelet monitoring. Advanced users only. Setting this to `false` is not supported and will break the New Relic experience. | +| kubelet.extraEnv | list | `[]` | Add user environment variables to the agent | +| kubelet.extraEnvFrom | list | `[]` | Add user environment from configMaps or secrets as variables to the agent | +| kubelet.extraVolumeMounts | list | `[]` | Defines where to mount volumes specified with `extraVolumes` | +| kubelet.extraVolumes | list | `[]` | Volumes to mount in the containers | +| kubelet.hostNetwork | bool | Not set | Sets pod's hostNetwork. When set bypasses global/common variable | +| kubelet.tolerations | list | Schedules in all tainted nodes | Tolerations for the control plane DaemonSet. | +| labels | object | `{}` | Additional labels for chart objects. Can be configured also with `global.labels` | +| licenseKey | string | `""` | This set this license key to use. Can be configured also with `global.licenseKey` | +| lowDataMode | bool | `false` (See [Low data mode](README.md#low-data-mode)) | Send less data by incrementing the interval from `15s` (the default when `lowDataMode` is `false` or `nil`) to `30s`. Non-nil values of `common.config.interval` will override this value. | +| nameOverride | string | `""` | Override the name of the chart | +| nodeSelector | object | `{}` | Sets pod's node selector almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) | +| nrStaging | bool | `false` | Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` | +| podAnnotations | object | `{}` | Annotations to be added to all pods created by the integration. | +| podLabels | object | `{}` | Additional labels for chart pods. Can be configured also with `global.podLabels` | +| podSecurityContext | object | `{}` | Sets security context (at pod level). Can be configured also with `global.podSecurityContext` | +| priorityClassName | string | `""` | Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` | +| privileged | bool | `true` | Run the integration with full access to the host filesystem and network. Running in this mode allows reporting fine-grained cpu, memory, process and network metrics for your nodes. | +| proxy | string | `""` | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port`. Can be configured also with `global.proxy` | +| rbac.create | bool | `true` | Whether the chart should automatically create the RBAC objects required to run. | +| rbac.pspEnabled | bool | `false` | Whether the chart should create Pod Security Policy objects. | +| selfMonitoring.pixie.enabled | bool | `false` | Enables the Pixie Health Check nri-flex config. This Flex config performs periodic checks of the Pixie /healthz and /statusz endpoints exposed by the Pixie Cloud Connector. A status for each endpoint is sent to New Relic in a pixieHealthCheck event. | +| serviceAccount | object | See `values.yaml` | Settings controlling ServiceAccount creation. | +| serviceAccount.create | bool | `true` | Whether the chart should automatically create the ServiceAccount objects required to run. | +| sink.http.probeBackoff | string | `"5s"` | The amount of time the scraper container to backoff when it fails to probe infra agent sidecar. | +| sink.http.probeTimeout | string | `"90s"` | The amount of time the scraper container to probe infra agent sidecar container before giving up and restarting during pod starts. | +| strategy | object | `type: Recreate` | Update strategy for the deployed Deployments. | +| tolerations | list | `[]` | Sets pod's tolerations to node taints almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) | +| updateStrategy | object | See `values.yaml` | Update strategy for the deployed DaemonSets. | +| verboseLog | bool | `false` | Sets the debug logs to this integration or all integrations if it is set globally. Can be configured also with `global.verboseLog` | + +## Maintainers + +* [juanjjaramillo](https://github.com/juanjjaramillo) +* [csongnr](https://github.com/csongnr) +* [dbudziwojskiNR](https://github.com/dbudziwojskiNR) + +## Past Contributors + +Previous iterations of this chart started as a community project in the [stable Helm chart repository](github.com/helm/charts/). New Relic is very thankful for all the 15+ community members that contributed and helped maintain the chart there over the years: + +* coreypobrien +* sstarcher +* jmccarty3 +* slayerjain +* ryanhope2 +* rk295 +* michaelajr +* isindir +* idirouhab +* ismferd +* enver +* diclophis +* jeffdesc +* costimuraru +* verwilst +* ezelenka diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md.gotmpl new file mode 100644 index 0000000000..84f2f90834 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/README.md.gotmpl @@ -0,0 +1,137 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-kubernetes https://newrelic.github.io/nri-kubernetes +helm upgrade --install newrelic-infrastructure nri-kubernetes/newrelic-infrastructure -f your-custom-values.yaml +``` + +{{ template "chart.sourcesSection" . }} + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Chart particularities + +### Low data mode +There are two mechanisms to reduce the amount of data that this integration sends to New Relic. See this snippet from the `values.yaml` file: +```yaml +common: + config: + interval: 15s + +lowDataMode: false +``` + +The `lowDataMode` toggle is the simplest way to reduce data send to Newrelic. Setting it to `true` changes the default scrape interval from 15 seconds +(the default) to 30 seconds. + +If you need for some reason to fine-tune the number of seconds you can use `common.config.interval` directly. If you take a look at the `values.yaml` +file, the value there is `nil`. If any value is set there, the `lowDataMode` toggle is ignored as this value takes precedence. + +Setting this interval above 40 seconds can make you experience issues with the Kubernetes Cluster Explorer so this chart limits setting the interval +inside the range of 10 to 40 seconds. + +### Affinities and tolerations + +The New Relic common library allows to set affinities, tolerations, and node selectors globally using e.g. `.global.affinity` to ease the configuration +when you use this chart using `nri-bundle`. This chart has an extra level of granularity to the components that it deploys: +control plane, ksm, and kubelet. + +Take this snippet as an example: +```yaml +global: + affinity: {} +affinity: {} + +kubelet: + affinity: {} +ksm: + affinity: {} +controlPlane: + affinity: {} +``` + +The order to set an affinity is to set first any `kubelet.affinity`, `ksm.affinity`, or `controlPlane.affinity`. If these values are empty the chart +fallbacks to `affinity` (at root level), and if that value is empty, the chart fallbacks to `global.affinity`. + +The same procedure applies to `nodeSelector` and `tolerations`. + +On the other hand, some components have affinities and tolerations predefined e.g. to be able to run kubelet pods on nodes that are tainted as master +nodes or to schedule the KSM scraper on the same node of KSM to reduce the inter-node traffic. + +If you are having problems assigning pods to nodes it may be because of this. Take a look at the [`values.yaml`](values.yaml) to see if the pod that is +not having your expected behavior has any predefined value. + +### `hostNetwork` toggle + +In versions below v3, changing the `privileged` mode affected the `hostNetwork`. We changed this behavior and now you can set pods to use `hostNetwork` +using the corresponding [flags from the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md) +(`.global.hostNetwork` and `.hostNetwork`) but the component that scrapes data from the control plane has always set `hostNetwork` enabled by default +(Look in the [`values.yaml`](values.yaml) for `controlPlane.hostNetwork: true`) + +This is because the most common configuration of the control plane components is to be configured to listen only to `localhost`. + +If your cluster security policy does not allow to use `hostNetwork`, you can disable it control plane monitoring by setting `controlPlane.enabled` to +`false.` + +### `privileged` toggle + +The default value for `privileged` [from the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md) is +`false` but in this particular this chart it is set to `true` (Look in the [`values.yaml`](values.yaml) for `privileged: true`) + +This is because when `kubelet` pods need to run in privileged mode to fetch cpu, memory, process, and network metrics of your nodes. + +If your cluster security policy does not allow to have `privileged` in your pod' security context, you can disable it by setting `privileged` to +`false` taking into account that you will lose all the metrics from the host and some metadata from the host that are added to the metrics of the +integrations that you have configured. + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} + +## Past Contributors + +Previous iterations of this chart started as a community project in the [stable Helm chart repository](github.com/helm/charts/). New Relic is very thankful for all the 15+ community members that contributed and helped maintain the chart there over the years: + +* coreypobrien +* sstarcher +* jmccarty3 +* slayerjain +* ryanhope2 +* rk295 +* michaelajr +* isindir +* idirouhab +* ismferd +* enver +* diclophis +* jeffdesc +* costimuraru +* verwilst +* ezelenka diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-cplane-kind-deployment-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-cplane-kind-deployment-values.yaml new file mode 100644 index 0000000000..1e2c36d21f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-cplane-kind-deployment-values.yaml @@ -0,0 +1,135 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +common: + agentConfig: + # We set it in order for the kubelet to not crash when posting tho the agent. Since the License_Key is + # not valid, the Identity Api doesn't return an AgentID and the server from the Agent takes to long to respond + is_forward_only: true + config: + sink: + http: + timeout: 180s + +customAttributes: + new: relic + loren: ipsum + +# Disable KSM scraper as it is not enabled when testing this chart individually. +ksm: + enabled: false + +# K8s DaemonSets update strategy. +updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + +enableProcessMetrics: "false" +serviceAccount: + create: true + +podAnnotations: + annotation1: "annotation" +podLabels: + label1: "label" + +securityContext: + runAsUser: 1000 + runAsGroup: 2000 + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + +privileged: true + +rbac: + create: true + pspEnabled: false + +prefixDisplayNameWithCluster: false +useNodeNameAsDisplayName: true +integrations_config: [] + +kubelet: + enabled: true + annotations: {} + tolerations: + - operator: "Exists" + effect: "NoSchedule" + - operator: "Exists" + effect: "NoExecute" + extraEnv: + - name: ENV_VAR1 + value: "var1" + - name: ENV_VAR2 + value: "var2" + resources: + limits: + memory: 400M + requests: + cpu: 100m + memory: 180M + config: + scheme: "http" + +controlPlane: + kind: Deployment + enabled: true + config: + etcd: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=etcd" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:4001 + insecureSkipVerify: true + auth: + type: bearer + - url: http://localhost:2381 + scheduler: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-scheduler" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10259 + insecureSkipVerify: true + auth: + type: bearer + controllerManager: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-controller-manager" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + mtls: + secretName: secret-name + secretNamespace: default + apiServer: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-apiserver" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:8443 + insecureSkipVerify: true + auth: + type: bearer + mtls: + secretName: secret-name4 + - url: http://localhost:8080 + +images: + integration: + tag: test + repository: e2e/nri-kubernetes diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-values.yaml new file mode 100644 index 0000000000..125a496079 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/ci/test-values.yaml @@ -0,0 +1,134 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +common: + agentConfig: + # We set it in order for the kubelet to not crash when posting tho the agent. Since the License_Key is + # not valid, the Identity Api doesn't return an AgentID and the server from the Agent takes to long to respond + is_forward_only: true + config: + sink: + http: + timeout: 180s + +customAttributes: + new: relic + loren: ipsum + +# Disable KSM scraper as it is not enabled when testing this chart individually. +ksm: + enabled: false + +# K8s DaemonSets update strategy. +updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + +enableProcessMetrics: "false" +serviceAccount: + create: true + +podAnnotations: + annotation1: "annotation" +podLabels: + label1: "label" + +securityContext: + runAsUser: 1000 + runAsGroup: 2000 + allowPrivilegeEscalation: false + readOnlyRootFilesystem: true + +privileged: true + +rbac: + create: true + pspEnabled: false + +prefixDisplayNameWithCluster: false +useNodeNameAsDisplayName: true +integrations_config: [] + +kubelet: + enabled: true + annotations: {} + tolerations: + - operator: "Exists" + effect: "NoSchedule" + - operator: "Exists" + effect: "NoExecute" + extraEnv: + - name: ENV_VAR1 + value: "var1" + - name: ENV_VAR2 + value: "var2" + resources: + limits: + memory: 400M + requests: + cpu: 100m + memory: 180M + config: + scheme: "http" + +controlPlane: + enabled: true + config: + etcd: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=etcd" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:4001 + insecureSkipVerify: true + auth: + type: bearer + - url: http://localhost:2381 + scheduler: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-scheduler" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10259 + insecureSkipVerify: true + auth: + type: bearer + controllerManager: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-controller-manager" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + mtls: + secretName: secret-name + secretNamespace: default + apiServer: + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-apiserver" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:8443 + insecureSkipVerify: true + auth: + type: bearer + mtls: + secretName: secret-name4 + - url: http://localhost:8080 + +images: + integration: + tag: test + repository: e2e/nri-kubernetes diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/NOTES.txt new file mode 100644 index 0000000000..16cc6ea13d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/NOTES.txt @@ -0,0 +1,131 @@ +{{- if not .Values.forceUnsupportedInterval }} +{{- $max := 40 }} +{{- $min := 10 }} +{{- if not (.Values.common.config.interval | hasSuffix "s") }} +{{ fail (printf "Interval must be between %ds and %ds" $min $max ) }} +{{- end }} +{{- if gt ( .Values.common.config.interval | trimSuffix "s" | int64 ) $max }} +{{ fail (printf "Intervals larger than %ds are not supported" $max) }} +{{- end }} +{{- if lt ( .Values.common.config.interval | trimSuffix "s" | int64 ) $min }} +{{ fail (printf "Intervals smaller than %ds are not supported" $min) }} +{{- end }} +{{- end }} + +{{- if or (not .Values.ksm.enabled) (not .Values.kubelet.enabled) }} +Warning: +======== + +You have specified ksm or kubelet integration components as not enabled. +Those components are needed to have the full experience on NROne kubernetes explorer. +{{- end }} + +{{- if and .Values.controlPlane.enabled (not (include "nriKubernetes.controlPlane.hostNetwork" .)) }} +Warning: +======== + +Most Control Plane components listen in the loopback address only, which is not reachable without `hostNetwork: true`. +Control plane autodiscovery might not work as expected. +You can enable hostNetwork for all pods by setting `global.hotNetwork`, `hostNetwork` or only for the control +plane pods by setting `controlPlane.hostNetwork: true`. Alternatively, you can disable control plane monitoring altogether with +`controlPlane.enabled: false`. +{{- end }} + +{{- if and (include "newrelic.fargate" .) .Values.kubelet.affinity }} +Warning: +======== + +You have specified both an EKS Fargate environment (global.fargate) and custom +nodeAffinity rules, so we couldn't automatically exclude the kubelet daemonSet from +Fargate nodes. In order for the integration to work, you MUST manually exclude +the daemonSet from Fargate nodes. + +Please make sure your `values.yaml' contains a .kubelet.affinity.nodeAffinity that achieve the same effect as: + +affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: eks.amazonaws.com/compute-type + operator: NotIn + values: + - fargate +{{- end }} + +{{- if and .Values.nodeAffinity .Values.controlPlane.enabled }} +WARNING: `nodeAffinity` is deprecated +===================================== + +We have applied the old `nodeAffinity` to KSM and Kubelet components, but *NOT* to the control plane component as it +might conflict with the default nodeSelector. +This shimming will be removed in the future, please convert your `nodeAffinity` item into: +`ksm.affinity.nodeAffinity`, `controlPlane.affinity.nodeAffinity`, and `kubelet.affinity.nodeAffinity`. +{{- end }} + +{{- if and .Values.integrations_config }} +WARNING: `integrations_config` is deprecated +============================================ + +We have automatically translated `integrations_config` to the new format, but this shimming will be removed in the +future. Please migrate your configs to the new format in the `integrations` key. +{{- end }} + +{{- if or .Values.kubeStateMetricsScheme .Values.kubeStateMetricsPort .Values.kubeStateMetricsUrl .Values.kubeStateMetricsPodLabel .Values.kubeStateMetricsNamespace }} +WARNING: `kubeStateMetrics*` are deprecated +=========================================== + +We have automatically translated your `kubeStateMetrics*` values to the new format, but this shimming will be removed in +the future. Please migrate your configs to the new format in the `ksm.config` key. +{{- end }} + +{{- if .Values.runAsUser }} +WARNING: `runAsUser` is deprecated +================================== + +We have automatically translated your `runAsUser` setting to the new format, but this shimming will be removed in the +future. Please migrate your configs to the new format in the `securityContext` key. +{{- end }} + +{{- if .Values.config }} +WARNING: `config` is deprecated +=============================== + +We have automatically translated your `config` setting to the new format, but this shimming will be removed in the +future. Please migrate your agent config to the new format in the `common.agentConfig` key. +{{- end }} + + +{{ $errors:= "" }} + +{{- if .Values.logFile }} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.logFile" . ) }} +{{- end }} + +{{- if .Values.resources }} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.resources" . ) }} +{{- end }} + +{{- if .Values.image }} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.image" . ) }} +{{- end }} + +{{- if .Values.enableWindows }} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.windows" . ) }} +{{- end }} + +{{- if ( or .Values.controllerManagerEndpointUrl .Values.schedulerEndpointUrl .Values.etcdEndpointUrl .Values.apiServerEndpointUrl )}} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.apiURL" . ) }} +{{- end }} + +{{- if ( or .Values.etcdTlsSecretName .Values.etcdTlsSecretNamespace )}} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.etcdSecrets" . ) }} +{{- end }} + +{{- if .Values.apiServerSecurePort }} +{{ $errors = printf "%s\n\n%s" $errors (include "newrelic.compatibility.message.apiServerSecurePort" . ) }} +{{- end }} + +{{- if $errors | trim}} +{{- fail (printf "\n\n%s\n%s" (include "newrelic.compatibility.message.common" . ) $errors ) }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers.tpl new file mode 100644 index 0000000000..033ef0bfcf --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers.tpl @@ -0,0 +1,118 @@ +{{/* +Create a default fully qualified app name. + +This is a copy and paste from the common-library's name helper because the overriding system was broken. +As we have to change the logic to use "nrk8s" instead of `.Chart.Name` we need to maintain here a version +of the fullname helper + +By default the full name will be "" just in if it has "nrk8s" included in that, if not +it will be concatenated like "-nrk8s". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "nriKubernetes.naming.fullname" -}} +{{- $name := .Values.nameOverride | default "nrk8s" -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end -}} + + + +{{- /* Naming helpers*/ -}} +{{- define "nriKubernetes.naming.secrets" }} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "secrets") -}} +{{- end -}} + + + +{{- /* Return a YAML with the mode to be added to the labels */ -}} +{{- define "nriKubernetes._mode" -}} +{{- if include "newrelic.common.privileged" . -}} + mode: privileged +{{- else -}} + mode: unprivileged +{{- end -}} +{{- end -}} + + + +{{/* +Add `mode` label to the labels that come from the common library for all the objects +*/}} +{{- define "nriKubernetes.labels" -}} +{{- $labels := include "newrelic.common.labels" . | fromYaml -}} +{{- $mode := fromYaml ( include "nriKubernetes._mode" . ) -}} + +{{- mustMergeOverwrite $labels $mode | toYaml -}} +{{- end -}} + + + +{{/* +Add `mode` label to the labels that come from the common library for podLabels +*/}} +{{- define "nriKubernetes.labels.podLabels" -}} +{{- $labels := include "newrelic.common.labels.podLabels" . | fromYaml -}} +{{- $mode := fromYaml ( include "nriKubernetes._mode" . ) -}} + +{{- mustMergeOverwrite $labels $mode | toYaml -}} +{{- end -}} + + + +{{/* +Returns fargate +*/}} +{{- define "newrelic.fargate" -}} +{{- if .Values.fargate -}} + {{- .Values.fargate -}} +{{- else if .Values.global -}} + {{- if .Values.global.fargate -}} + {{- .Values.global.fargate -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- define "newrelic.integrationConfigDefaults" -}} +{{- if include "newrelic.common.lowDataMode" . -}} +interval: 30s +{{- else -}} +interval: 15s +{{- end -}} +{{- end -}} + + + +{{- /* These are the defaults that are used for all the containers in this chart (except the kubelet's agent */ -}} +{{- define "nriKubernetes.securityContext.containerDefaults" -}} +runAsUser: 1000 +runAsGroup: 2000 +allowPrivilegeEscalation: false +readOnlyRootFilesystem: true +{{- end -}} + + + +{{- /* Allow to change pod defaults dynamically based if we are running in privileged mode or not */ -}} +{{- define "nriKubernetes.securityContext.container" -}} +{{- $defaults := fromYaml ( include "nriKubernetes.securityContext.containerDefaults" . ) -}} +{{- $compatibilityLayer := include "newrelic.compatibility.securityContext" . | fromYaml -}} +{{- $commonLibrary := include "newrelic.common.securityContext.container" . | fromYaml -}} + +{{- $finalSecurityContext := dict -}} +{{- if $commonLibrary -}} + {{- $finalSecurityContext = mustMergeOverwrite $commonLibrary $compatibilityLayer -}} +{{- else -}} + {{- $finalSecurityContext = mustMergeOverwrite $defaults $compatibilityLayer -}} +{{- end -}} + +{{- toYaml $finalSecurityContext -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers_compatibility.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers_compatibility.tpl new file mode 100644 index 0000000000..07365e5a1a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/_helpers_compatibility.tpl @@ -0,0 +1,202 @@ +{{/* +Returns true if .Values.ksm.enabled is true and the legacy disableKubeStateMetrics is not set +*/}} +{{- define "newrelic.compatibility.ksm.enabled" -}} +{{- if and .Values.ksm.enabled (not .Values.disableKubeStateMetrics) -}} +true +{{- end -}} +{{- end -}} + +{{/* +Returns legacy ksm values +*/}} +{{- define "newrelic.compatibility.ksm.legacyData" -}} +enabled: true +{{- if .Values.kubeStateMetricsScheme }} +scheme: {{ .Values.kubeStateMetricsScheme }} +{{- end -}} +{{- if .Values.kubeStateMetricsPort }} +port: {{ .Values.kubeStateMetricsPort }} +{{- end -}} +{{- if .Values.kubeStateMetricsUrl }} +staticURL: {{ .Values.kubeStateMetricsUrl }} +{{- end -}} +{{- if .Values.kubeStateMetricsPodLabel }} +selector: {{ printf "%s=kube-state-metrics" .Values.kubeStateMetricsPodLabel }} +{{- end -}} +{{- if .Values.kubeStateMetricsNamespace }} +namespace: {{ .Values.kubeStateMetricsNamespace}} +{{- end -}} +{{- end -}} + +{{/* +Returns the new value if available, otherwise falling back on the legacy one +*/}} +{{- define "newrelic.compatibility.valueWithFallback" -}} +{{- if .supported }} +{{- toYaml .supported}} +{{- else if .legacy -}} +{{- toYaml .legacy}} +{{- end }} +{{- end -}} + +{{/* +Returns a dictionary with legacy runAsUser config +*/}} +{{- define "newrelic.compatibility.securityContext" -}} +{{- if .Values.runAsUser -}} +{{ dict "runAsUser" .Values.runAsUser | toYaml }} +{{- end -}} +{{- end -}} + +{{/* +Returns legacy annotations if available +*/}} +{{- define "newrelic.compatibility.annotations" -}} +{{- with .Values.daemonSet -}} +{{- with .annotations -}} +{{- toYaml . }} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Returns agent configmap merged with legacy config and legacy eventQueueDepth config +*/}} +{{- define "newrelic.compatibility.agentConfig" -}} +{{- $oldConfig := deepCopy (.Values.config | default dict) -}} +{{- $newConfig := deepCopy .Values.common.agentConfig -}} +{{- $eventQueueDepth := dict -}} + +{{- if .Values.eventQueueDepth -}} +{{- $eventQueueDepth = dict "event_queue_depth" .Values.eventQueueDepth -}} +{{- end -}} + +{{- mustMergeOverwrite $oldConfig $newConfig $eventQueueDepth | toYaml -}} +{{- end -}} + +{{- /* +Return a valid podSpec.affinity object from the old `.Values.nodeAffinity`. +*/ -}} +{{- define "newrelic.compatibility.nodeAffinity" -}} +{{- if .Values.nodeAffinity -}} +nodeAffinity: + {{- toYaml .Values.nodeAffinity | nindent 2 }} +{{- end -}} +{{- end -}} + +{{/* +Returns legacy integrations_config configmap data +*/}} +{{- define "newrelic.compatibility.integrations" -}} +{{- if .Values.integrations_config -}} +{{- range .Values.integrations_config }} +{{ .name -}}: |- + {{- toYaml .data | nindent 2 }} +{{- end -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic.compatibility.message.logFile" -}} +The 'logFile' option is no longer supported and has been replaced by: + - common.agentConfig.log_file. + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.resources" -}} +You have specified the legacy 'resources' option in your values, which is not fully compatible with the v3 version. +This version deploys three different components and therefore you'll need to specify resources for each of them. +Please use + - ksm.resources, + - controlPlane.resources, + - kubelet.resources. + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.apiServerSecurePort" -}} +You have specified the legacy 'apiServerSecurePort' option in your values, which is not fully compatible with the v3 +version. +Please configure the API Server port as a part of 'apiServer.autodiscover[].endpoints' + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.windows" -}} +nri-kubernetes v3 does not support deploying into windows Nodes. +Please use the latest 2.x version of the chart. + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.etcdSecrets" -}} +Values "etcdTlsSecretName" and "etcdTlsSecretNamespace" are no longer supported, please specify them as a part of the +'etcd' config in the values, for example: + - endpoints: + - url: https://localhost:9979 + insecureSkipVerify: true + auth: + type: mTLS + mtls: + secretName: {{ .Values.etcdTlsSecretName | default "etcdTlsSecretName"}} + secretNamespace: {{ .Values.etcdTlsSecretNamespace | default "etcdTlsSecretNamespace"}} + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.apiURL" -}} +Values "controllerManagerEndpointUrl", "etcdEndpointUrl", "apiServerEndpointUrl", "schedulerEndpointUrl" are no longer +supported, please specify them as a part of the 'controlplane' config in the values, for example + autodiscover: + - selector: "tier=control-plane,component=etcd" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:4001 + insecureSkipVerify: true + auth: + type: bearer + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.image" -}} +Configuring image repository an tag under 'image' is no longer supported. +The following values are no longer supported and are currently ignored: + - image.repository + - image.tag + - image.pullPolicy + - image.pullSecrets + +Notice that the 3.x version of the integration uses 3 different images. +Please set: + - images.forwarder.* to configure the infrastructure-agent forwarder. + - images.agent.* to configure the image bundling the infrastructure-agent and on-host integrations. + - images.integration.* to configure the image in charge of scraping k8s data. + +------ +{{- end -}} + +{{- define "newrelic.compatibility.message.customAttributes" -}} +We still support using custom attributes but we support it as a map and dropped it as a string. +customAttributes: {{ .Values.customAttributes | quote }} + +You should change your values to something like this: + +customAttributes: +{{- range $k, $v := fromJson .Values.customAttributes -}} + {{- $k | nindent 2 }}: {{ $v | quote }} +{{- end }} + +**NOTE**: If you read above errors like "invalid character ':' after top-level value" or "json: cannot unmarshal string into Go value of type map[string]interface {}" means that the string you have in your values is not a valid JSON, Helm is not able to parse it and we could not show you how you should change it. Sorry. +{{- end -}} + +{{- define "newrelic.compatibility.message.common" -}} +###### +The chart cannot be rendered since the values listed below are not supported. Please replace those with the new ones compatible with newrelic-infrastructure V3. + +Keep in mind that the flag "--reuse-values" is not supported when migrating from V2 to V3. +Further information can be found in the official docs https://docs.newrelic.com/docs/kubernetes-pixie/kubernetes-integration/get-started/changes-since-v3#migration-guide" +###### +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrole.yaml new file mode 100644 index 0000000000..391dc1e1fd --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrole.yaml @@ -0,0 +1,35 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }} +rules: + - apiGroups: [""] + resources: + - "nodes/metrics" + - "nodes/stats" + - "nodes/proxy" + verbs: ["get", "list"] + - apiGroups: [ "" ] + resources: + - "endpoints" + - "services" + - "nodes" + - "namespaces" + - "pods" + verbs: [ "get", "list", "watch" ] + - nonResourceURLs: ["/metrics"] + verbs: ["get"] + {{- if .Values.rbac.pspEnabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + resourceNames: + - privileged-{{ include "newrelic.common.naming.fullname" . }} + verbs: + - use + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..fc5dfb8dae --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic.common.naming.fullname" . }} +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_affinity_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_affinity_helper.tpl new file mode 100644 index 0000000000..320d16dae1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_affinity_helper.tpl @@ -0,0 +1,11 @@ +{{- /* +As this chart deploys what it should be three charts to maintain the transition to v3 as smooth as possible. +This means that this chart has 3 affinity so a helper should be done per scraper. +*/ -}} +{{- define "nriKubernetes.controlPlane.affinity" -}} +{{- if .Values.controlPlane.affinity -}} + {{- toYaml .Values.controlPlane.affinity -}} +{{- else if include "newrelic.common.affinity" . -}} + {{- include "newrelic.common.affinity" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_agent-config_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_agent-config_helper.tpl new file mode 100644 index 0000000000..e113def82f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_agent-config_helper.tpl @@ -0,0 +1,20 @@ +{{- /* +Defaults for controlPlane's agent config +*/ -}} +{{- define "nriKubernetes.controlPlane.agentConfig.defaults" -}} +is_forward_only: true +http_server_enabled: true +http_server_port: 8001 +{{- end -}} + + + +{{- define "nriKubernetes.controlPlane.agentConfig" -}} +{{- $agentDefaults := fromYaml ( include "newrelic.common.agentConfig.defaults" . ) -}} +{{- $controlPlane := fromYaml ( include "nriKubernetes.controlPlane.agentConfig.defaults" . ) -}} +{{- $agentConfig := fromYaml ( include "newrelic.compatibility.agentConfig" . ) -}} +{{- $cpAgentConfig := .Values.controlPlane.agentConfig -}} +{{- $customAttributes := dict "custom_attributes" (dict "clusterName" (include "newrelic.common.cluster" . )) -}} + +{{- mustMergeOverwrite $agentDefaults $controlPlane $agentConfig $cpAgentConfig $customAttributes | toYaml -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_host_network.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_host_network.tpl new file mode 100644 index 0000000000..2f3bdf2d94 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_host_network.tpl @@ -0,0 +1,22 @@ +{{/* Returns whether the controlPlane scraper should run with hostNetwork: true based on the user configuration. */}} +{{- define "nriKubernetes.controlPlane.hostNetwork" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values.controlPlane "hostNetwork" | kindIs "bool" -}} + {{- if .Values.controlPlane.hostNetwork -}} + {{- .Values.controlPlane.hostNetwork -}} + {{- end -}} +{{- else if include "newrelic.common.hostNetwork" . -}} + {{- include "newrelic.common.hostNetwork" . -}} +{{- end -}} +{{- end -}} + + + +{{/* Abstraction of "nriKubernetes.controlPlane.hostNetwork" that returns true of false directly */}} +{{- define "nriKubernetes.controlPlane.hostNetwork.value" -}} +{{- if include "nriKubernetes.controlPlane.hostNetwork" . -}} + true +{{- else -}} + false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_naming.tpl new file mode 100644 index 0000000000..4b9ef22e33 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_naming.tpl @@ -0,0 +1,16 @@ +{{- /* Naming helpers*/ -}} +{{- define "nriKubernetes.controlplane.fullname" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "controlplane") -}} +{{- end -}} + +{{- define "nriKubernetes.controlplane.fullname.agent" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "agent-controlplane") -}} +{{- end -}} + +{{- define "nriKubernetes.controlplane.fullname.serviceAccount" -}} +{{- if include "newrelic.common.serviceAccount.create" . -}} + {{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "controlplane") -}} +{{- else -}} + {{- include "newrelic.common.serviceAccount.name" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_rbac.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_rbac.tpl new file mode 100644 index 0000000000..a279df6b4c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_rbac.tpl @@ -0,0 +1,40 @@ +{{/* +Returns the list of namespaces where secrets need to be accessed by the controlPlane integration to do mTLS Auth +*/}} +{{- define "nriKubernetes.controlPlane.roleBindingNamespaces" -}} +{{ $namespaceList := list }} +{{- range $components := .Values.controlPlane.config }} + {{- if $components }} + {{- if kindIs "map" $components -}} + {{- if $components.staticEndpoint }} + {{- if $components.staticEndpoint.auth }} + {{- if $components.staticEndpoint.auth.mtls }} + {{- if $components.staticEndpoint.auth.mtls.secretNamespace }} + {{- $namespaceList = append $namespaceList $components.staticEndpoint.auth.mtls.secretNamespace -}} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- if $components.autodiscover }} + {{- range $autodiscover := $components.autodiscover }} + {{- if $autodiscover }} + {{- if $autodiscover.endpoints }} + {{- range $endpoint := $autodiscover.endpoints }} + {{- if $endpoint.auth }} + {{- if $endpoint.auth.mtls }} + {{- if $endpoint.auth.mtls.secretNamespace }} + {{- $namespaceList = append $namespaceList $endpoint.auth.mtls.secretNamespace -}} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} +roleBindingNamespaces: + {{- uniq $namespaceList | toYaml | nindent 2 }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_tolerations_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_tolerations_helper.tpl new file mode 100644 index 0000000000..3c82e82f52 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/_tolerations_helper.tpl @@ -0,0 +1,11 @@ +{{- /* +As this chart deploys what it should be three charts to maintain the transition to v3 as smooth as possible. +This means that this chart has 3 tolerations so a helper should be done per scraper. +*/ -}} +{{- define "nriKubernetes.controlPlane.tolerations" -}} +{{- if .Values.controlPlane.tolerations -}} + {{- toYaml .Values.controlPlane.tolerations -}} +{{- else if include "newrelic.common.tolerations" . -}} + {{- include "newrelic.common.tolerations" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/agent-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/agent-configmap.yaml new file mode 100644 index 0000000000..77f2e11dde --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/agent-configmap.yaml @@ -0,0 +1,18 @@ +{{- if .Values.controlPlane.enabled -}} +{{- if .Values.customAttributes | kindIs "string" }} +{{- fail ( include "newrelic.compatibility.message.customAttributes" . ) -}} +{{- else -}} +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.controlplane.fullname.agent" . }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "nriKubernetes.controlPlane.agentConfig" . | nindent 4 }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrole.yaml new file mode 100644 index 0000000000..57633e7f7b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrole.yaml @@ -0,0 +1,47 @@ +{{- if and (.Values.controlPlane.enabled) (.Values.rbac.create) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.controlplane.fullname" . }} +rules: + - apiGroups: [""] + resources: + - "nodes/metrics" + - "nodes/stats" + - "nodes/proxy" + verbs: ["get", "list"] + - apiGroups: [ "" ] + resources: + - "pods" + - "nodes" + verbs: [ "get", "list", "watch" ] + - nonResourceURLs: ["/metrics"] + verbs: ["get", "head"] + {{- if .Values.rbac.pspEnabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + resourceNames: + - privileged-{{ include "newrelic.common.naming.fullname" . }} + verbs: + - use + {{- end -}} +{{- $namespaces := include "nriKubernetes.controlPlane.roleBindingNamespaces" . | fromYaml -}} +{{- if $namespaces.roleBindingNamespaces}} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.naming.secrets" . }} +rules: + - apiGroups: [""] + resources: + - "secrets" + verbs: ["get", "list", "watch"] +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrolebinding.yaml new file mode 100644 index 0000000000..4e35300942 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if and (.Values.controlPlane.enabled) (.Values.rbac.create) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.controlplane.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "nriKubernetes.controlplane.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "nriKubernetes.controlplane.fullname.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/daemonset.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/daemonset.yaml new file mode 100644 index 0000000000..938fc48d45 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/daemonset.yaml @@ -0,0 +1,205 @@ +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: apps/v1 +kind: {{ .Values.controlPlane.kind }} +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "nriKubernetes.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.controlplane.fullname" . }} + {{- $legacyAnnotation:= fromYaml (include "newrelic.compatibility.annotations" .) -}} + {{- with include "newrelic.compatibility.valueWithFallback" (dict "legacy" $legacyAnnotation "supported" .Values.controlPlane.annotations )}} + annotations: {{ . | nindent 4 }} + {{- end }} +spec: + {{- if eq .Values.controlPlane.kind "DaemonSet"}} + {{- with .Values.updateStrategy }} + updateStrategy: {{ toYaml . | nindent 4 }} + {{- end }} + {{- end }} + {{- if eq .Values.controlPlane.kind "Deployment"}} + {{- with .Values.strategy }} + strategy: {{ toYaml . | nindent 4 }} + {{- end }} + {{- end }} + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + app.kubernetes.io/component: controlplane + template: + metadata: + annotations: + checksum/nri-kubernetes: {{ include (print $.Template.BasePath "/controlplane/scraper-configmap.yaml") . | sha256sum }} + checksum/agent-config: {{ include (print $.Template.BasePath "/controlplane/agent-configmap.yaml") . | sha256sum }} + {{- if include "newrelic.common.license.secret" . }}{{- /* If the is secret to template */}} + checksum/license-secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- end }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "nriKubernetes.labels.podLabels" . | nindent 8 }} + app.kubernetes.io/component: controlplane + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.images.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + hostNetwork: {{ include "nriKubernetes.controlPlane.hostNetwork.value" . }} + {{- if include "nriKubernetes.controlPlane.hostNetwork" . }} + dnsPolicy: ClusterFirstWithHostNet + {{- end }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + serviceAccountName: {{ include "nriKubernetes.controlplane.fullname.serviceAccount" . }} + + {{- if .Values.controlPlane.initContainers }} + initContainers: {{- tpl (.Values.controlPlane.initContainers | toYaml) . | nindent 8 }} + {{- end }} + containers: + - name: controlplane + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} + imagePullPolicy: {{ .Values.images.integration.pullPolicy }} + {{- with include "nriKubernetes.securityContext.container" . | fromYaml }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + env: + - name: "NRI_KUBERNETES_SINK_HTTP_PORT" + value: {{ get (fromYaml (include "nriKubernetes.controlPlane.agentConfig" .)) "http_server_port" | quote }} + - name: "NRI_KUBERNETES_CLUSTERNAME" + value: {{ include "newrelic.common.cluster" . }} + - name: "NRI_KUBERNETES_VERBOSE" + value: {{ include "newrelic.common.verboseLog.valueAsBoolean" . | quote }} + + - name: "NRI_KUBERNETES_NODENAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + - name: "NRI_KUBERNETES_NODEIP" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "status.hostIP" + + {{- with .Values.controlPlane.extraEnv }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.controlPlane.extraEnvFrom }} + envFrom: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: nri-kubernetes-config + mountPath: /etc/newrelic-infra/nri-kubernetes.yml + subPath: nri-kubernetes.yml + {{- with .Values.controlPlane.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.controlPlane.resources }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + - name: forwarder + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.forwarder "context" .) }} + imagePullPolicy: {{ .Values.images.forwarder.pullPolicy }} + {{- with include "nriKubernetes.securityContext.container" . | fromYaml }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + ports: + - containerPort: {{ get (fromYaml (include "nriKubernetes.controlPlane.agentConfig" .)) "http_server_port" }} + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + + - name: "NRIA_DNS_HOSTNAME_RESOLUTION" + value: "false" + + - name: "K8S_NODE_NAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + + {{- if .Values.useNodeNameAsDisplayName }} + - name: "NRIA_DISPLAY_NAME" + {{- if .Values.prefixDisplayNameWithCluster }} + value: "{{ include "newrelic.common.cluster" . }}:$(K8S_NODE_NAME)" + {{- else }} + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + {{- end }} + {{- end }} + + {{- with .Values.controlPlane.extraEnv }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.controlPlane.extraEnvFrom }} + envFrom: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - mountPath: /var/db/newrelic-infra/data + name: forwarder-tmpfs-data + - mountPath: /var/db/newrelic-infra/user_data + name: forwarder-tmpfs-user-data + - mountPath: /tmp + name: forwarder-tmpfs-tmp + - name: config + mountPath: /etc/newrelic-infra.yml + subPath: newrelic-infra.yml + {{- with .Values.controlPlane.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.controlPlane.resources }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + volumes: + - name: nri-kubernetes-config + configMap: + name: {{ include "nriKubernetes.controlplane.fullname" . }} + items: + - key: nri-kubernetes.yml + path: nri-kubernetes.yml + - name: forwarder-tmpfs-data + emptyDir: {} + - name: forwarder-tmpfs-user-data + emptyDir: {} + - name: forwarder-tmpfs-tmp + emptyDir: {} + - name: config + configMap: + name: {{ include "nriKubernetes.controlplane.fullname.agent" . }} + items: + - key: newrelic-infra.yml + path: newrelic-infra.yml + {{- with .Values.controlPlane.extraVolumes }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with include "nriKubernetes.controlPlane.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + {{- with include "nriKubernetes.controlPlane.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{- with .Values.controlPlane.nodeSelector | default (fromYaml (include "newrelic.common.nodeSelector" .)) }} + {{- toYaml . | nindent 8 }} + {{- end -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/rolebinding.yaml new file mode 100644 index 0000000000..d97fc181a4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/rolebinding.yaml @@ -0,0 +1,21 @@ +{{- if .Values.rbac.create }} +{{- $namespaces := (include "nriKubernetes.controlPlane.roleBindingNamespaces" . | fromYaml) -}} +{{- range $namespaces.roleBindingNamespaces }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + {{- include "newrelic.common.labels" $ | nindent 4 }} + name: {{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" $) "suffix" .) }} + namespace: {{ . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "nriKubernetes.naming.secrets" $ }} +subjects: +- kind: ServiceAccount + name: {{ include "nriKubernetes.controlplane.fullname.serviceAccount" $ }} + namespace: {{ $.Release.Namespace }} +{{- end -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/scraper-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/scraper-configmap.yaml new file mode 100644 index 0000000000..454665deda --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/scraper-configmap.yaml @@ -0,0 +1,36 @@ +{{- if .Values.controlPlane.enabled -}} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.controlplane.fullname" . }} + namespace: {{ .Release.Namespace }} +data: + nri-kubernetes.yml: |- + {{- (merge .Values.common.config (include "newrelic.integrationConfigDefaults" . | fromYaml)) | toYaml | nindent 4 }} + controlPlane: + {{- omit .Values.controlPlane.config "etcd" "scheduler" "controllerManager" "apiServer" | toYaml | nindent 6 }} + enabled: true + + {{- if .Values.controlPlane.config.etcd.enabled }} + etcd: + {{- toYaml .Values.controlPlane.config.etcd | nindent 8 -}} + {{- end -}} + + {{- if .Values.controlPlane.config.scheduler.enabled }} + scheduler: + {{- toYaml .Values.controlPlane.config.scheduler | nindent 8 -}} + {{- end -}} + + {{- if .Values.controlPlane.config.controllerManager.enabled }} + controllerManager: + {{- toYaml .Values.controlPlane.config.controllerManager | nindent 8 -}} + {{- end -}} + + {{- if .Values.controlPlane.config.apiServer.enabled }} + apiServer: + {{- toYaml .Values.controlPlane.config.apiServer | nindent 8 -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/serviceaccount.yaml new file mode 100644 index 0000000000..502e1c9865 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/controlplane/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.controlplane.fullname.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_affinity_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_affinity_helper.tpl new file mode 100644 index 0000000000..ce795708d2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_affinity_helper.tpl @@ -0,0 +1,14 @@ +{{- /* +As this chart deploys what it should be three charts to maintain the transition to v3 as smooth as possible. +This means that this chart has 3 affinity so a helper should be done per scraper. +*/ -}} +{{- define "nriKubernetes.ksm.affinity" -}} +{{- if or .Values.ksm.affinity .Values.nodeAffinity -}} + {{- $legacyNodeAffinity := fromYaml ( include "newrelic.compatibility.nodeAffinity" . ) | default dict -}} + {{- $valuesAffinity := .Values.ksm.affinity | default dict -}} + {{- $affinity := mustMergeOverwrite $legacyNodeAffinity $valuesAffinity -}} + {{- toYaml $affinity -}} +{{- else if include "newrelic.common.affinity" . -}} + {{- include "newrelic.common.affinity" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_agent-config_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_agent-config_helper.tpl new file mode 100644 index 0000000000..e7b55644c2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_agent-config_helper.tpl @@ -0,0 +1,20 @@ +{{- /* +Defaults for ksm's agent config +*/ -}} +{{- define "nriKubernetes.ksm.agentConfig.defaults" -}} +is_forward_only: true +http_server_enabled: true +http_server_port: 8002 +{{- end -}} + + + +{{- define "nriKubernetes.ksm.agentConfig" -}} +{{- $agentDefaults := fromYaml ( include "newrelic.common.agentConfig.defaults" . ) -}} +{{- $ksm := fromYaml ( include "nriKubernetes.ksm.agentConfig.defaults" . ) -}} +{{- $agentConfig := fromYaml ( include "newrelic.compatibility.agentConfig" . ) -}} +{{- $ksmAgentConfig := .Values.ksm.agentConfig -}} +{{- $customAttributes := dict "custom_attributes" (dict "clusterName" (include "newrelic.common.cluster" . )) -}} + +{{- mustMergeOverwrite $agentDefaults $ksm $agentConfig $ksmAgentConfig $customAttributes | toYaml -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_host_network.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_host_network.tpl new file mode 100644 index 0000000000..59a6db7be1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_host_network.tpl @@ -0,0 +1,22 @@ +{{/* Returns whether the ksm scraper should run with hostNetwork: true based on the user configuration. */}} +{{- define "nriKubernetes.ksm.hostNetwork" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values.ksm "hostNetwork" | kindIs "bool" -}} + {{- if .Values.ksm.hostNetwork -}} + {{- .Values.ksm.hostNetwork -}} + {{- end -}} +{{- else if include "newrelic.common.hostNetwork" . -}} + {{- include "newrelic.common.hostNetwork" . -}} +{{- end -}} +{{- end -}} + + + +{{/* Abstraction of "nriKubernetes.ksm.hostNetwork" that returns true of false directly */}} +{{- define "nriKubernetes.ksm.hostNetwork.value" -}} +{{- if include "nriKubernetes.ksm.hostNetwork" . -}} + true +{{- else -}} + false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_naming.tpl new file mode 100644 index 0000000000..d8c283c430 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_naming.tpl @@ -0,0 +1,8 @@ +{{- /* Naming helpers*/ -}} +{{- define "nriKubernetes.ksm.fullname" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "ksm") -}} +{{- end -}} + +{{- define "nriKubernetes.ksm.fullname.agent" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "agent-ksm") -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_tolerations_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_tolerations_helper.tpl new file mode 100644 index 0000000000..e1a9fd80c2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/_tolerations_helper.tpl @@ -0,0 +1,11 @@ +{{- /* +As this chart deploys what it should be three charts to maintain the transition to v3 as smooth as possible. +This means that this chart has 3 tolerations so a helper should be done per scraper. +*/ -}} +{{- define "nriKubernetes.ksm.tolerations" -}} +{{- if .Values.ksm.tolerations -}} + {{- toYaml .Values.ksm.tolerations -}} +{{- else if include "newrelic.common.tolerations" . -}} + {{- include "newrelic.common.tolerations" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/agent-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/agent-configmap.yaml new file mode 100644 index 0000000000..6a438e9a3c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/agent-configmap.yaml @@ -0,0 +1,18 @@ +{{- if .Values.ksm.enabled -}} +{{- if .Values.customAttributes | kindIs "string" }} +{{- fail ( include "newrelic.compatibility.message.customAttributes" . ) -}} +{{- else -}} +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.ksm.fullname.agent" . }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "nriKubernetes.ksm.agentConfig" . | nindent 4 }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/deployment.yaml new file mode 100644 index 0000000000..507199d5a7 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/deployment.yaml @@ -0,0 +1,192 @@ +{{- if include "newrelic.compatibility.ksm.enabled" . -}} +apiVersion: apps/v1 +kind: Deployment +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "nriKubernetes.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.ksm.fullname" . }} + {{- $legacyAnnotation:= fromYaml (include "newrelic.compatibility.annotations" .) -}} + {{- with include "newrelic.compatibility.valueWithFallback" (dict "legacy" $legacyAnnotation "supported" .Values.ksm.annotations )}} + annotations: {{ . | nindent 4 }} + {{- end }} +spec: + {{- with .Values.strategy }} + strategy: {{ toYaml . | nindent 4 }} + {{- end }} + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + app.kubernetes.io/component: ksm + template: + metadata: + annotations: + checksum/nri-kubernetes: {{ include (print $.Template.BasePath "/ksm/scraper-configmap.yaml") . | sha256sum }} + checksum/agent-config: {{ include (print $.Template.BasePath "/ksm/agent-configmap.yaml") . | sha256sum }} + {{- if include "newrelic.common.license.secret" . }}{{- /* If the is secret to template */}} + checksum/license-secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- end }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "nriKubernetes.labels.podLabels" . | nindent 8 }} + app.kubernetes.io/component: ksm + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.images.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + hostNetwork: {{ include "nriKubernetes.ksm.hostNetwork.value" . }} + {{- if include "nriKubernetes.ksm.hostNetwork" . }} + dnsPolicy: ClusterFirstWithHostNet + {{- end }} + + {{- if .Values.ksm.initContainers }} + initContainers: {{- tpl (.Values.ksm.initContainers | toYaml) . | nindent 8 }} + {{- end }} + containers: + - name: ksm + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} + imagePullPolicy: {{ .Values.images.integration.pullPolicy }} + {{- with include "nriKubernetes.securityContext.container" . | fromYaml }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + env: + - name: "NRI_KUBERNETES_SINK_HTTP_PORT" + value: {{ get (fromYaml (include "nriKubernetes.ksm.agentConfig" .)) "http_server_port" | quote }} + - name: "NRI_KUBERNETES_CLUSTERNAME" + value: {{ include "newrelic.common.cluster" . }} + - name: "NRI_KUBERNETES_VERBOSE" + value: {{ include "newrelic.common.verboseLog.valueAsBoolean" . | quote }} + + - name: "NRI_KUBERNETES_NODENAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + + {{- with .Values.ksm.extraEnv }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.ksm.extraEnvFrom }} + envFrom: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: nri-kubernetes-config + mountPath: /etc/newrelic-infra/nri-kubernetes.yml + subPath: nri-kubernetes.yml + {{- with .Values.ksm.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.ksm.resources }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + - name: forwarder + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.forwarder "context" .) }} + imagePullPolicy: {{ .Values.images.forwarder.pullPolicy }} + {{- with include "nriKubernetes.securityContext.container" . | fromYaml }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + ports: + - containerPort: {{ get (fromYaml (include "nriKubernetes.ksm.agentConfig" .)) "http_server_port" }} + env: + - name: NRIA_LICENSE_KEY + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + + - name: "NRIA_DNS_HOSTNAME_RESOLUTION" + value: "false" + + - name: "K8S_NODE_NAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + + {{- if .Values.useNodeNameAsDisplayName }} + - name: "NRIA_DISPLAY_NAME" + {{- if .Values.prefixDisplayNameWithCluster }} + value: "{{ include "newrelic.common.cluster" . }}:$(K8S_NODE_NAME)" + {{- else }} + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + {{- end }} + {{- end }} + + {{- with .Values.ksm.env }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.ksm.extraEnvFrom }} + envFrom: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - mountPath: /var/db/newrelic-infra/data + name: forwarder-tmpfs-data + - mountPath: /var/db/newrelic-infra/user_data + name: forwarder-tmpfs-user-data + - mountPath: /tmp + name: forwarder-tmpfs-tmp + - name: config + mountPath: /etc/newrelic-infra.yml + subPath: newrelic-infra.yml + {{- with .Values.ksm.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.ksm.resources }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + volumes: + - name: nri-kubernetes-config + configMap: + name: {{ include "nriKubernetes.ksm.fullname" . }} + items: + - key: nri-kubernetes.yml + path: nri-kubernetes.yml + - name: forwarder-tmpfs-data + emptyDir: {} + - name: forwarder-tmpfs-user-data + emptyDir: {} + - name: forwarder-tmpfs-tmp + emptyDir: {} + - name: config + configMap: + name: {{ include "nriKubernetes.ksm.fullname.agent" . }} + items: + - key: newrelic-infra.yml + path: newrelic-infra.yml + {{- with .Values.ksm.extraVolumes }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with include "nriKubernetes.ksm.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + {{- with include "nriKubernetes.ksm.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{- with .Values.ksm.nodeSelector | default (fromYaml (include "newrelic.common.nodeSelector" .)) }} + {{- toYaml . | nindent 8 }} + {{- end -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/scraper-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/scraper-configmap.yaml new file mode 100644 index 0000000000..3314df9c77 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/ksm/scraper-configmap.yaml @@ -0,0 +1,15 @@ +{{- if include "newrelic.compatibility.ksm.enabled" . -}} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.ksm.fullname" . }} + namespace: {{ .Release.Namespace }} +data: + nri-kubernetes.yml: |- + {{- (merge .Values.common.config (include "newrelic.integrationConfigDefaults" . | fromYaml)) | toYaml | nindent 4 }} + ksm: + {{- mustMergeOverwrite .Values.ksm.config (include "newrelic.compatibility.ksm.legacyData" . | fromYaml) | toYaml | nindent 6 -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_affinity_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_affinity_helper.tpl new file mode 100644 index 0000000000..a3abf0855e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_affinity_helper.tpl @@ -0,0 +1,33 @@ +{{- /* +Patch to add affinity in case we are running in fargate mode +*/ -}} +{{- define "nriKubernetes.kubelet.affinity.fargateDefaults" -}} +nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: eks.amazonaws.com/compute-type + operator: NotIn + values: + - fargate +{{- end -}} + + + +{{- /* +As this chart deploys what it should be three charts to maintain the transition to v3 as smooth as possible. +This means that this chart has 3 affinity so a helper should be done per scraper. +*/ -}} +{{- define "nriKubernetes.kubelet.affinity" -}} + +{{- if or .Values.kubelet.affinity .Values.nodeAffinity -}} + {{- $legacyNodeAffinity := fromYaml ( include "newrelic.compatibility.nodeAffinity" . ) | default dict -}} + {{- $valuesAffinity := .Values.kubelet.affinity | default dict -}} + {{- $affinity := mustMergeOverwrite $legacyNodeAffinity $valuesAffinity -}} + {{- toYaml $affinity -}} +{{- else if include "newrelic.common.affinity" . -}} + {{- include "newrelic.common.affinity" . -}} +{{- else if include "newrelic.fargate" . -}} + {{- include "nriKubernetes.kubelet.affinity.fargateDefaults" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_agent-config_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_agent-config_helper.tpl new file mode 100644 index 0000000000..ea6ffc25f3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_agent-config_helper.tpl @@ -0,0 +1,31 @@ +{{- /* +Defaults for kubelet's agent config +*/ -}} +{{- define "nriKubernetes.kubelet.agentConfig.defaults" -}} +http_server_enabled: true +http_server_port: 8003 +features: + docker_enabled: false +{{- if not ( include "newrelic.common.privileged" . ) }} +is_secure_forward_only: true +{{- end }} +{{- /* +`enableProcessMetrics` is commented in the values and we want to configure it when it is set to something +either `true` or `false`. So we test if the variable is a boolean and in that case simply use it. +*/}} +{{- if (get .Values "enableProcessMetrics" | kindIs "bool") }} +enable_process_metrics: {{ .Values.enableProcessMetrics }} +{{- end }} +{{- end -}} + + + +{{- define "nriKubernetes.kubelet.agentConfig" -}} +{{- $agentDefaults := fromYaml ( include "newrelic.common.agentConfig.defaults" . ) -}} +{{- $kubelet := fromYaml ( include "nriKubernetes.kubelet.agentConfig.defaults" . ) -}} +{{- $agentConfig := fromYaml ( include "newrelic.compatibility.agentConfig" . ) -}} +{{- $kubeletAgentConfig := .Values.kubelet.agentConfig -}} +{{- $customAttributes := dict "custom_attributes" (dict "clusterName" (include "newrelic.common.cluster" . )) -}} + +{{- mustMergeOverwrite $agentDefaults $kubelet $agentConfig $kubeletAgentConfig $customAttributes | toYaml -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_host_network.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_host_network.tpl new file mode 100644 index 0000000000..7944f98a72 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_host_network.tpl @@ -0,0 +1,22 @@ +{{/* Returns whether the kubelet scraper should run with hostNetwork: true based on the user configuration. */}} +{{- define "nriKubernetes.kubelet.hostNetwork" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values.kubelet "hostNetwork" | kindIs "bool" -}} + {{- if .Values.kubelet.hostNetwork -}} + {{- .Values.kubelet.hostNetwork -}} + {{- end -}} +{{- else if include "newrelic.common.hostNetwork" . -}} + {{- include "newrelic.common.hostNetwork" . -}} +{{- end -}} +{{- end -}} + + + +{{/* Abstraction of "nriKubernetes.kubelet.hostNetwork" that returns true of false directly */}} +{{- define "nriKubernetes.kubelet.hostNetwork.value" -}} +{{- if include "nriKubernetes.kubelet.hostNetwork" . -}} + true +{{- else -}} + false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_naming.tpl new file mode 100644 index 0000000000..71c142156d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_naming.tpl @@ -0,0 +1,12 @@ +{{- /* Naming helpers*/ -}} +{{- define "nriKubernetes.kubelet.fullname" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "kubelet") -}} +{{- end -}} + +{{- define "nriKubernetes.kubelet.fullname.agent" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "agent-kubelet") -}} +{{- end -}} + +{{- define "nriKubernetes.kubelet.fullname.integrations" -}} +{{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "nriKubernetes.naming.fullname" .) "suffix" "integrations-cfg") -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_security_context_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_security_context_helper.tpl new file mode 100644 index 0000000000..4e334466c1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_security_context_helper.tpl @@ -0,0 +1,32 @@ +{{- /*This defines the defaults that the privileged mode has for the agent's securityContext */ -}} +{{- define "nriKubernetes.kubelet.securityContext.privileged" -}} +runAsUser: 0 +runAsGroup: 0 +allowPrivilegeEscalation: true +privileged: true +readOnlyRootFilesystem: true +{{- end -}} + + + +{{- /* This is the container security context for the agent */ -}} +{{- define "nriKubernetes.kubelet.securityContext.agentContainer" -}} +{{- $defaults := dict -}} +{{- if include "newrelic.common.privileged" . -}} +{{- $defaults = fromYaml ( include "nriKubernetes.kubelet.securityContext.privileged" . ) -}} +{{- else -}} +{{- $defaults = fromYaml ( include "nriKubernetes.securityContext.containerDefaults" . ) -}} +{{- end -}} + +{{- $compatibilityLayer := include "newrelic.compatibility.securityContext" . | fromYaml -}} +{{- $commonLibrary := include "newrelic.common.securityContext.container" . | fromYaml -}} + +{{- $finalSecurityContext := dict -}} +{{- if $commonLibrary -}} + {{- $finalSecurityContext = mustMergeOverwrite $commonLibrary $compatibilityLayer -}} +{{- else -}} + {{- $finalSecurityContext = mustMergeOverwrite $defaults $compatibilityLayer -}} +{{- end -}} + +{{- toYaml $finalSecurityContext -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_tolerations_helper.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_tolerations_helper.tpl new file mode 100644 index 0000000000..e46d83d69d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/_tolerations_helper.tpl @@ -0,0 +1,11 @@ +{{- /* +As this chart deploys what it should be three charts to maintain the transition to v3 as smooth as possible. +This means that this chart has 3 tolerations so a helper should be done per scraper. +*/ -}} +{{- define "nriKubernetes.kubelet.tolerations" -}} +{{- if .Values.kubelet.tolerations -}} + {{- toYaml .Values.kubelet.tolerations -}} +{{- else if include "newrelic.common.tolerations" . -}} + {{- include "newrelic.common.tolerations" . -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/agent-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/agent-configmap.yaml new file mode 100644 index 0000000000..0f71f129ae --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/agent-configmap.yaml @@ -0,0 +1,18 @@ +{{- if .Values.kubelet.enabled -}} +{{- if .Values.customAttributes | kindIs "string" }} +{{- fail ( include "newrelic.compatibility.message.customAttributes" . ) -}} +{{- else -}} +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.kubelet.fullname.agent" . }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "nriKubernetes.kubelet.agentConfig" . | nindent 4 }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/daemonset.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/daemonset.yaml new file mode 100644 index 0000000000..517079be76 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/daemonset.yaml @@ -0,0 +1,265 @@ +{{- if (.Values.kubelet.enabled) }} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "nriKubernetes.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.kubelet.fullname" . }} + {{- $legacyAnnotation:= fromYaml (include "newrelic.compatibility.annotations" .) -}} + {{- with include "newrelic.compatibility.valueWithFallback" (dict "legacy" $legacyAnnotation "supported" .Values.kubelet.annotations )}} + annotations: {{ . | nindent 4 }} + {{- end }} +spec: + {{- with .Values.updateStrategy }} + updateStrategy: {{ toYaml . | nindent 4 }} + {{- end }} + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + app.kubernetes.io/component: kubelet + template: + metadata: + annotations: + checksum/nri-kubernetes: {{ include (print $.Template.BasePath "/kubelet/scraper-configmap.yaml") . | sha256sum }} + checksum/agent-config: {{ include (print $.Template.BasePath "/kubelet/agent-configmap.yaml") . | sha256sum }} + {{- if include "newrelic.common.license.secret" . }}{{- /* If the is secret to template */}} + checksum/license-secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- end }} + checksum/integrations_config: {{ include (print $.Template.BasePath "/kubelet/integrations-configmap.yaml") . | sha256sum }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "nriKubernetes.labels.podLabels" . | nindent 8 }} + app.kubernetes.io/component: kubelet + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.images.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + hostNetwork: {{ include "nriKubernetes.kubelet.hostNetwork.value" . }} + {{- if include "nriKubernetes.kubelet.hostNetwork" . }} + dnsPolicy: ClusterFirstWithHostNet + {{- end }} + + {{- if .Values.kubelet.initContainers }} + initContainers: {{- tpl (.Values.kubelet.initContainers | toYaml) . | nindent 8 }} + {{- end }} + containers: + - name: kubelet + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} + imagePullPolicy: {{ .Values.images.integration.pullPolicy }} + {{- with include "nriKubernetes.securityContext.container" . | fromYaml }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + env: + - name: "NRI_KUBERNETES_SINK_HTTP_PORT" + value: {{ get (fromYaml (include "nriKubernetes.kubelet.agentConfig" .)) "http_server_port" | quote }} + - name: "NRI_KUBERNETES_CLUSTERNAME" + value: {{ include "newrelic.common.cluster" . }} + - name: "NRI_KUBERNETES_VERBOSE" + value: {{ include "newrelic.common.verboseLog.valueAsBoolean" . | quote }} + + - name: "NRI_KUBERNETES_NODENAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + # Required to connect to the kubelet + - name: "NRI_KUBERNETES_NODEIP" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "status.hostIP" + + {{- with .Values.kubelet.extraEnv }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.kubelet.extraEnvFrom }} + envFrom: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: nri-kubernetes-config + mountPath: /etc/newrelic-infra/nri-kubernetes.yml + subPath: nri-kubernetes.yml + {{- with .Values.kubelet.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.kubelet.resources }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + - name: agent + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} + args: [ "newrelic-infra" ] + imagePullPolicy: {{ .Values.images.agent.pullPolicy }} + {{- with include "nriKubernetes.kubelet.securityContext.agentContainer" . | fromYaml }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + ports: + - containerPort: {{ get (fromYaml (include "nriKubernetes.kubelet.agentConfig" .)) "http_server_port" }} + env: + - name: NRIA_LICENSE_KEY + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + + - name: "NRIA_OVERRIDE_HOSTNAME_SHORT" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + + - name: "NRIA_OVERRIDE_HOSTNAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + + {{- if not (include "newrelic.common.privileged" .) }} + # Override NRIA_OVERRIDE_HOST_ROOT to empty if unprivileged. This must be done as an env var as the + # `k8s-events-forwarder` and `infrastructure-bundle` images ship this very same env var set to /host. + - name: "NRIA_OVERRIDE_HOST_ROOT" + value: "" + {{- end }} + + - name: "NRI_KUBERNETES_NODE_NAME" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + + {{- if .Values.useNodeNameAsDisplayName }} + - name: "NRIA_DISPLAY_NAME" + {{- if .Values.prefixDisplayNameWithCluster }} + value: "{{ include "newrelic.common.cluster" . }}:$(NRI_KUBERNETES_NODE_NAME)" + {{- else }} + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "spec.nodeName" + {{- end }} + {{- end }} + + {{- /* Needed to populate clustername in integration metrics */}} + - name: "CLUSTER_NAME" + value: {{ include "newrelic.common.cluster" . }} + - name: "NRIA_PASSTHROUGH_ENVIRONMENT" + value: "CLUSTER_NAME" + + {{- /* Needed for autodiscovery since hostNetwork=false */}} + - name: "NRIA_HOST" + valueFrom: + fieldRef: + apiVersion: "v1" + fieldPath: "status.hostIP" + + {{- with .Values.kubelet.extraEnv }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.kubelet.extraEnvFrom }} + envFrom: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config + mountPath: /etc/newrelic-infra.yml + subPath: newrelic-infra.yml + - name: nri-integrations-cfg-volume + mountPath: /etc/newrelic-infra/integrations.d/ + {{- if include "newrelic.common.privileged" . }} + - name: dev + mountPath: /dev + - name: host-containerd-socket + mountPath: /run/containerd/containerd.sock + - name: host-docker-socket + mountPath: /var/run/docker.sock + - name: log + mountPath: /var/log + - name: host-volume + mountPath: /host + mountPropagation: HostToContainer + readOnly: true + {{- end }} + - mountPath: /var/db/newrelic-infra/data + name: agent-tmpfs-data + - mountPath: /var/db/newrelic-infra/user_data + name: agent-tmpfs-user-data + - mountPath: /tmp + name: agent-tmpfs-tmp + {{- with .Values.kubelet.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.kubelet.resources }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + volumes: + {{- if include "newrelic.common.privileged" . }} + - name: dev + hostPath: + path: /dev + - name: host-containerd-socket + hostPath: + path: /run/containerd/containerd.sock + - name: host-docker-socket + hostPath: + path: /var/run/docker.sock + - name: log + hostPath: + path: /var/log + - name: host-volume + hostPath: + path: / + {{- end }} + - name: agent-tmpfs-data + emptyDir: {} + - name: agent-tmpfs-user-data + emptyDir: {} + - name: agent-tmpfs-tmp + emptyDir: {} + - name: nri-kubernetes-config + configMap: + name: {{ include "nriKubernetes.kubelet.fullname" . }} + items: + - key: nri-kubernetes.yml + path: nri-kubernetes.yml + - name: config + configMap: + name: {{ include "nriKubernetes.kubelet.fullname.agent" . }} + items: + - key: newrelic-infra.yml + path: newrelic-infra.yml + - name: nri-integrations-cfg-volume + configMap: + name: {{ include "nriKubernetes.kubelet.fullname.integrations" . }} + {{- with .Values.kubelet.extraVolumes }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with include "nriKubernetes.kubelet.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + {{- with include "nriKubernetes.kubelet.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{- with .Values.kubelet.nodeSelector | default (fromYaml (include "newrelic.common.nodeSelector" .)) }} + {{- toYaml . | nindent 8 }} + {{- end -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/integrations-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/integrations-configmap.yaml new file mode 100644 index 0000000000..abf381f383 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/integrations-configmap.yaml @@ -0,0 +1,72 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.kubelet.fullname.integrations" . }} +data: + # This ConfigMap holds config files for integrations. They should have the following format: + #redis-config.yml: | + # # Run auto discovery to find pods with label "app=redis" + # discovery: + # command: + # # Run discovery for Kubernetes. Use the following optional arguments: + # # --namespaces: Comma separated list of namespaces to discover pods on + # # --tls: Use secure (TLS) connection + # # --port: Port used to connect to the kubelet. Default is 10255 + # exec: /var/db/newrelic-infra/nri-discovery-kubernetes --port PORT --tls + # match: + # label.app: redis + # integrations: + # - name: nri-redis + # env: + # # using the discovered IP as the hostname address + # HOSTNAME: ${discovery.ip} + # PORT: 6379 + # KEYS: '{"0":[""],"1":[""]}' + # REMOTE_MONITORING: true + # labels: + # env: production + {{- if .Values.integrations -}} + {{- range $k, $v := .Values.integrations -}} + {{- $k | trimSuffix ".yaml" | trimSuffix ".yml" | nindent 2 -}}.yaml: |- + {{- $v | toYaml | nindent 4 -}} + {{- end }} + {{- end }} + + {{- /* This template will add and template the integrations in the old .Values.integrations_config */}} + {{- include "newrelic.compatibility.integrations" . | nindent 2 }} + + {{- /* This template will add Pixie Health check to the integrations */}} + {{- if .Values.selfMonitoring.pixie.enabled }} + pixie-health-check.yaml: | + --- + # This Flex config performs periodic checks of the Pixie + # /healthz and /statusz endpoints exposed by the Pixie Cloud Connector. + # A status for each endpoint is sent to New Relic in a pixieHealthCheck event. + # + # If Pixie is not installed in the cluster, no events will be generated. + # This can also be disabled with enablePixieHealthCheck: false in the values.yaml file. + discovery: + command: + exec: /var/db/newrelic-infra/nri-discovery-kubernetes --tls --port 10250 + match: + label.name: vizier-cloud-connector + integrations: + - name: nri-flex + interval: 60s + config: + name: pixie-health-check + apis: + - event_type: pixieHealth + commands: + - run: curl --insecure -s https://${discovery.ip}:50800/healthz | xargs | awk '{print "cloud_connector_health:"$1}' + split_by: ":" + merge: pixieHealthCheck + - event_type: pixieStatus + commands: + - run: curl --insecure -s https://${discovery.ip}:50800/statusz | awk '{if($1 == ""){ print "cloud_connector_status:OK" } else { print "cloud_connector_status:"$1 }}' + split_by: ":" + merge: pixieHealthCheck + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/scraper-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/scraper-configmap.yaml new file mode 100644 index 0000000000..e43b5227fc --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/kubelet/scraper-configmap.yaml @@ -0,0 +1,18 @@ +{{- if .Values.kubelet.enabled -}} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "nriKubernetes.kubelet.fullname" . }} + namespace: {{ .Release.Namespace }} +data: + nri-kubernetes.yml: | + {{- (merge .Values.common.config (include "newrelic.integrationConfigDefaults" . | fromYaml)) | toYaml | nindent 4 }} + kubelet: + enabled: true + {{- if .Values.kubelet.config }} + {{- toYaml .Values.kubelet.config | nindent 6 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/podsecuritypolicy.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..5b5058511b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/podsecuritypolicy.yaml @@ -0,0 +1,26 @@ +{{- if .Values.rbac.pspEnabled }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: privileged-{{ include "newrelic.common.naming.fullname" . }} +spec: + allowedCapabilities: + - '*' + fsGroup: + rule: RunAsAny + privileged: true + runAsUser: + rule: RunAsAny + seLinux: + rule: RunAsAny + supplementalGroups: + rule: RunAsAny + volumes: + - '*' + hostPID: true + hostIPC: true + hostNetwork: true + hostPorts: + - min: 1 + max: 65536 +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/secret.yaml new file mode 100644 index 0000000000..f558ee86c9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/secret.yaml @@ -0,0 +1,2 @@ +{{- /* Common library will take care of creating the secret or not. */}} +{{- include "newrelic.common.license.secret" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/serviceaccount.yaml new file mode 100644 index 0000000000..f987cc5124 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/values.yaml new file mode 100644 index 0000000000..8cda93b3bb --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-infrastructure/values.yaml @@ -0,0 +1,602 @@ +# -- Override the name of the chart +nameOverride: "" +# -- Override the full name of the release +fullnameOverride: "" + +# -- Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster` +cluster: "" +# -- This set this license key to use. Can be configured also with `global.licenseKey` +licenseKey: "" +# -- In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` +customSecretName: "" +# -- In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` +customSecretLicenseKey: "" + +# -- Images used by the chart for the integration and agents. +# @default -- See `values.yaml` +images: + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + # -- Image for the New Relic Infrastructure Agent sidecar. + # @default -- See `values.yaml` + forwarder: + registry: "" + repository: newrelic/k8s-events-forwarder + tag: 1.57.2 + pullPolicy: IfNotPresent + # -- Image for the New Relic Infrastructure Agent plus integrations. + # @default -- See `values.yaml` + agent: + registry: "" + repository: newrelic/infrastructure-bundle + tag: 3.2.55 + pullPolicy: IfNotPresent + # -- Image for the New Relic Kubernetes integration. + # @default -- See `values.yaml` + integration: + registry: "" + repository: newrelic/nri-kubernetes + tag: + pullPolicy: IfNotPresent + +# -- Config that applies to all instances of the solution: kubelet, ksm, control plane and sidecars. +# @default -- See `values.yaml` +common: + # Configuration entries that apply to all instances of the integration: kubelet, ksm and control plane. + config: + # common.config.interval -- (duration) Intervals larger than 40s are not supported and will cause the NR UI to not + # behave properly. Any non-nil value will override the `lowDataMode` default. + # @default -- `15s` (See [Low data mode](README.md#low-data-mode)) + interval: + # -- Config for filtering ksm and kubelet metrics by namespace. + namespaceSelector: {} + # If you want to include only namespaces with a given label you could do so by adding: + # matchLabels: + # newrelic.com/scrape: true + # Otherwise you can build more complex filters and include or exclude certain namespaces by adding one or multiple + # expressions that are added, for instance: + # matchExpressions: + # - {key: newrelic.com/scrape, operator: NotIn, values: ["false"]} + + # -- Config for the Infrastructure agent. + # Will be used by the forwarder sidecars and the agent running integrations. + # See: https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + agentConfig: {} + +# lowDataMode -- (bool) Send less data by incrementing the interval from `15s` (the default when `lowDataMode` is `false` or `nil`) to `30s`. +# Non-nil values of `common.config.interval` will override this value. +# @default -- `false` (See [Low data mode](README.md#low-data-mode)) +lowDataMode: + +# sink - Configuration for the scraper sink. +sink: + http: + # -- The amount of time the scraper container to probe infra agent sidecar container before giving up and restarting during pod starts. + probeTimeout: 90s + # -- The amount of time the scraper container to backoff when it fails to probe infra agent sidecar. + probeBackoff: 5s + +# kubelet -- Configuration for the DaemonSet that collects metrics from the Kubelet. +# @default -- See `values.yaml` +kubelet: + # -- Enable kubelet monitoring. + # Advanced users only. Setting this to `false` is not supported and will break the New Relic experience. + enabled: true + annotations: {} + # -- Tolerations for the control plane DaemonSet. + # @default -- Schedules in all tainted nodes + tolerations: + - operator: "Exists" + effect: "NoSchedule" + - operator: "Exists" + effect: "NoExecute" + nodeSelector: {} + # -- (bool) Sets pod's hostNetwork. When set bypasses global/common variable + # @default -- Not set + hostNetwork: + affinity: {} + # -- Config for the Infrastructure agent that will forward the metrics to the backend and will run the integrations in this cluster. + # It will be merged with the configuration in `.common.agentConfig`. You can see all the agent configurations in + # [New Relic docs](https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/) + # e.g. you can set `passthrough_environment` int the [config file](https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/configure-infrastructure-agent/#config-file) + # so the agent let use that environment variables to the integrations. + agentConfig: {} + # passthrough_environment: + # - A_ENVIRONMENT_VARIABLE_SET_IN_extraEnv + # - A_ENVIRONMENT_VARIABLE_SET_IN_A_CONFIG_MAP_SET_IN_entraEnvForm + + # -- Add user environment variables to the agent + extraEnv: [] + # -- Add user environment from configMaps or secrets as variables to the agent + extraEnvFrom: [] + # -- Volumes to mount in the containers + extraVolumes: [] + # -- Defines where to mount volumes specified with `extraVolumes` + extraVolumeMounts: [] + initContainers: [] + resources: + limits: + memory: 300M + requests: + cpu: 100m + memory: 150M + config: + # -- Timeout for the kubelet APIs contacted by the integration + timeout: 10s + # -- Number of retries after timeout expired + retries: 3 + # -- Max number of scraper rerun when scraper runtime error happens + scraperMaxReruns: 4 + # port: + # scheme: + +# ksm -- Configuration for the Deployment that collects state metrics from KSM (kube-state-metrics). +# @default -- See `values.yaml` +ksm: + # -- Enable cluster state monitoring. + # Advanced users only. Setting this to `false` is not supported and will break the New Relic experience. + enabled: true + annotations: {} + # -- Tolerations for the KSM Deployment. + # @default -- Schedules in all tainted nodes + tolerations: + - operator: "Exists" + effect: "NoSchedule" + - operator: "Exists" + effect: "NoExecute" + nodeSelector: {} + # -- (bool) Sets pod's hostNetwork. When set bypasses global/common variable + # @default -- Not set + hostNetwork: + # -- Affinity for the KSM Deployment. + # @default -- Deployed in the same node as KSM + affinity: + podAffinity: + preferredDuringSchedulingIgnoredDuringExecution: + - podAffinityTerm: + topologyKey: kubernetes.io/hostname + labelSelector: + matchLabels: + app.kubernetes.io/name: kube-state-metrics + weight: 100 + # -- Config for the Infrastructure agent that will forward the metrics to the backend. It will be merged with the configuration in `.common.agentConfig` + # See: https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + agentConfig: {} + extraEnv: [] + extraEnvFrom: [] + extraVolumes: [] + extraVolumeMounts: [] + initContainers: [] + # -- Resources for the KSM scraper pod. + # Keep in mind that sharding is not supported at the moment, so memory usage for this component ramps up quickly on + # large clusters. + # @default -- 100m/150M -/850M + resources: + limits: + memory: 850M # Bump me up if KSM pod shows restarts. + requests: + cpu: 100m + memory: 150M + config: + # -- Timeout for the ksm API contacted by the integration + timeout: 10s + # -- Number of retries after timeout expired + retries: 3 + # -- if specified autodiscovery is not performed and the specified URL is used + # staticUrl: "http://test.io:8080/metrics" + # -- Label selector that will be used to automatically discover an instance of kube-state-metrics running in the cluster. + selector: "app.kubernetes.io/name=kube-state-metrics" + # -- Scheme to use to connect to kube-state-metrics. Supported values are `http` and `https`. + scheme: "http" + # -- Restrict autodiscovery of the kube-state-metrics endpoint to those using a specific port. If empty or `0`, all endpoints are considered regardless of their port (recommended). + # port: 8080 + # -- Restrict autodiscovery of the kube-state-metrics service to a particular namespace. + # @default -- All namespaces are searched (recommended). + # namespace: "ksm-namespace" + +# controlPlane -- Configuration for the control plane scraper. +# @default -- See `values.yaml` +controlPlane: + # -- Deploy control plane monitoring component. + enabled: true + annotations: {} + # -- Tolerations for the control plane DaemonSet. + # @default -- Schedules in all tainted nodes + tolerations: + - operator: "Exists" + effect: "NoSchedule" + - operator: "Exists" + effect: "NoExecute" + nodeSelector: {} + # -- Affinity for the control plane DaemonSet. + # @default -- Deployed only in master nodes. + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: node-role.kubernetes.io/control-plane + operator: Exists + - matchExpressions: + - key: node-role.kubernetes.io/controlplane + operator: Exists + - matchExpressions: + - key: node-role.kubernetes.io/etcd + operator: Exists + - matchExpressions: + - key: node-role.kubernetes.io/master + operator: Exists + # -- How to deploy the control plane scraper. If autodiscovery is in use, it should be `DaemonSet`. + # Advanced users using static endpoints set this to `Deployment` to avoid reporting metrics twice. + kind: DaemonSet + # -- Run Control Plane scraper with `hostNetwork`. + # `hostNetwork` is required for most control plane configurations, as they only accept connections from localhost. + hostNetwork: true + # -- Config for the Infrastructure agent that will forward the metrics to the backend. It will be merged with the configuration in `.common.agentConfig` + # See: https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + agentConfig: {} + extraEnv: [] + extraEnvFrom: [] + extraVolumes: [] + extraVolumeMounts: [] + initContainers: [] + resources: + limits: + memory: 300M + requests: + cpu: 100m + memory: 150M + config: + # -- Timeout for the Kubernetes APIs contacted by the integration + timeout: 10s + # -- Number of retries after timeout expired + retries: 3 + # -- etcd monitoring configuration + # @default -- Common settings for most K8s distributions. + etcd: + # -- Enable etcd monitoring. Might require manual configuration in some environments. + enabled: true + # Discover etcd pods using the following namespaces and selectors. + # If a pod matches the selectors, the scraper will attempt to reach it through the `endpoints` defined below. + autodiscover: + - selector: "tier=control-plane,component=etcd" + namespace: kube-system + # Set to true to consider only pods sharing the node with the scraper pod. + # This should be set to `true` if Kind is Daemonset, `false` otherwise. + matchNode: true + # Try to reach etcd using the following endpoints. + endpoints: + - url: https://localhost:4001 + insecureSkipVerify: true + auth: + type: bearer + - url: http://localhost:2381 + - selector: "k8s-app=etcd-manager-main" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:4001 + insecureSkipVerify: true + auth: + type: bearer + - selector: "k8s-app=etcd" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:4001 + insecureSkipVerify: true + auth: + type: bearer + # Openshift users might want to remove previous autodiscover entries and add this one instead. + # Manual steps are required to create a secret containing the required TLS certificates to connect to etcd. + # - selector: "app=etcd,etcd=true,k8s-app=etcd" + # namespace: openshift-etcd + # matchNode: true + # endpoints: + # - url: https://localhost:9979 + # insecureSkipVerify: true + # auth: + # type: mTLS + # mtls: + # secretName: secret-name + # secretNamespace: secret-namespace + + # -- staticEndpoint configuration. + # It is possible to specify static endpoint to scrape. If specified 'autodiscover' section is ignored. + # If set the static endpoint should be reachable, otherwise an error will be returned and the integration stops. + # Notice that if deployed as a daemonSet and not as a Deployment setting static URLs could lead to duplicate data + # staticEndpoint: + # url: https://url:port + # insecureSkipVerify: true + # auth: {} + + # -- Scheduler monitoring configuration + # @default -- Common settings for most K8s distributions. + scheduler: + # -- Enable scheduler monitoring. + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-scheduler" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10259 + insecureSkipVerify: true + auth: + type: bearer + - selector: "k8s-app=kube-scheduler" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10259 + insecureSkipVerify: true + auth: + type: bearer + - selector: "app=openshift-kube-scheduler,scheduler=true" + namespace: openshift-kube-scheduler + matchNode: true + endpoints: + - url: https://localhost:10259 + insecureSkipVerify: true + auth: + type: bearer + - selector: "app=openshift-kube-scheduler,scheduler=true" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10259 + insecureSkipVerify: true + auth: + type: bearer + # -- staticEndpoint configuration. + # It is possible to specify static endpoint to scrape. If specified 'autodiscover' section is ignored. + # If set the static endpoint should be reachable, otherwise an error will be returned and the integration stops. + # Notice that if deployed as a daemonSet and not as a Deployment setting static URLs could lead to duplicate data + # staticEndpoint: + # url: https://url:port + # insecureSkipVerify: true + # auth: {} + + # -- Controller manager monitoring configuration + # @default -- Common settings for most K8s distributions. + controllerManager: + # -- Enable controller manager monitoring. + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-controller-manager" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + - selector: "k8s-app=kube-controller-manager" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + - selector: "app=kube-controller-manager,kube-controller-manager=true" + namespace: openshift-kube-controller-manager + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + - selector: "app=kube-controller-manager,kube-controller-manager=true" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + - selector: "app=controller-manager,controller-manager=true" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:10257 + insecureSkipVerify: true + auth: + type: bearer + # mtls: + # secretName: secret-name + # secretNamespace: secret-namespace + # -- staticEndpoint configuration. + # It is possible to specify static endpoint to scrape. If specified 'autodiscover' section is ignored. + # If set the static endpoint should be reachable, otherwise an error will be returned and the integration stops. + # Notice that if deployed as a daemonSet and not as a Deployment setting static URLs could lead to duplicate data + # staticEndpoint: + # url: https://url:port + # insecureSkipVerify: true + # auth: {} + + # -- API Server monitoring configuration + # @default -- Common settings for most K8s distributions. + apiServer: + # -- Enable API Server monitoring + enabled: true + autodiscover: + - selector: "tier=control-plane,component=kube-apiserver" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:8443 + insecureSkipVerify: true + auth: + type: bearer + # Endpoint distributions target: Kind(v1.22.1) + - url: https://localhost:6443 + insecureSkipVerify: true + auth: + type: bearer + - url: http://localhost:8080 + - selector: "k8s-app=kube-apiserver" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:8443 + insecureSkipVerify: true + auth: + type: bearer + - url: http://localhost:8080 + - selector: "app=openshift-kube-apiserver,apiserver=true" + namespace: openshift-kube-apiserver + matchNode: true + endpoints: + - url: https://localhost:8443 + insecureSkipVerify: true + auth: + type: bearer + - url: https://localhost:6443 + insecureSkipVerify: true + auth: + type: bearer + - selector: "app=openshift-kube-apiserver,apiserver=true" + namespace: kube-system + matchNode: true + endpoints: + - url: https://localhost:8443 + insecureSkipVerify: true + auth: + type: bearer + # -- staticEndpoint configuration. + # It is possible to specify static endpoint to scrape. If specified 'autodiscover' section is ignored. + # If set the static endpoint should be reachable, otherwise an error will be returned and the integration stops. + # Notice that if deployed as a daemonSet and not as a Deployment setting static URLs could lead to duplicate data + # staticEndpoint: + # url: https://url:port + # insecureSkipVerify: true + # auth: {} + +# -- Update strategy for the deployed DaemonSets. +# @default -- See `values.yaml` +updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 1 + +# -- Update strategy for the deployed Deployments. +# @default -- `type: Recreate` +strategy: + type: Recreate + +# -- Adds extra attributes to the cluster and all the metrics emitted to the backend. Can be configured also with `global.customAttributes` +customAttributes: {} + +# -- Settings controlling ServiceAccount creation. +# @default -- See `values.yaml` +serviceAccount: + # -- (bool) Whether the chart should automatically create the ServiceAccount objects required to run. + # @default -- `true` + create: + annotations: {} + # If not set and create is true, a name is generated using the fullname template + name: "" + +# -- Additional labels for chart objects. Can be configured also with `global.labels` +labels: {} +# -- Annotations to be added to all pods created by the integration. +podAnnotations: {} +# -- Additional labels for chart pods. Can be configured also with `global.podLabels` +podLabels: {} + +# -- Run the integration with full access to the host filesystem and network. +# Running in this mode allows reporting fine-grained cpu, memory, process and network metrics for your nodes. +privileged: true +# -- Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` +priorityClassName: "" +# -- (bool) Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` +# @default -- `false` +hostNetwork: +# -- Sets security context (at pod level). Can be configured also with `global.podSecurityContext` +podSecurityContext: {} +# -- Sets security context (at container level). Can be configured also with `global.containerSecurityContext` +containerSecurityContext: {} + +# -- Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` +dnsConfig: {} + +# Settings controlling RBAC objects creation. +rbac: + # rbac.create -- Whether the chart should automatically create the RBAC objects required to run. + create: true + # rbac.pspEnabled -- Whether the chart should create Pod Security Policy objects. + pspEnabled: false + +# -- Sets pod/node affinities set almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) +affinity: {} +# -- Sets pod's node selector almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) +nodeSelector: {} +# -- Sets pod's tolerations to node taints almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) +tolerations: [] + +# -- Config files for other New Relic integrations that should run in this cluster. +integrations: {} +# If you wish to monitor services running on Kubernetes you can provide integrations +# configuration under `integrations`. You just need to create a new entry where +# the key is the filename of the configuration file and the value is the content of +# the integration configuration. +# The data is the actual integration configuration as described in the spec here: +# https://docs.newrelic.com/docs/integrations/integrations-sdk/file-specifications/integration-configuration-file-specifications-agent-v180 +# For example, if you wanted to monitor a Redis instance that has a label "app=sampleapp" +# you could do so by adding following entry: +# nri-redis-sampleapp: +# discovery: +# command: +# # Run NRI Discovery for Kubernetes +# # https://github.com/newrelic/nri-discovery-kubernetes +# exec: /var/db/newrelic-infra/nri-discovery-kubernetes +# match: +# label.app: sampleapp +# integrations: +# - name: nri-redis +# env: +# # using the discovered IP as the hostname address +# HOSTNAME: ${discovery.ip} +# PORT: 6379 +# labels: +# env: test + +# -- (bool) Collect detailed metrics from processes running in the host. +# This defaults to true for accounts created before July 20, 2020. +# ref: https://docs.newrelic.com/docs/release-notes/infrastructure-release-notes/infrastructure-agent-release-notes/new-relic-infrastructure-agent-1120 +# @default -- `false` +enableProcessMetrics: + +# Prefix nodes display name with cluster to reduce chances of collisions +# prefixDisplayNameWithCluster: false + +# 'true' will use the node name as the name for the "host", +# note that it may cause data collision if the node name is the same in different clusters +# and prefixDisplayNameWithCluster is not set to true. +# 'false' will use the host name as the name for the "host". +# useNodeNameAsDisplayName: true + +selfMonitoring: + pixie: + # selfMonitoring.pixie.enabled -- Enables the Pixie Health Check nri-flex config. + # This Flex config performs periodic checks of the Pixie /healthz and /statusz endpoints exposed by the Pixie + # Cloud Connector. A status for each endpoint is sent to New Relic in a pixieHealthCheck event. + enabled: false + + +# -- Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port`. Can be configured also with `global.proxy` +proxy: "" + +# -- (bool) Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` +# @default -- `false` +nrStaging: +fedramp: + # -- (bool) Enables FedRAMP. Can be configured also with `global.fedramp.enabled` + # @default -- `false` + enabled: + +# -- (bool) Sets the debug logs to this integration or all integrations if it is set globally. Can be configured also with `global.verboseLog` +# @default -- `false` +verboseLog: diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/.helmignore new file mode 100644 index 0000000000..1ed4e226ea --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/.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/ + +templates/apiservice/job-patch/README.md diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.lock new file mode 100644 index 0000000000..23b2bd33c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-08-30T23:31:11.079152974Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.yaml new file mode 100644 index 0000000000..7fb6703e2b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/Chart.yaml @@ -0,0 +1,25 @@ +apiVersion: v2 +appVersion: 0.13.4 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy the New Relic Kubernetes Metrics Adapter. +home: https://hub.docker.com/r/newrelic/newrelic-k8s-metrics-adapter +icon: https://newrelic.com/assets/newrelic/source/NewRelic-logo-square.svg +keywords: +- infrastructure +- newrelic +- monitoring +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: newrelic-k8s-metrics-adapter +sources: +- https://github.com/newrelic/newrelic-k8s-metrics-adapter +- https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/main/charts/newrelic-k8s-metrics-adapter +version: 1.11.4 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md new file mode 100644 index 0000000000..9c5428201c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md @@ -0,0 +1,139 @@ +[![New Relic Experimental header](https://github.com/newrelic/opensource-website/raw/master/src/images/categories/Experimental.png)](https://opensource.newrelic.com/oss-category/#new-relic-experimental) + +# newrelic-k8s-metrics-adapter + +A Helm chart to deploy the New Relic Kubernetes Metrics Adapter. + +**Homepage:** + +## Source Code + +* +* + +## Requirements + +| Repository | Name | Version | +|------------|------|---------| +| https://helm-charts.newrelic.com | common-library | 1.3.0 | + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Node affinity to use for scheduling. | +| apiServicePatchJob.image | object | See `values.yaml`. | Registry, repository, tag, and pull policy for the job container image. | +| apiServicePatchJob.volumeMounts | list | `[]` | Additional Volume mounts for Cert Job, you might want to mount tmp if Pod Security Policies. | +| apiServicePatchJob.volumes | list | `[]` | Additional Volumes for Cert Job. | +| certManager.enabled | bool | `false` | Use cert manager for APIService certs, rather than the built-in patch job. | +| config.accountID | string | `nil` | New Relic [Account ID](https://docs.newrelic.com/docs/accounts/accounts-billing/account-structure/account-id/) where the configured metrics are sourced from. (**Required**) | +| config.cacheTTLSeconds | int | `30` | Period of time in seconds in which a cached value of a metric is consider valid. | +| config.externalMetrics | string | See `values.yaml` | Contains all the external metrics definition of the adapter. Each key of the externalMetric entry represents the metric name and contains the parameters that defines it. | +| config.nrdbClientTimeoutSeconds | int | 30 | Defines the NRDB client timeout. The maximum allowed value is 120. | +| config.region | string | Automatically detected from `licenseKey`. | New Relic account region. If not set, it will be automatically derived from the License Key. | +| containerSecurityContext | string | `nil` | Configure containerSecurityContext | +| extraEnv | list | `[]` | Array to add extra environment variables | +| extraEnvFrom | list | `[]` | Array to add extra envFrom | +| extraVolumeMounts | list | `[]` | Add extra volume mounts | +| extraVolumes | list | `[]` | Array to add extra volumes | +| fullnameOverride | string | `""` | To fully override common.naming.fullname | +| image | object | See `values.yaml`. | Registry, repository, tag, and pull policy for the container image. | +| image.pullSecrets | list | `[]` | The image pull secrets. | +| nodeSelector | object | `{}` | Node label to use for scheduling. | +| personalAPIKey | string | `nil` | New Relic [Personal API Key](https://docs.newrelic.com/docs/apis/intro-apis/new-relic-api-keys/#user-api-key) (stored in a secret). Used to connect to NerdGraph in order to fetch the configured metrics. (**Required**) | +| podAnnotations | string | `nil` | Additional annotations to apply to the pod(s). | +| podSecurityContext | string | `nil` | Configure podSecurityContext | +| proxy | string | `nil` | Configure proxy for the metrics-adapter. | +| rbac.pspEnabled | bool | `false` | Whether the chart should create Pod Security Policy objects. | +| replicas | int | `1` | Number of replicas in the deployment. | +| resources | object | See `values.yaml` | Resources you wish to assign to the pod. | +| serviceAccount.create | string | `true` | Specifies whether a ServiceAccount should be created for the job and the deployment. false avoids creation, true or empty will create the ServiceAccount | +| serviceAccount.name | string | Automatically generated. | If `serviceAccount.create` this will be the name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. If create is false, a serviceAccount with the given name must exist | +| tolerations | list | `[]` | List of node taints to tolerate (requires Kubernetes >= 1.6) | +| verboseLog | bool | `false` | Enable metrics adapter verbose logs. | + +## Example + +Make sure you have [added the New Relic chart repository.](../../README.md#install) + +Because of metrics configuration, we recommend to use an external values file to deploy the chart. An example with the required parameters looks like: + +```yaml +cluster: ClusterName +personalAPIKey: +config: + accountID: + externalMetrics: + nginx_average_requests: + query: "FROM Metric SELECT average(nginx.server.net.requestsPerSecond) SINCE 2 MINUTES AGO" +``` + +Then, to install this chart, run the following command: + +```sh +helm upgrade --install [release-name] newrelic-k8s-metrics-adapter/newrelic-k8s-metrics-adapter --values [values file path] +``` + +Once deployed the metric `nginx_average_requests` will be available to use by any HPA. This is and example of an HPA yaml using this metric: + +```yaml +kind: HorizontalPodAutoscaler +apiVersion: autoscaling/v2beta2 +metadata: + name: nginx-scaler +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: nginx + minReplicas: 1 + maxReplicas: 10 + metrics: + - type: External + external: + metric: + name: nginx_average_requests + selector: + matchLabels: + k8s.namespaceName: nginx + target: + type: Value + value: 10000 +``` + +The NRQL query that will be run to get the `nginx_average_requests` value will be: + +```sql +FROM Metric SELECT average(nginx.server.net.requestsPerSecond) WHERE clusterName='ClusterName' AND `k8s.namespaceName`='nginx' SINCE 2 MINUTES AGO +``` + +## External Metrics + +An example of multiple external metrics defined: + +```yaml +externalMetrics: + nginx_average_requests: + query: "FROM Metric SELECT average(nginx.server.net.requestsPerSecond) SINCE 2 MINUTES AGO" + container_average_cores_utilization: + query: "FROM Metric SELECT average(`k8s.container.cpuCoresUtilization`) SINCE 2 MINUTES AGO" +``` + +## Resources + +The default set of resources assigned to the newrelic-k8s-metrics-adapter pods is shown below: + +```yaml +resources: + limits: + memory: 80M + requests: + cpu: 100m + memory: 30M +``` + +## Maintainers + +* [juanjjaramillo](https://github.com/juanjjaramillo) +* [csongnr](https://github.com/csongnr) +* [dbudziwojskiNR](https://github.com/dbudziwojskiNR) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md.gotmpl new file mode 100644 index 0000000000..1de8c95531 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/README.md.gotmpl @@ -0,0 +1,107 @@ +[![New Relic Experimental header](https://github.com/newrelic/opensource-website/raw/master/src/images/categories/Experimental.png)](https://opensource.newrelic.com/oss-category/#new-relic-experimental) + +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +{{ template "chart.sourcesSection" . }} + +{{ template "chart.requirementsSection" . }} + +{{ template "chart.valuesSection" . }} + +## Example + +Make sure you have [added the New Relic chart repository.](../../README.md#install) + +Because of metrics configuration, we recommend to use an external values file to deploy the chart. An example with the required parameters looks like: + +```yaml +cluster: ClusterName +personalAPIKey: +config: + accountID: + externalMetrics: + nginx_average_requests: + query: "FROM Metric SELECT average(nginx.server.net.requestsPerSecond) SINCE 2 MINUTES AGO" +``` + +Then, to install this chart, run the following command: + +```sh +helm upgrade --install [release-name] newrelic-k8s-metrics-adapter/newrelic-k8s-metrics-adapter --values [values file path] +``` + +Once deployed the metric `nginx_average_requests` will be available to use by any HPA. This is and example of an HPA yaml using this metric: + +```yaml +kind: HorizontalPodAutoscaler +apiVersion: autoscaling/v2beta2 +metadata: + name: nginx-scaler +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: nginx + minReplicas: 1 + maxReplicas: 10 + metrics: + - type: External + external: + metric: + name: nginx_average_requests + selector: + matchLabels: + k8s.namespaceName: nginx + target: + type: Value + value: 10000 +``` + +The NRQL query that will be run to get the `nginx_average_requests` value will be: + +```sql +FROM Metric SELECT average(nginx.server.net.requestsPerSecond) WHERE clusterName='ClusterName' AND `k8s.namespaceName`='nginx' SINCE 2 MINUTES AGO +``` + +## External Metrics + +An example of multiple external metrics defined: + +```yaml +externalMetrics: + nginx_average_requests: + query: "FROM Metric SELECT average(nginx.server.net.requestsPerSecond) SINCE 2 MINUTES AGO" + container_average_cores_utilization: + query: "FROM Metric SELECT average(`k8s.container.cpuCoresUtilization`) SINCE 2 MINUTES AGO" +``` + +## Resources + +The default set of resources assigned to the newrelic-k8s-metrics-adapter pods is shown below: + +```yaml +resources: + limits: + memory: 80M + requests: + cpu: 100m + memory: 30M +``` + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/ci/test-values.yaml new file mode 100644 index 0000000000..f0f9be1f92 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/ci/test-values.yaml @@ -0,0 +1,14 @@ +global: + cluster: test-cluster + +personalAPIKey: "a21321" +verboseLog: false + +config: + accountID: 111 + region: EU + nrdbClientTimeoutSeconds: 30 + +image: + repository: e2e/newrelic-metrics-adapter + tag: "test" # Defaults to AppVersion diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/_helpers.tpl new file mode 100644 index 0000000000..6a5f765037 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/_helpers.tpl @@ -0,0 +1,57 @@ +{{/* vim: set filetype=mustache: */}} + +{{- /* Allow to change pod defaults dynamically based if we are running in privileged mode or not */ -}} +{{- define "newrelic-k8s-metrics-adapter.securityContext.pod" -}} +{{- if include "newrelic.common.securityContext.pod" . -}} +{{- include "newrelic.common.securityContext.pod" . -}} +{{- else -}} +fsGroup: 1001 +runAsUser: 1001 +runAsGroup: 1001 +{{- end -}} +{{- end -}} + + + +{{/* +Select a value for the region +When this value is empty the New Relic client region will be the default 'US' +*/}} +{{- define "newrelic-k8s-metrics-adapter.region" -}} +{{- if .Values.config.region -}} + {{- .Values.config.region | upper -}} +{{- else if (include "newrelic.common.nrStaging" .) -}} +Staging +{{- else if hasPrefix "eu" (include "newrelic.common.license._licenseKey" .) -}} +EU +{{- end -}} +{{- end -}} + + + +{{- /* +Naming helpers +*/ -}} +{{- define "newrelic-k8s-metrics-adapter.name.apiservice" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "apiservice") }} +{{- end -}} + +{{- define "newrelic-k8s-metrics-adapter.name.apiservice.serviceAccount" -}} +{{- if include "newrelic.common.serviceAccount.create" . -}} + {{- include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "apiservice") -}} +{{- else -}} + {{- include "newrelic.common.serviceAccount.name" . -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-k8s-metrics-adapter.name.apiservice-create" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "apiservice-create") }} +{{- end -}} + +{{- define "newrelic-k8s-metrics-adapter.name.apiservice-patch" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "apiservice-patch") }} +{{- end -}} + +{{- define "newrelic-k8s-metrics-adapter.name.hpa-controller" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "hpa-controller") }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-clusterrolebinding.yaml new file mode 100644 index 0000000000..40bcba8b63 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-clusterrolebinding.yaml @@ -0,0 +1,14 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic.common.naming.fullname" . }}:system:auth-delegator + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:auth-delegator +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-rolebinding.yaml new file mode 100644 index 0000000000..afb5d2d550 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/adapter-rolebinding.yaml @@ -0,0 +1,15 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: kube-system + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: extension-apiserver-authentication-reader +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/apiservice.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/apiservice.yaml new file mode 100644 index 0000000000..8f01b64071 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/apiservice.yaml @@ -0,0 +1,19 @@ +apiVersion: apiregistration.k8s.io/v1 +kind: APIService +metadata: + name: v1beta1.external.metrics.k8s.io + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +{{- if .Values.certManager.enabled }} + annotations: + certmanager.k8s.io/inject-ca-from: {{ printf "%s/%s-root-cert" .Release.Namespace (include "newrelic.common.naming.fullname" .) | quote }} + cert-manager.io/inject-ca-from: {{ printf "%s/%s-root-cert" .Release.Namespace (include "newrelic.common.naming.fullname" .) | quote }} +{{- end }} +spec: + service: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + group: external.metrics.k8s.io + version: v1beta1 + groupPriorityMinimum: 100 + versionPriority: 100 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrole.yaml new file mode 100644 index 0000000000..5c364eb375 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrole.yaml @@ -0,0 +1,26 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - apiregistration.k8s.io + resources: + - apiservices + verbs: + - get + - update +{{- if .Values.rbac.pspEnabled }} + - apiGroups: ['policy'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrolebinding.yaml new file mode 100644 index 0000000000..8aa95792e8 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/clusterrolebinding.yaml @@ -0,0 +1,19 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} +subjects: + - kind: ServiceAccount + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-createSecret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-createSecret.yaml new file mode 100644 index 0000000000..6cf89b79e6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-createSecret.yaml @@ -0,0 +1,55 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: batch/v1 +kind: Job +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice-create" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + template: + metadata: + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice-create" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.image.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: create + image: {{ include "newrelic.common.images.image" ( dict "defaultRegistry" "registry.k8s.io" "imageRoot" .Values.apiServicePatchJob.image "context" .) }} + imagePullPolicy: {{ .Values.apiServicePatchJob.image.pullPolicy }} + args: + - create + - --host={{ include "newrelic.common.naming.fullname" . }},{{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }}.svc + - --namespace={{ .Release.Namespace }} + - --secret-name={{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + - --cert-name=tls.crt + - --key-name=tls.key + {{- with .Values.apiServicePatchJob.volumeMounts }} + volumeMounts: + {{- toYaml . | nindent 10 }} + {{- end }} + {{- with .Values.apiServicePatchJob.volumes }} + volumes: + {{- toYaml . | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + serviceAccountName: {{ include "newrelic-k8s-metrics-adapter.name.apiservice.serviceAccount" . }} + securityContext: + runAsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-patchAPIService.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-patchAPIService.yaml new file mode 100644 index 0000000000..9d651c2108 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/job-patchAPIService.yaml @@ -0,0 +1,53 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: batch/v1 +kind: Job +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice-patch" . }} + annotations: + "helm.sh/hook": post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + template: + metadata: + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice-patch" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.image.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: patch + image: {{ include "newrelic.common.images.image" ( dict "defaultRegistry" "registry.k8s.io" "imageRoot" .Values.apiServicePatchJob.image "context" .) }} + imagePullPolicy: {{ .Values.apiServicePatchJob.image.pullPolicy }} + args: + - patch + - --namespace={{ .Release.Namespace }} + - --secret-name={{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + - --apiservice-name=v1beta1.external.metrics.k8s.io + {{- with .Values.apiServicePatchJob.volumeMounts }} + volumeMounts: + {{- toYaml . | nindent 10 }} + {{- end }} + {{- with .Values.apiServicePatchJob.volumes }} + volumes: + {{- toYaml . | nindent 6 }} + {{- end }} + restartPolicy: OnFailure + serviceAccountName: {{ include "newrelic-k8s-metrics-adapter.name.apiservice.serviceAccount" . }} + securityContext: + runAsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/psp.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/psp.yaml new file mode 100644 index 0000000000..1dd6bc1a6f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/psp.yaml @@ -0,0 +1,49 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled) (.Values.rbac.pspEnabled)) }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + privileged: false + # Required to prevent escalations to root. + # allowPrivilegeEscalation: false + # This is redundant with non-root + disallow privilege escalation, + # but we can provide it for defense in depth. + # requiredDropCapabilities: + # - ALL + # Allow core volume types. + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + 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: + # Forbid adding the root group. + - min: 0 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 0 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/role.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/role.yaml new file mode 100644 index 0000000000..1e870e0828 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/role.yaml @@ -0,0 +1,20 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - secrets + verbs: + - get + - create +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/rolebinding.yaml new file mode 100644 index 0000000000..cbe8bdb72d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} +subjects: + - kind: ServiceAccount + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/serviceaccount.yaml new file mode 100644 index 0000000000..68a3cfd730 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/apiservice/job-patch/serviceaccount.yaml @@ -0,0 +1,18 @@ +{{- $createServiceAccount := include "newrelic.common.serviceAccount.create" . -}} +{{- if (and $createServiceAccount (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic-k8s-metrics-adapter.name.apiservice.serviceAccount" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + # When hooks are sorted by weight and name, kind order gets overwritten, + # then this serviceAccount doesn't get created before dependent objects causing a failure. + # This weight is set, forcing it always to get created before the other objects. + # We submitted this PR to fix the issue: https://github.com/helm/helm/pull/10787 + "helm.sh/hook-weight": "-1" + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/configmap.yaml new file mode 100644 index 0000000000..8e88ad59e4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/configmap.yaml @@ -0,0 +1,19 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + config.yaml: | + accountID: {{ .Values.config.accountID | required "config.accountID is required" }} + {{- with (include "newrelic-k8s-metrics-adapter.region" .) }} + region: {{ . }} + {{- end }} + cacheTTLSeconds: {{ .Values.config.cacheTTLSeconds | default "0" }} + {{- with .Values.config.externalMetrics }} + externalMetrics: + {{- toYaml . | nindent 6 }} + {{- end }} + nrdbClientTimeoutSeconds: {{ .Values.config.nrdbClientTimeoutSeconds | default "30" }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/deployment.yaml new file mode 100644 index 0000000000..1b96459a5d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/deployment.yaml @@ -0,0 +1,113 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + replicas: {{ .Values.replicas }} + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + checksum/secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} + {{- if .Values.podAnnotations }} + {{- toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + {{- with include "newrelic-k8s-metrics-adapter.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.image.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: {{ include "newrelic.common.naming.name" . }} + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.image "context" .) }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 10 }} + {{- end }} + args: + - --tls-cert-file=/tmp/k8s-metrics-adapter/serving-certs/tls.crt + - --tls-private-key-file=/tmp/k8s-metrics-adapter/serving-certs/tls.key + {{- if .Values.verboseLog }} + - --v=10 + {{- else }} + - --v=1 + {{- end }} + readinessProbe: + httpGet: + scheme: HTTPS + path: /healthz + port: 6443 + initialDelaySeconds: 1 + {{- if .Values.resources }} + resources: + {{- toYaml .Values.resources | nindent 10 }} + {{- end }} + env: + - name: CLUSTER_NAME + value: {{ include "newrelic.common.cluster" . }} + - name: NEWRELIC_API_KEY + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.naming.fullname" . }} + key: personalAPIKey + {{- with (include "newrelic.common.proxy" .) }} + - name: HTTPS_PROXY + value: {{ . }} + {{- end }} + {{- with .Values.extraEnv }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.extraEnvFrom }} + envFrom: {{ toYaml . | nindent 8 }} + {{- end }} + volumeMounts: + - name: tls-key-cert-pair + mountPath: /tmp/k8s-metrics-adapter/serving-certs/ + - name: config + mountPath: /etc/newrelic/adapter/ + {{- with .Values.extraVolumeMounts }} + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: tls-key-cert-pair + secret: + secretName: {{ include "newrelic-k8s-metrics-adapter.name.apiservice" . }} + - name: config + configMap: + name: {{ include "newrelic.common.naming.fullname" . }} + {{- with .Values.extraVolumes }} + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrole.yaml new file mode 100644 index 0000000000..402fece01e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrole.yaml @@ -0,0 +1,15 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic.common.naming.fullname" . }}:external-metrics + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: +- apiGroups: + - external.metrics.k8s.io + resources: + - "*" + verbs: + - list + - get + - watch diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrolebinding.yaml new file mode 100644 index 0000000000..390fab452b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/hpa-clusterrolebinding.yaml @@ -0,0 +1,14 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic-k8s-metrics-adapter.name.hpa-controller" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic.common.naming.fullname" . }}:external-metrics +subjects: +- kind: ServiceAccount + name: horizontal-pod-autoscaler + namespace: kube-system diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/secret.yaml new file mode 100644 index 0000000000..09a70ab655 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/secret.yaml @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: Secret +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +type: Opaque +stringData: + personalAPIKey: {{ .Values.personalAPIKey | required "personalAPIKey must be set" | quote }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/service.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/service.yaml new file mode 100644 index 0000000000..82015830c6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/service.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +kind: Service +metadata: + namespace: {{ .Release.Namespace }} + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + ports: + - port: 443 + targetPort: 6443 + selector: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 4 }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/serviceaccount.yaml new file mode 100644 index 0000000000..b1e74523e5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- if include "newrelic.common.serviceAccount.annotations" . }} + annotations: + {{- include "newrelic.common.serviceAccount.annotations" . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/apiservice_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/apiservice_test.yaml new file mode 100644 index 0000000000..086160edc5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/apiservice_test.yaml @@ -0,0 +1,22 @@ +suite: test naming helper for APIService's certmanager annotations and service name +templates: + - templates/apiservice/apiservice.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: Annotations are correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 11111111 + certManager: + enabled: true + asserts: + - matchRegex: + path: metadata.annotations["certmanager.k8s.io/inject-ca-from"] + pattern: ^my-namespace\/.*-root-cert + - matchRegex: + path: metadata.annotations["cert-manager.io/inject-ca-from"] + pattern: ^my-namespace\/.*-root-cert diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/common_extra_naming_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/common_extra_naming_test.yaml new file mode 100644 index 0000000000..82098ba1c4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/common_extra_naming_test.yaml @@ -0,0 +1,27 @@ +suite: test naming helpers +templates: + - templates/adapter-clusterrolebinding.yaml + - templates/hpa-clusterrole.yaml + - templates/hpa-clusterrolebinding.yaml + - templates/apiservice/job-patch/clusterrole.yaml + - templates/apiservice/job-patch/clusterrolebinding.yaml + - templates/apiservice/job-patch/job-createSecret.yaml + - templates/apiservice/job-patch/job-patchAPIService.yaml + - templates/apiservice/job-patch/psp.yaml + - templates/apiservice/job-patch/rolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: default values has its name correctly defined + set: + cluster: test-cluster + personalAPIKey: 21321 + config: + accountID: 11111111 + rbac: + pspEnabled: true + asserts: + - matchRegex: + path: metadata.name + pattern: ^.*(-apiservice|-hpa-controller|:external-metrics|:system:auth-delegator) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/configmap_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/configmap_test.yaml new file mode 100644 index 0000000000..90b8798a7b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/configmap_test.yaml @@ -0,0 +1,104 @@ +suite: test configmap region helper and externalMetrics +templates: + - templates/configmap.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: has the correct region when defined in local values + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - equal: + path: data["config.yaml"] + value: | + accountID: 111 + region: A-REGION + cacheTTLSeconds: 30 + nrdbClientTimeoutSeconds: 30 + - it: has the correct region when global staging + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + global: + nrStaging: true + asserts: + - equal: + path: data["config.yaml"] + value: | + accountID: 111 + region: Staging + cacheTTLSeconds: 30 + nrdbClientTimeoutSeconds: 30 + - it: has the correct region when global values and licenseKey is from eu + set: + personalAPIKey: 21321 + licenseKey: eu-whatever + cluster: test-cluster + config: + accountID: 111 + global: + aRandomGlobalValue: true + asserts: + - equal: + path: data["config.yaml"] + value: | + accountID: 111 + region: EU + cacheTTLSeconds: 30 + nrdbClientTimeoutSeconds: 30 + - it: has the correct region when no global values exist and licenseKey is from eu + set: + personalAPIKey: 21321 + cluster: test-cluster + licenseKey: eu-whatever + config: + accountID: 111 + asserts: + - equal: + path: data["config.yaml"] + value: | + accountID: 111 + region: EU + cacheTTLSeconds: 30 + nrdbClientTimeoutSeconds: 30 + - it: has no region when not defined and licenseKey is not from eu + set: + personalAPIKey: 21321 + cluster: test-cluster + licenseKey: us-whatever + config: + accountID: 111 + asserts: + - equal: + path: data["config.yaml"] + value: | + accountID: 111 + cacheTTLSeconds: 30 + nrdbClientTimeoutSeconds: 30 + - it: has externalMetrics when defined + set: + personalAPIKey: 21321 + cluster: test-cluster + licenseKey: us-whatever + config: + accountID: 111 + externalMetrics: + nginx_average_requests: + query: "FROM Metric SELECT average(nginx.server.net.requestsPerSecond)" + asserts: + - equal: + path: data["config.yaml"] + value: | + accountID: 111 + cacheTTLSeconds: 30 + externalMetrics: + nginx_average_requests: + query: FROM Metric SELECT average(nginx.server.net.requestsPerSecond) + nrdbClientTimeoutSeconds: 30 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/deployment_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/deployment_test.yaml new file mode 100644 index 0000000000..7a18987907 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/deployment_test.yaml @@ -0,0 +1,99 @@ +suite: test deployent images +release: + name: my-release + namespace: my-namespace +tests: + - it: has the correct image + set: + global: + cluster: test-cluster + personalAPIKey: 21321 + image: + repository: newrelic/newrelic-k8s-metrics-adapter + tag: "latest" + pullSecrets: + - name: regsecret + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: spec.template.spec.containers[0].image + pattern: ^.*newrelic/newrelic-k8s-metrics-adapter:latest + template: templates/deployment.yaml + - equal: + path: spec.template.spec.imagePullSecrets + value: + - name: regsecret + template: templates/deployment.yaml + - it: correctly uses the cluster helper + set: + personalAPIKey: 21321 + config: + accountID: 111 + region: A-REGION + cluster: a-cluster + asserts: + - equal: + path: spec.template.spec.containers[0].env[0].value + value: a-cluster + template: templates/deployment.yaml + - it: correctly uses common.securityContext.podDefaults + set: + personalAPIKey: 21321 + config: + accountID: 111 + region: A-REGION + cluster: a-cluster + asserts: + - equal: + path: spec.template.spec.securityContext + value: + fsGroup: 1001 + runAsGroup: 1001 + runAsUser: 1001 + template: templates/deployment.yaml + - it: correctly uses common.proxy + set: + personalAPIKey: 21321 + config: + accountID: 111 + region: A-REGION + cluster: a-cluster + proxy: localhost:1234 + asserts: + - equal: + path: spec.template.spec.containers[0].env[2].value + value: localhost:1234 + template: templates/deployment.yaml + + - it: has a linux node selector by default + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + template: templates/deployment.yaml + + - it: has a linux node selector and additional selectors + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue + template: templates/deployment.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/hpa_clusterrolebinding_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/hpa_clusterrolebinding_test.yaml new file mode 100644 index 0000000000..4fba87fbe1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/hpa_clusterrolebinding_test.yaml @@ -0,0 +1,18 @@ +suite: test naming helper for clusterRolebBinding roleRef +templates: + - templates/hpa-clusterrolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: roleRef.name has its name correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: roleRef.name + pattern: ^.*:external-metrics diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_cluster_rolebinding_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_cluster_rolebinding_test.yaml new file mode 100644 index 0000000000..dd582313ea --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_cluster_rolebinding_test.yaml @@ -0,0 +1,22 @@ +suite: test job-patch RoleBinding and ClusterRoleBinding rendering and roleRef/Subjects names +templates: + - templates/apiservice/job-patch/rolebinding.yaml + - templates/apiservice/job-patch/clusterrolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: roleRef apiGroup and Subjets are correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: roleRef.name + pattern: ^.*-apiservice + - matchRegex: + path: subjects[0].name + pattern: ^.*-apiservice diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_clusterrole_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_clusterrole_test.yaml new file mode 100644 index 0000000000..33a1eaa731 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_clusterrole_test.yaml @@ -0,0 +1,20 @@ +suite: test job-patch clusterRole rule resourceName and rendering +templates: + - templates/apiservice/job-patch/clusterrole.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: PodSecurityPolicy rule resourceName is correctly defined + set: + rbac: + pspEnabled: true + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: rules[1].resourceNames[0] + pattern: ^.*-apiservice diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_common_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_common_test.yaml new file mode 100644 index 0000000000..91cd791d1c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_common_test.yaml @@ -0,0 +1,27 @@ +suite: test labels and rendering for job-batch objects +templates: + - templates/apiservice/job-patch/clusterrole.yaml + - templates/apiservice/job-patch/clusterrolebinding.yaml + - templates/apiservice/job-patch/job-createSecret.yaml + - templates/apiservice/job-patch/job-patchAPIService.yaml + - templates/apiservice/job-patch/psp.yaml + - templates/apiservice/job-patch/role.yaml + - templates/apiservice/job-patch/rolebinding.yaml + - templates/apiservice/job-patch/serviceaccount.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: If customTLSCertificate and Certmanager enabled do not render + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + customTLSCertificate: a-tls-cert + certManager: + enabled: true + asserts: + - hasDocuments: + count: 0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_createsecret_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_createsecret_test.yaml new file mode 100644 index 0000000000..6db79234fd --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_createsecret_test.yaml @@ -0,0 +1,47 @@ +suite: test naming helper for job-createSecret +templates: + - templates/apiservice/job-patch/job-createSecret.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: spec metadata name is is correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - equal: + path: spec.template.metadata.name + value: my-release-newrelic-k8s-metrics-adapter-apiservice-create + - it: container args are correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: spec.template.spec.containers[0].args[1] + pattern: --host=.*,.*\.my-namespace.svc + - matchRegex: + path: spec.template.spec.containers[0].args[3] + pattern: --secret-name=.*-apiservice + - it: has the correct image + set: + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + personalAPIKey: 21321 + apiServicePatchJob: + image: + repository: registry.k8s.io/ingress-nginx/kube-webhook-certgen + tag: "latest" + asserts: + - matchRegex: + path: spec.template.spec.containers[0].image + pattern: ^.*registry.k8s.io/ingress-nginx/kube-webhook-certgen:latest diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_patchapiservice_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_patchapiservice_test.yaml new file mode 100644 index 0000000000..0be0833135 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_patch_job_patchapiservice_test.yaml @@ -0,0 +1,56 @@ +suite: test naming helper for job-patchAPIService +templates: + - templates/apiservice/job-patch/job-patchAPIService.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: spec metadata name is is correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: spec.template.metadata.name + pattern: .*-apiservice-patch$ + - it: container args are correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: spec.template.spec.containers[0].args[2] + pattern: ^--secret-name=.*-apiservice + + - it: serviceAccountName is correctly defined + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - matchRegex: + path: spec.template.spec.serviceAccountName + pattern: .*-apiservice$ + - it: has the correct image + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + apiServicePatchJob: + image: + repository: registry.k8s.io/ingress-nginx/kube-webhook-certgen + tag: "latest" + asserts: + - matchRegex: + path: spec.template.spec.containers[0].image + pattern: .*registry.k8s.io/ingress-nginx/kube-webhook-certgen:latest$ diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_serviceaccount_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_serviceaccount_test.yaml new file mode 100644 index 0000000000..9b6207c350 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/job_serviceaccount_test.yaml @@ -0,0 +1,79 @@ +suite: test job' serviceAccount +templates: + - templates/apiservice/job-patch/job-createSecret.yaml + - templates/apiservice/job-patch/job-patchAPIService.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: RBAC points to the service account that is created by default + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: my-release-newrelic-k8s-metrics-adapter-apiservice + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: sa-test + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: default + + - it: has a linux node selector by default + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + + - it: has a linux node selector and additional selectors + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/rbac_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/rbac_test.yaml new file mode 100644 index 0000000000..78884c0229 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/tests/rbac_test.yaml @@ -0,0 +1,50 @@ +suite: test RBAC creation +templates: + - templates/apiservice/job-patch/rolebinding.yaml + - templates/apiservice/job-patch/clusterrolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: RBAC points to the service account that is created by default + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: subjects[0].name + value: my-release-newrelic-k8s-metrics-adapter-apiservice + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: subjects[0].name + value: sa-test + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + personalAPIKey: 21321 + cluster: test-cluster + config: + accountID: 111 + region: A-REGION + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: subjects[0].name + value: default diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/values.yaml new file mode 100644 index 0000000000..5c610f7926 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-k8s-metrics-adapter/values.yaml @@ -0,0 +1,156 @@ +# IMPORTANT: The Kubernetes cluster name +# https://docs.newrelic.com/docs/kubernetes-monitoring-integration +# +# licenseKey: +# cluster: +# IMPORTANT: the previous values can also be set as global so that they +# can be shared by other newrelic product's charts. +# +# global: +# licenseKey: +# cluster: +# nrStaging: + +# -- New Relic [Personal API Key](https://docs.newrelic.com/docs/apis/intro-apis/new-relic-api-keys/#user-api-key) (stored in a secret). Used to connect to NerdGraph in order to fetch the configured metrics. (**Required**) +personalAPIKey: + +# -- Enable metrics adapter verbose logs. +verboseLog: false + +config: + # -- New Relic [Account ID](https://docs.newrelic.com/docs/accounts/accounts-billing/account-structure/account-id/) where the configured metrics are sourced from. (**Required**) + accountID: + + # config.region -- New Relic account region. If not set, it will be automatically derived from the License Key. + # @default -- Automatically detected from `licenseKey`. + region: + # For US-based accounts, the region is: `US`. + # For EU-based accounts, the region is: `EU`. + # For Staging accounts, the region is: 'Staging' this is also automatically derived form `global.nrStaging` + + + # config.cacheTTLSeconds -- Period of time in seconds in which a cached value of a metric is consider valid. + cacheTTLSeconds: 30 + # Not setting it or setting it to '0' disables the cache. + + # config.externalMetrics -- Contains all the external metrics definition of the adapter. Each key of the externalMetric entry represents the metric name and contains the parameters that defines it. + # @default -- See `values.yaml` + externalMetrics: + # Names cannot contain uppercase characters and + # "/" or "%" characters. + # my_external_metric_name_example: + # + # NRQL query that will executed to obtain the metric value. + # The query must return just one value so is recommended to use aggregator functions like average or latest. + # Default time span for aggregator func is 1h so is recommended to use the SINCE clause to reduce the time span. + # query: "FROM Metric SELECT average(`k8s.container.cpuCoresUtilization`) SINCE 2 MINUTES AGO" + # + # By default a cluster filter is added to the query to ensure no cross cluster metrics are taking into account. + # The added filter is equivalent to WHERE `clusterName`=. + # If metrics are not from the cluster use removeClusterFilter. Default value for this parameter is false. + # removeClusterFilter: false + + # config.nrdbClientTimeoutSeconds -- Defines the NRDB client timeout. The maximum allowed value is 120. + # @default -- 30 + nrdbClientTimeoutSeconds: 30 + +# image -- Registry, repository, tag, and pull policy for the container image. +# @default -- See `values.yaml`. +image: + registry: + repository: newrelic/newrelic-k8s-metrics-adapter + tag: "" + pullPolicy: IfNotPresent + # It is possible to specify docker registry credentials. + # See https://kubernetes.io/docs/concepts/containers/images/#specifying-imagepullsecrets-on-a-pod + # image.pullSecrets -- The image pull secrets. + pullSecrets: [] + # - name: regsecret + +# -- Number of replicas in the deployment. +replicas: 1 + +# -- Resources you wish to assign to the pod. +# @default -- See `values.yaml` +resources: + limits: + memory: 80M + requests: + cpu: 100m + memory: 30M + +serviceAccount: + # -- Specifies whether a ServiceAccount should be created for the job and the deployment. + # false avoids creation, true or empty will create the ServiceAccount + # @default -- `true` + create: + # -- If `serviceAccount.create` this will be the name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + # If create is false, a serviceAccount with the given name must exist + # @default -- Automatically generated. + name: + +# -- Configure podSecurityContext +podSecurityContext: + +# -- Configure containerSecurityContext +containerSecurityContext: + +# -- Array to add extra environment variables +extraEnv: [] +# -- Array to add extra envFrom +extraEnvFrom: [] +# -- Array to add extra volumes +extraVolumes: [] +# -- Add extra volume mounts +extraVolumeMounts: [] + +# -- Additional annotations to apply to the pod(s). +podAnnotations: + +# Due to security restrictions, some users might require to use a https proxy to route traffic over the internet. +# In this specific case, when the metrics adapter sends a request to the New Relic backend. If this is the case +# for you, set this value to your http proxy endpoint. +# -- Configure proxy for the metrics-adapter. +proxy: + +# Pod scheduling priority +# Ref: https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ +# priorityClassName: high-priority + +# fullnameOverride -- To fully override common.naming.fullname +fullnameOverride: "" +# -- Node affinity to use for scheduling. +affinity: {} +# -- Node label to use for scheduling. +nodeSelector: {} +# -- List of node taints to tolerate (requires Kubernetes >= 1.6) +tolerations: [] + +apiServicePatchJob: + # apiServicePatchJob.image -- Registry, repository, tag, and pull policy for the job container image. + # @default -- See `values.yaml`. + image: + registry: # defaults to registry.k8s.io + repository: ingress-nginx/kube-webhook-certgen + tag: v1.3.0 + pullPolicy: IfNotPresent + + # -- Additional Volumes for Cert Job. + volumes: [] + # - name: tmp + # emptyDir: {} + + # -- Additional Volume mounts for Cert Job, you might want to mount tmp if Pod Security Policies. + volumeMounts: [] + # - name: tmp + # mountPath: /tmp + # Enforce a read-only root. + +certManager: + # -- Use cert manager for APIService certs, rather than the built-in patch job. + enabled: false + +rbac: + # rbac.pspEnabled -- Whether the chart should create Pod Security Policy objects. + pspEnabled: false diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.lock new file mode 100644 index 0000000000..064abf8aae --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.2.0 +digest: sha256:fa87cb007564a39a72739a3e850a91d6b03c0fc27a1115deac042b3ef77b4142 +generated: "2024-07-17T19:29:15.951407+05:30" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.yaml new file mode 100644 index 0000000000..9163e72ddc --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/Chart.yaml @@ -0,0 +1,20 @@ +apiVersion: v2 +appVersion: 2.0.2 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.2.0 +description: A Helm chart to deploy New Relic Kubernetes Logging as a DaemonSet, supporting + both Linux and Windows nodes and containers +home: https://github.com/newrelic/kubernetes-logging +icon: https://newrelic.com/assets/newrelic/source/NewRelic-logo-square.svg +keywords: +- logging +- newrelic +maintainers: +- email: logging-team@newrelic.com + name: jsubirat +- name: danybmx +- name: sdaubin +name: newrelic-logging +version: 1.23.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/README.md new file mode 100644 index 0000000000..1635b0d863 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/README.md @@ -0,0 +1,268 @@ +# newrelic-logging + + +## Chart Details +New Relic offers a [Fluent Bit](https://fluentbit.io/) output [plugin](https://github.com/newrelic/newrelic-fluent-bit-output) to easily forward your logs to [New Relic Logs](https://docs.newrelic.com/docs/logs/new-relic-logs/get-started/introduction-new-relic-logs). This plugin is also provided in a standalone Docker image that can be installed in a [Kubernetes](https://kubernetes.io/) cluster in the form of a [DaemonSet](https://kubernetes.io/docs/concepts/workloads/controllers/daemonset/), which we refer as the Kubernetes plugin. + +This document explains how to install it in your cluster using our [Helm](https://helm.sh/) chart. + + +## Install / Upgrade / Uninstall instructions +Despite the `newrelic-logging` chart being able to work standalone, we recommend installing it as part of the [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) chart. The best way of doing so is through the guided installation process documented [here](https://docs.newrelic.com//docs/kubernetes-pixie/kubernetes-integration/installation/kubernetes-integration-install-configure/). This guided install can generate the Helm 3 commands required to install it (select "Helm 3" in Step 3 from the previous documentation link). You can also opt to install it manually using Helm by following [these steps](https://docs.newrelic.com//docs/kubernetes-pixie/kubernetes-integration/installation/install-kubernetes-integration-using-helm/#install-k8-helm). To uninstall it, refer to the steps outlined in [this page](https://docs.newrelic.com/docs/kubernetes-pixie/kubernetes-integration/uninstall-kubernetes/). + +### Installing or updating the helm New Relic repository + +To install the repo you can run: +``` +helm repo add newrelic https://helm-charts.newrelic.com +``` + +To update the repo you can run: +``` +helm repo update newrelic +``` + +## Configuration + +### How to configure the chart +The `newrelic-logging` chart can be installed either alone or as part of the [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) chart (recommended). The chart default settings should be suitable for most users. Nevertheless, you may be interested in overriding the defaults, either by passing them through a `values-newrelic.yaml` file or via the command line when installing the chart. Depending on how you installed it, you'll need to specify the `newrelic-logging`-specific configuration values using the chart name (`newrelic-logging`) as a prefix. In the table below, you can find a quick reference of how to configure the chart in these scenarios. The example depicts how you'd specify the mandatory `licenseKey` and `cluster` settings and how you'd override the `fluentBit.retryLimit` setting to `10`. + + + + + + + + + + + + + + + + + +
Installation methodConfiguration via values.yamlConfiguration via command line
Standalone newrelic-logging + + +``` +# values-newrelic.yaml configuration contents + +licenseKey: _YOUR_NEW_RELIC_LICENSE_KEY_ +cluster: _K8S_CLUSTER_NAME_ + +fluentBit: + retryLimit: 10 +``` + +``` +# Install / upgrade command + +helm upgrade --install newrelic-logging newrelic/newrelic-logging \ +--namespace newrelic \ +--create-namespace \ +-f values-newrelic.yaml +``` + + +``` +# Install / upgrade command + +helm upgrade --install newrelic-logging newrelic/newrelic-logging \ +--namespace=newrelic \ +--set licenseKey=_YOUR_NEW_RELIC_LICENSE_KEY_ \ +--set cluster=_K8S_CLUSTER_NAME_ \ +--set fluentBit.retryLimit=10 +``` +
As part of nri-bundle + +``` +# values-newrelic.yaml configuration contents + +# General settings that apply to all the child charts +global: + licenseKey: _YOUR_NEW_RELIC_LICENSE_KEY_ + cluster: _K8S_CLUSTER_NAME_ + +# Specific configuration for the newrelic-logging child chart +newrelic-logging: + fluentBit: + retryLimit: 10 +``` + +``` +# Install / upgrade command + +helm upgrade --install newrelic-bundle newrelic/nri-bundle \ + --namespace newrelic \ + --create-namespace \ + -f values-newrelic.yaml \ +``` + + +``` +# Install / upgrade command + +helm upgrade --install newrelic-bundle newrelic/nri-bundle \ +--namespace=newrelic \ +--set global.licenseKey=_YOUR_NEW_RELIC_LICENSE_KEY_ \ +--set global.cluster=_K8S_CLUSTER_NAME_ \ +--set newrelic-logging.fluentBit.retryLimit=10 +``` +
+ + +### Supported configuration parameters +See [values.yaml](values.yaml) for the default values + +| Parameter | Description | Default | +| ------------------------------------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | ------------------------------------------------------------------------------- | +| `global.cluster` - `cluster` | The cluster name for the Kubernetes cluster. | | +| `global.licenseKey` - `licenseKey` | The [license key](https://docs.newrelic.com/docs/accounts/install-new-relic/account-setup/license-key) for your New Relic Account. This will be the preferred configuration option if both `licenseKey` and `customSecret*` values are specified. | | +| `global.customSecretName` - `customSecretName` | Name of the Secret object where the license key is stored | | +| `global.customSecretLicenseKey` - `customSecretLicenseKey` | Key in the Secret object where the license key is stored. | | +| `global.fargate` | Must be set to `true` when deploying in an EKS Fargate environment. Prevents DaemonSet pods from being scheduled in Fargate nodes. | | +| `global.lowDataMode` - `lowDataMode` | If `true`, send minimal attributes on Kubernetes logs. Labels and annotations are not sent when lowDataMode is enabled. | `false` | +| `rbac.create` | Enable Role-based authentication | `true` | +| `rbac.pspEnabled` | Enable pod security policy support | `false` | +| `image.repository` | The container to pull. | `newrelic/newrelic-fluentbit-output` | +| `image.pullPolicy` | The pull policy. | `IfNotPresent` | +| `image.pullSecrets` | Image pull secrets. | `nil` | +| `image.tag` | The version of the container to pull. | See value in [values.yaml]` | +| `exposedPorts` | Any ports you wish to expose from the pod. Ex. 2020 for metrics | `[]` | +| `resources` | Any resources you wish to assign to the pod. | See Resources below | +| `priorityClassName` | Scheduling priority of the pod | `nil` | +| `nodeSelector` | Node label to use for scheduling on Linux nodes | `{ kubernetes.io/os: linux }` | +| `windowsNodeSelector` | Node label to use for scheduling on Windows nodes | `{ kubernetes.io/os: windows, node.kubernetes.io/windows-build: BUILD_NUMBER }` | +| `tolerations` | List of node taints to tolerate (requires Kubernetes >= 1.6) | See Tolerations below | +| `updateStrategy` | Strategy for DaemonSet updates (requires Kubernetes >= 1.6) | `RollingUpdate` | +| `extraVolumeMounts` | Additional DaemonSet volume mounts | `[]` | +| `extraVolumes` | Additional DaemonSet volumes | `[]` | +| `initContainers` | [Init containers](https://kubernetes.io/docs/concepts/workloads/pods/init-containers/) that will be executed before the actual container in charge of shipping logs to New Relic is initialized. Use this if you are using a custom Fluent Bit configuration that requires downloading certain files inside the volumes being accessed by the log-shipping pod. | `[]` | +| `windows.initContainers` | [Init containers](https://kubernetes.io/docs/concepts/workloads/pods/init-containers/) that will be executed before the actual container in charge of shipping logs to New Relic is initialized. Use this if you are using a custom Fluent Bit configuration that requires downloading certain files inside the volumes being accessed by the log-shipping pod. | `[]` | +| `serviceAccount.create` | If true, a service account would be created and assigned to the deployment | `true` | +| `serviceAccount.name` | The service account to assign to the deployment. If `serviceAccount.create` is true then this name will be used when creating the service account | | +| `serviceAccount.annotations` | The annotations to add to the service account if `serviceAccount.create` is set to true. | | +| `global.nrStaging` - `nrStaging` | Send data to staging (requires a staging license key) | `false` | +| `fluentBit.path` | Node path logs are forwarded from. Patterns are supported, as well as specifying multiple paths/patterns separated by commas. | `/var/log/containers/*.log` | +| `fluentBit.linuxMountPath` | The path mounted on linux Fluent-Bit pods to read logs from. Defaults to /var because some engines write the logs to /var/log and others to /var/lib (symlinked to /var/log) so Fluent-Bit need access to both in those cases | `/var` | +| `fluentBit.db` | Node path used by Fluent Bit to store a database file to keep track of monitored files and offsets. | `/var/log/containers/*.log` | +| `fluentBit.k8sBufferSize` | Set the buffer size for HTTP client when reading responses from Kubernetes API server. A value of 0 results in no limit and the buffer will expand as needed. | `32k` | +| `fluentBit.k8sLoggingExclude` | Set to "true" to allow excluding pods by adding the annotation `fluentbit.io/exclude: "true"` to pods you wish to exclude. | `false` | +| `fluentBit.additionalEnvVariables` | Additional environmental variables for fluentbit pods | `[]]` | +| `fluentBit.persistence.mode` | The [persistence mode](#Fluent-Bit-persistence-modes) you want to use, options are "hostPath", "none" or "persistentVolume" (this last one available only for linux) | +| `fluentBit.persistence.persistentVolume.storageClass` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), indicates the storage class that will be used for create the PersistentVolume and PersistentVolumeClaim. | | +| `fluentBit.persistence.persistentVolume.size` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), indicates the capacity for the PersistentVolume and PersistentVolumeClaim | 10Gi | +| `fluentBit.persistence.persistentVolume.dynamicProvisioning` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), indicates if the storage class used provide dynamic provisioning. If it does, only the PersistentVolumeClaim will be created. | true | +| `fluentBit.persistence.persistentVolume.existingVolume` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), indicates and existing volume in case you want to reuse one, bear in mind that it should allow ReadWriteMany access mode. A PersistentVolumeClaim will be created using it. | | +| `fluentBit.persistence.persistentVolume.existingVolumeClaim` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), indicates and existing volume claim that will be used on the daemonset. It should allow ReadWriteMany access mode. | | +| `fluentBit.persistence.persistentVolume.annotations.volume` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), allows to add annotations to the PersistentVolume (if created). | | +| `fluentBit.persistence.persistentVolume.annotations.claim` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), allows to add annotations to the PersistentVolumeClaim (if created). | | +| `fluentBit.persistence.persistentVolume.extra.volume` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), allows to add extra properties to the PersistentVolume (if created). | | +| `fluentBit.persistence.persistentVolume.extra.claim` | On "persistentVolume" [persistence mode](#Fluent-Bit-persistence-modes), allows to add extra properties to the PersistentVolumeClaim (if created). | | +| `daemonSet.annotations` | The annotations to add to the `DaemonSet`. | | +| `podAnnotations` | The annotations to add to the `DaemonSet` created `Pod`s. | | +| `hostNetwork` | Set the hostNetwork property for fluentbit pods. | | +| `enableLinux` | Enable log collection from Linux containers. This is the default behavior. In case you are only interested of collecting logs from Windows containers, set this to `false`. | `true` | +| `enableWindows` | Enable log collection from Windows containers. Please refer to the [Windows support](#windows-support) section for more details. | `false` | +| `fluentBit.config.service` | Contains fluent-bit.conf Service config | | +| `fluentBit.config.inputs` | Contains fluent-bit.conf Inputs config | | +| `fluentBit.config.extraInputs` | Contains extra fluent-bit.conf Inputs config | | +| `fluentBit.config.filters` | Contains fluent-bit.conf Filters config | | +| `fluentBit.config.extraFilters` | Contains extra fluent-bit.conf Filters config | | +| `fluentBit.config.lowDataModeFilters` | Contains fluent-bit.conf Filters config for lowDataMode | | +| `fluentBit.config.outputs` | Contains fluent-bit.conf Outputs config | | +| `fluentBit.config.extraOutputs` | Contains extra fluent-bit.conf Outputs config | | +| `fluentBit.config.parsers` | Contains parsers.conf Parsers config | | +| `fluentBit.retryLimit` | Amount of times to retry sending a given batch of logs to New Relic. This prevents data loss if there is a temporary network disruption, if a request to the Logs API is lost or when receiving a recoverable HTTP response. Set it to "False" for unlimited retries. | 5 | +| `fluentBit.sendMetrics` | Enable the collection of Fluent Bit internal metrics in Prometheus format as well as newrelic-fluent-bit-output internal plugin metrics. See [this documentation page](https://docs.newrelic.com/docs/logs/forward-logs/kubernetes-plugin-log-forwarding/#troubleshoot-installation) for more details. | `false` | +| `dnsConfig` | [DNS configuration](https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/#pod-dns-config) that will be added to the pods. Can be configured also with `global.dnsConfig`. | `{}` | +| `fluentBit.criEnabled` | We assume that `kubelet`directly communicates with the container engine using the [CRI](https://kubernetes.io/docs/concepts/overview/components/#container-runtime) specification. Set this to `false` if your K8s installation uses [dockershim](https://kubernetes.io/docs/tasks/administer-cluster/migrating-from-dockershim/) instead, in order to get the logs properly parsed. | `true` | + +### Fluent Bit persistence modes + +Fluent Bit uses a database file to keep track of log lines read from files (offsets). This database file is stored in the host node by default, using a `hostPath` mount. It's specifically stored (by default) in `/var/log/flb_kube.db` to keep things simple, as we're already mounting `/var` for accessing container logs. + +Sometimes the security constraints of some clusters don't allow mounting `hostPath`s in read-write mode. That's why you can chose among the following +persistence modes. Each one has their pros and cons. + +- `hostPath` (default) will use a `hostPath` mount to store the DB file on the node disk. This is the easiest, cheapest an most reliable option, but prohibited by some cloud vendor security policies. +- `none` will disable the Fluent Bit DB file. This can cause log duplication or data loss in case Fluent Bit gets restarted. +- `persistentVolume` (Linux only) will use a `ReadWriteMany` persistent volume to store the DB file. This will override the `fluentBit.db` path and use `/db/${NODE_NAME}-fb.db` instead. If you use this option in a Windows cluster it will default to `none` on Windows nodes. + +#### GKE Autopilot example + +If you're using the `persistentVolume` persistence mode you need to provide at least the `storageClass`, and it should be `ReadWriteMany`. This is an example of the configuration for persistence in [GKE Autopilot](https://cloud.google.com/kubernetes-engine/docs/concepts/autopilot-overview). + +``` +fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: standard-rwx + linuxMountPath: /var/log +``` + +### Proxy support + +Since Fluent Bit Kubernetes plugin is using [newrelic-fluent-bit-output](https://github.com/newrelic/newrelic-fluent-bit-output) we can configure the [proxy support](https://github.com/newrelic/newrelic-fluent-bit-output#proxy-support) in order to set up the proxy configuration. + +#### As environment variables + +The easiest way to configure the proxy is by means of specifying the `HTTP_PROXY` or `HTTPS_PROXY` variables as follows: + +``` +# values-newrelic.yml + +fluentBit: + additionalEnvVariables: + - name: HTTPS_PROXY + value: https://your-https-proxy-hostname:3129 +``` + + +#### Custom proxy configuration (for proxies using self-signed certificates) + +If you need to use a proxy using self-signed certificates, you'll need to mount a volume with the Certificate Authority +bundle file and reference it from the Fluent Bit configuration as follows: + +``` +# values-newrelic.yaml +extraVolumes: [] + - name: proxyConfig + # Example using hostPath. You can also place the caBundleFile.pem contents in a ConfigMap and reference it here instead, + # as explained here: https://kubernetes.io/docs/concepts/storage/volumes/#configmap + hostPath: + path: /path/in/node/to/your/caBundleFile.pem + +extraVolumeMounts: [] + - name: proxyConfig + mountPath: /proxyConfig/caBundleFile.pem + +fluentBit: + config: + outputs: | + [OUTPUT] + Name newrelic + Match * + licenseKey ${LICENSE_KEY} + endpoint ${ENDPOINT} + lowDataMode ${LOW_DATA_MODE} + Retry_Limit ${RETRY_LIMIT} + proxy https://your-https-proxy-hostname:3129 + caBundleFile /proxyConfig/caBundleFile.pem +``` + + +## Windows support + +Since version `1.7.0`, this Helm chart supports shipping logs from Windows containers. To this end, you need to set the `enableWindows` configuration parameter to `true`. + +Windows containers have some constraints regarding Linux containers. The main one being that they can only be executed on _hosts_ using the exact same Windows version and build number. On the other hand, Kubernetes nodes only supports the Windows versions listed [here](https://kubernetes.io/docs/setup/production-environment/windows/intro-windows-in-kubernetes/#windows-os-version-support). + +This Helm chart deploys one `DaemonSet` for each of the Windows versions it supports, while ensuring that only containers matching the host operating system will be deployed in each host. + +This Helm chart currently supports the following Windows versions: +- Windows Server LTSC 2019, build 10.0.17763 +- Windows Server LTSC 2022, build 10.0.20348 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..b65ac15d4c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.2.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..3ccc108e2d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/DEVELOPERS.md @@ -0,0 +1,663 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..647b4ff439 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-enable-windows-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-enable-windows-values.yaml new file mode 100644 index 0000000000..870bc082ac --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-enable-windows-values.yaml @@ -0,0 +1,2 @@ +enableLinux: false +enableWindows: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-lowdatamode-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-lowdatamode-values.yaml new file mode 100644 index 0000000000..7740338b05 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-lowdatamode-values.yaml @@ -0,0 +1 @@ +lowDataMode: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-override-global-lowdatamode.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-override-global-lowdatamode.yaml new file mode 100644 index 0000000000..22dd7e05e5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-override-global-lowdatamode.yaml @@ -0,0 +1,3 @@ +global: + lowDataMode: true +lowDataMode: false diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-staging-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-staging-values.yaml new file mode 100644 index 0000000000..efbdccaf81 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-staging-values.yaml @@ -0,0 +1 @@ +nrStaging: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-global.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-global.yaml new file mode 100644 index 0000000000..490a0b7ed3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-global.yaml @@ -0,0 +1 @@ +global: {} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/ci/test-with-empty-values.yaml new file mode 100644 index 0000000000..e69de29bb2 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/fluent-bit-and-plugin-metrics-dashboard-template.json b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/fluent-bit-and-plugin-metrics-dashboard-template.json new file mode 100644 index 0000000000..cafdaf85ca --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/fluent-bit-and-plugin-metrics-dashboard-template.json @@ -0,0 +1,2237 @@ +{ + "name": "Kubernetes Fluent Bit monitoring", + "description": null, + "permissions": "PUBLIC_READ_WRITE", + "pages": [ + { + "name": "Fluent Bit metrics: General", + "description": null, + "widgets": [ + { + "title": "", + "layout": { + "column": 1, + "row": 1, + "width": 6, + "height": 6 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# README\n\n## About this page\nThis page represents most of [Fluent Bit's internal metrics](https://docs.fluentbit.io/manual/administration/monitoring#for-v2-metrics). The metric representations are grouped by categories and faceted by each plugin instance where appropriate.\n\n## How to filter\n1. Select the Kubernetes cluster you want to troubleshoot in the \"Cluster Name\" variable above.\n2. [OPTIONAL] You can use any of the values in the `Node name` and `Pod name` columns on the \"Fluent Bit version\" table to further filter the metrics displayed in the graphs below. To do so, you need to enable [facet filtering](https://docs.newrelic.com/docs/query-your-data/explore-query-data/dashboards/filter-new-relic-one-dashboards-facets/) on that table by clicking on the \"Edit\" submenu and select \"Filter the current dashboard\" under \"Facet Linking\". \n\n## Legend\n### Metric dimensions\n- **name**: the name of the Fluent Bit plugin. Version 1.21.0 of our Helm chart names them according to the plugin names described in the following section.\n- **pod_name**: the `newrelic-logging` pod (Fluent Bit instance) that emitted this metric.\n- **node_name**: physical Kubernetes node where the `newrelic-logging` pod is running.\n\n### Plugin names\n- **pod-logs-tailer**: `tail` *INPUT* plugin normally reading from `/var/log/containers/*.log`\n- **kubernetes-enricher**: `kubernetes` *FILTER* plugin that queries the Kubernetes API to enrich the logs with pod/container metadata.\n- **node-attributes-enricher**: `record_modifier` *FILTER* plugin that enriches logs with `cluster_name`.\n- **kubernetes-attribute-lifter** (only when in low data mode): `nest` *FILTER* plugin that lifts all the keys under `kubernetes`. This plugin is transparent to the final shape of the log.\n- **node-attributes-enricher-filter** (only when in low data mode): same as node-attributes-enricher`, but it also removes attributes that are not strictly necessary for correct platform functioning.\n- **newrelic-logs-forwarder**: `newrelic` *OUTPUT* plugin that sends logs to the New Relic Logs API" + } + }, + { + "title": "Fluent Bit version", + "layout": { + "column": 7, + "row": 1, + "width": 6, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.table" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT latest(os) as 'OS', latest(version) as 'FB version', latest(cluster_name) FROM Metric where metricName = 'fluentbit_build_info' AND cluster_name IN ({{cluster_name}}) since 1 hour ago facet pod_name, node_name limit max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + } + } + }, + { + "title": "Fluent Bit uptime", + "layout": { + "column": 7, + "row": 4, + "width": 6, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT latest(fluentbit_uptime) FROM Metric where cluster_name IN ({{cluster_name}}) facet pod_name timeseries" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 7, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# INPUTS" + } + }, + { + "title": "Input byte rate (bytes/minute)", + "layout": { + "column": 1, + "row": 8, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_input_bytes_total), 1 minute) as 'bytes/minute' FROM Metric where name != 'fb-metrics-collector' and cluster_name IN ({{cluster_name}}) timeseries max facet name, pod_name" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Input log rate (records/minute)", + "layout": { + "column": 5, + "row": 8, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_input_records_total), 1 minute) as 'logs/minute' FROM Metric where name != 'fb-metrics-collector' and cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Average incoming record size (bytes)", + "layout": { + "column": 9, + "row": 8, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT sum(fluentbit_input_bytes_total)/sum(fluentbit_input_records_total) as 'Average incoming record size (bytes)' FROM Metric where name != 'fb-metrics-collector' and cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 11, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# FILTERS" + } + }, + { + "title": "Filter byte rate (bytes/minute)", + "layout": { + "column": 1, + "row": 12, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_filter_bytes_total), 1 minute) FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Filter log rate (records/minute)", + "layout": { + "column": 5, + "row": 12, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_filter_records_total), 1 minute) FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Average filtered record size (bytes)", + "layout": { + "column": 9, + "row": 12, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT sum(fluentbit_filter_bytes_total)/sum(fluentbit_filter_records_total) AS 'Average filtered record size (bytes)' FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Record add/drop rate per FILTER plugin", + "layout": { + "column": 1, + "row": 15, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_filter_add_records_total), 1 minute) as 'Added back to pipeline', rate(sum(fluentbit_filter_drop_records_total), 1 minute) as 'Removed from pipeline' FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries MAX" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "REQUESTS_PER_MINUTE" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 18, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# OUTPUTS" + } + }, + { + "title": "Output byte rate (bytes/minute)", + "layout": { + "column": 1, + "row": 19, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_proc_bytes_total), 1 minute) as 'bytes/minute' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Output log rate (records/minute)", + "layout": { + "column": 5, + "row": 19, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_proc_records_total), 1 minute) as 'records/minute' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Average outgoing record size (bytes)", + "layout": { + "column": 9, + "row": 19, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT sum(fluentbit_output_proc_bytes_total)/sum(fluentbit_output_proc_records_total) as 'bytes' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries MAX" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "newrelic plugin statistics (records/minute)", + "layout": { + "column": 1, + "row": 22, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_proc_records_total), 1 minute) as 'Processed', rate(sum(fluentbit_output_dropped_records_total), 1 minute) as 'Dropped', rate(sum(fluentbit_output_retried_records_total), 1 minute) as 'Retried' FROM Metric where cluster_name IN ({{cluster_name}}) AND name = 'newrelic-logs-forwarder' facet pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Other OUTPUT plugin statistics (records/minute)", + "layout": { + "column": 5, + "row": 22, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_proc_records_total), 1 minute) as 'Processed', rate(sum(fluentbit_output_dropped_records_total), 1 minute) as 'Dropped', rate(sum(fluentbit_output_retried_records_total), 1 minute) as 'Retried' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'newrelic-logs-forwarder' and name != 'fb-metrics-forwarder' facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Connections per OUTPUT plugin", + "layout": { + "column": 9, + "row": 22, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_output_upstream_total_connections) as 'Total', max(fluentbit_output_upstream_busy_connections) as 'Busy' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries MAX" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "newrelic plugin errors (errors/minute)", + "layout": { + "column": 1, + "row": 25, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_errors_total), 1 minute) AS 'Errors/minute' FROM Metric where cluster_name IN ({{cluster_name}}) AND name = 'newrelic-logs-forwarder' facet pod_name timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "newrelic plugin chunk retry statistics (retries/minute)", + "layout": { + "column": 5, + "row": 25, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_retries_total), 1 minute) as 'Retries', rate(sum(fluentbit_output_retries_failed_total), 1 minute) as 'Expirations' FROM Metric where cluster_name IN ({{cluster_name}}) AND name = 'newrelic-logs-forwarder' facet pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 28, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# MEMORY USAGE" + } + }, + { + "title": "Input plugin memory usage", + "layout": { + "column": 1, + "row": 29, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_memory_bytes) as 'Max' FROM Metric where cluster_name IN ({{cluster_name}}) and name != 'fb-metrics-collector' timeseries max facet name, pod_name " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "INPUT memory buffer over limit", + "layout": { + "column": 5, + "row": 29, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "colors": { + "seriesOverrides": [ + { + "color": "#013ef4", + "seriesName": "pod-logs-tailer" + } + ] + }, + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_overlimit) FROM Metric where cluster_name IN ({{cluster_name}}) and name != 'fb-metrics-collector' timeseries max facet name, pod_name" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true, + "thresholds": [ + { + "from": 0.95, + "name": "Mem buf overlimit", + "severity": "critical", + "to": 1.05 + } + ] + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Chunk statistics per INPUT plugin", + "layout": { + "column": 9, + "row": 29, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT average(fluentbit_input_storage_chunks_up) AS 'Up (in memory)', average(fluentbit_input_storage_chunks_down) AS 'Down (in fs)', average(fluentbit_input_storage_chunks_busy) AS 'Busy', average(fluentbit_input_storage_chunks) as 'Total' FROM Metric where name != 'fb-metrics-collector' since 1 hour ago timeseries MAX facet name, pod_name " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Buffered chunks", + "layout": { + "column": 1, + "row": 32, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_chunks) AS 'Total', max(fluentbit_storage_mem_chunks) AS 'Memory', max(fluentbit_storage_fs_chunks) AS 'Filesystem' FROM Metric where cluster_name IN ({{cluster_name}}) facet pod_name timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Busy chunks' size", + "layout": { + "column": 5, + "row": 32, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_chunks_busy_bytes) FROM Metric where name != 'fb-metrics-collector' facet name, pod_name timeseries MAX since 1 hour ago" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Filesystem chunks state", + "layout": { + "column": 9, + "row": 32, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT average(fluentbit_storage_fs_chunks_up) AS 'Up (in memory)', average(fluentbit_storage_fs_chunks_down) AS 'Down (fs only)' FROM Metric since '2024-02-29 13:22:00+0000' UNTIL '2024-02-29 14:31:00+0000' timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + } + ] + }, + { + "name": "Fluent Bit metrics: Pipeline View", + "description": null, + "widgets": [ + { + "title": "", + "layout": { + "column": 1, + "row": 1, + "width": 6, + "height": 6 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# README\n\n## About this page\nThis page represents the same metrics that are displayed in the \"Fluent Bit metrics: General\" page. Nevertheless, they are grouped differently to allow you to visualize a given metric across the whole pipeline with a single glance.\n\n## How to filter\n1. Select the Kubernetes cluster you want to troubleshoot in the \"Cluster Name\" variable above.\n2. [OPTIONAL] You can use any of the values in the `Node name` and `Pod name` columns on the \"Fluent Bit version\" table to further filter the metrics displayed in the graphs below. To do so, you need to enable [facet filtering](https://docs.newrelic.com/docs/query-your-data/explore-query-data/dashboards/filter-new-relic-one-dashboards-facets/) on that table by clicking on the \"Edit\" submenu and select \"Filter the current dashboard\" under \"Facet Linking\". \n\n## Legend\n### Metric dimensions\n- **name**: the name of the Fluent Bit plugin. Version 1.21.0 of our Helm chart names them according to the plugin names described in the following section.\n- **pod_name**: the `newrelic-logging` pod (Fluent Bit instance) that emitted this metric.\n- **node_name**: physical Kubernetes node where the `newrelic-logging` pod is running.\n\n### Plugin names\n- **pod-logs-tailer**: `tail` *INPUT* plugin normally reading from `/var/log/containers/*.log`\n- **kubernetes-enricher**: `kubernetes` *FILTER* plugin that queries the Kubernetes API to enrich the logs with pod/container metadata.\n- **node-attributes-enricher**: `record_modifier` *FILTER* plugin that enriches logs with `cluster_name`.\n- **kubernetes-attribute-lifter** (only when in low data mode): `nest` *FILTER* plugin that lifts all the keys under `kubernetes`. This plugin is transparent to the final shape of the log.\n- **node-attributes-enricher-filter** (only when in low data mode): same as node-attributes-enricher`, but it also removes attributes that are not strictly necessary for correct platform functioning.\n- **newrelic-logs-forwarder**: `newrelic` *OUTPUT* plugin that sends logs to the New Relic Logs API" + } + }, + { + "title": "Fluent Bit version", + "layout": { + "column": 7, + "row": 1, + "width": 6, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.table" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT latest(os) as 'OS', latest(version) as 'FB version', latest(cluster_name) FROM Metric where metricName = 'fluentbit_build_info' AND cluster_name IN ({{cluster_name}}) since 1 hour ago facet pod_name, node_name limit max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + } + } + }, + { + "title": "Fluent Bit uptime", + "layout": { + "column": 7, + "row": 4, + "width": 6, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT latest(fluentbit_uptime) FROM Metric where cluster_name IN ({{cluster_name}}) timeseries facet pod_name " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "SECONDS" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 7, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# BYTE RATES" + } + }, + { + "title": "Input byte rate (bytes/minute)", + "layout": { + "column": 1, + "row": 8, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_input_bytes_total), 1 minute) as 'bytes/minute' FROM Metric where name != 'fb-metrics-collector' and cluster_name IN ({{cluster_name}}) timeseries max facet name, pod_name" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Filter byte rate (bytes/minute)", + "layout": { + "column": 5, + "row": 8, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_filter_bytes_total), 1 minute) FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Output byte rate (bytes/minute)", + "layout": { + "column": 9, + "row": 8, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_proc_bytes_total), 1 minute) as 'bytes/minute' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 11, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# LOG RECORD RATES" + } + }, + { + "title": "Input log rate (records/minute)", + "layout": { + "column": 1, + "row": 12, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_input_records_total), 1 minute) as 'logs/minute' FROM Metric where name != 'fb-metrics-collector' and cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Filter log rate (records/minute)", + "layout": { + "column": 5, + "row": 12, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_filter_records_total), 1 minute) FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Output log rate (records/minute)", + "layout": { + "column": 9, + "row": 12, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_proc_records_total), 1 minute) as 'records/minute' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Record add/drop rate per FILTER plugin", + "layout": { + "column": 5, + "row": 15, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_filter_add_records_total), 1 minute) as 'Added back to pipeline', rate(sum(fluentbit_filter_drop_records_total), 1 minute) as 'Removed from pipeline' FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries MAX" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "REQUESTS_PER_MINUTE" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 18, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# AVERAGE LOG RECORD SIZES AT THE END OF EACH STAGE" + } + }, + { + "title": "Average incoming record size (bytes)", + "layout": { + "column": 1, + "row": 19, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT sum(fluentbit_input_bytes_total)/sum(fluentbit_input_records_total) as 'Average incoming record size (bytes)' FROM Metric where name != 'fb-metrics-collector' and cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Average filtered record size (bytes)", + "layout": { + "column": 5, + "row": 19, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT sum(fluentbit_filter_bytes_total)/sum(fluentbit_filter_records_total) AS 'Average filtered record size (bytes)' FROM Metric WHERE cluster_name IN ({{cluster_name}}) facet name, pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Average outgoing record size (bytes)", + "layout": { + "column": 9, + "row": 19, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT sum(fluentbit_output_proc_bytes_total)/sum(fluentbit_output_proc_records_total) as 'bytes' FROM Metric where cluster_name IN ({{cluster_name}}) AND name != 'fb-metrics-forwarder' facet name, pod_name timeseries MAX" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "", + "layout": { + "column": 1, + "row": 22, + "width": 12, + "height": 1 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# MEMORY USAGE AND BACKPRESSURE" + } + }, + { + "title": "Input plugin memory usage", + "layout": { + "column": 1, + "row": 23, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_memory_bytes) as 'Max' FROM Metric where cluster_name IN ({{cluster_name}}) and name != 'fb-metrics-collector' timeseries max facet name, pod_name " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Busy chunks' size", + "layout": { + "column": 5, + "row": 23, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_chunks_busy_bytes) FROM Metric where name != 'fb-metrics-collector' facet name, pod_name timeseries MAX since 1 hour ago" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "newrelic plugin chunk retry statistics (retries/minute)", + "layout": { + "column": 9, + "row": 23, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_retries_total), 1 minute) as 'Retries', rate(sum(fluentbit_output_retries_failed_total), 1 minute) as 'Expirations' FROM Metric where cluster_name IN ({{cluster_name}}) AND name = 'newrelic-logs-forwarder' facet pod_name timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "INPUT memory buffer over limit", + "layout": { + "column": 1, + "row": 26, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "colors": { + "seriesOverrides": [ + { + "color": "#013ef4", + "seriesName": "pod-logs-tailer" + } + ] + }, + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT max(fluentbit_input_storage_overlimit) FROM Metric where cluster_name IN ({{cluster_name}}) and name != 'fb-metrics-collector' timeseries max facet name, pod_name" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true, + "thresholds": [ + { + "from": 0.95, + "name": "Mem buf overlimit", + "severity": "critical", + "to": 1.05 + } + ] + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Chunk statistics per INPUT plugin", + "layout": { + "column": 5, + "row": 26, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT average(fluentbit_input_storage_chunks_up) AS 'Up (in memory)', average(fluentbit_input_storage_chunks_down) AS 'Down (in fs)', average(fluentbit_input_storage_chunks_busy) AS 'Busy', average(fluentbit_input_storage_chunks) as 'Total' FROM Metric where name != 'fb-metrics-collector' since 1 hour ago timeseries MAX facet name, pod_name " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "newrelic plugin errors (errors/minute)", + "layout": { + "column": 9, + "row": 26, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(fluentbit_output_errors_total), 1 minute) AS 'Errors/minute' FROM Metric where cluster_name IN ({{cluster_name}}) AND name = 'newrelic-logs-forwarder' facet pod_name timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": { + "isLabelVisible": true + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + } + ] + }, + { + "name": "newrelic-fluent-bit-output plugin metrics", + "description": null, + "widgets": [ + { + "title": "", + "layout": { + "column": 1, + "row": 1, + "width": 4, + "height": 9 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.markdown" + }, + "rawConfiguration": { + "text": "# README\n## About this page\nThis page displays metrics collected internally in the [New Relic Fluent Bit output plugin](https://github.com/newrelic/newrelic-fluent-bit-output) (in short, **NR FB plugin**). These metrics are independent of Fluent Bit's, and **must not be considered as a stable API: they can change its naming or dimensions at any time in newer plugin versions**.\n\nPlease note that **the NR FB plugin does not include the `pod_name` nor the `node_name` dimensions**. Therefore, the graphs below represent an aggregation of all your running Fluent Bit instances across one or more clusters. You can use the `cluster_name` dimension (or dashboard variable above) to narrow down the troubleshooting to one or more clusters.\n\n## Basic naming conventions\n- Fluent Bit aggregates logs in batches, also referred as **[chunks](https://docs.fluentbit.io/manual/administration/buffering-and-storage#chunks-memory-filesystem-and-backpressure)**. Each chunk therefore contains an unknown amount of logs.\n- Chunks are received sequentially at the NR FB plugin, which takes care of reading the logs they contain and splitting them into the so-called New Relic *payloads*.\n- Each **payload** is a compressed stream of bytes that can be [at most 1MB long](https://docs.newrelic.com/docs/logs/log-api/introduction-log-api/#limits), and follows the [data format required by the Logs API](https://docs.newrelic.com/docs/logs/log-api/introduction-log-api/#json-content).\n\n\n## Error-detection graphs and recommended actions\n\nThe following are the main graphs used to detect potential problems in your log forwarding setup. Refer to each section to learn the recommended actions for each graph.\n\n### Payload packaging errors\nRepresents the percentage of Fluent Bit chunks that threw an error when they were attempted to be packaged as New Relic payloads. Such errors are never expected to happen. Therefore, **any value greater than 0% should be thoroughly investigated**.\n\nIf you find errors in this graph, please open a support ticket and include a sample of your logs for further investigation.\n\n### Payload sending errors\nRepresents the percentage of New Relic payloads that threw an unexpected error when they were attempted to be sent to New Relic. Such errors can happen sporadically: timeouts due to poor network performance or sudden network changes can cause them from time to time. Observing **values greater than 0% can sometimes be normal, but any value above 10% should be considered as an annomalous situation and should be thoroughly investigated**.\n\nIf you find errors in this graph, please ensure that you don't have any weak spots in your network path to New Relic: are you using a proxy? Is it or any network hop introducing too much latency due to being saturated? If you can't find anything on you side, please open a support ticket and include as much information as possible from your network setup.\n\n### Payload send results\nRepresents the amount of API requests that were performed to send logs to New Relic. **Ideally, you should only observe 202 responses here**. Sometimes, intermediary CDN providers can introduce some errors (503 error codes) from time to time, in which case your logs will not be lost and reattempted to be sent.\n\nIf you find a considerable amount of non-202 responses in this graph, please open a customer support ticket.\n\n## Additional troubleshooting graphs\n\nThe following graphs include additional fine-grained information that will be useful for New Relic to troubleshoot your potential installation issues.\n\n### Average timings\nRepresents the average amount of time the plugin spent packaging the log payloads and sending them to New Relic, respectively.\n\n### Accumulated time per minute\nRepresents the amount of time per minute the plugin spent packaging the log payloads and sending them to New Relic, respectively.\n\n### Payload size\nRepresents the size in bytes of the individual compressed payloads sent to New Relic.\n\n### Payload packets per Fluent Bit chunk\nRepresents the amount of payloads sent to New Relic per each Fluent Bit chunk." + } + }, + { + "title": "Payload packaging errors", + "layout": { + "column": 5, + "row": 1, + "width": 2, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.billboard" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "FROM Metric SELECT percentage(count(`logs.fb.packaging.time`), WHERE hasError = true) AS 'packaging errors'" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": [ + { + "alertSeverity": "CRITICAL", + "value": 0 + } + ] + } + }, + { + "title": "Payload sending errors", + "layout": { + "column": 7, + "row": 1, + "width": 2, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.billboard" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "FROM Metric SELECT percentage(count(`logs.fb.payload.send.time`), WHERE hasError = true) AS 'send errors'" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "thresholds": [ + { + "alertSeverity": "WARNING", + "value": 0 + }, + { + "alertSeverity": "CRITICAL", + "value": 0.1 + } + ] + } + }, + { + "title": "Payload send results", + "layout": { + "column": 9, + "row": 1, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(count(logs.fb.payload.send.time), 1 minute) AS 'Status Code' FROM Metric FACET CASES(WHERE statusCode = 0 AS 'Send error') OR statusCode timeseries max" + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "units": { + "unit": "REQUESTS_PER_MINUTE" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Average timings", + "layout": { + "column": 5, + "row": 4, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT average(logs.fb.payload.send.time) AS 'Payload sending', average(logs.fb.packaging.time) AS 'Payload packaging' FROM Metric timeseries max" + } + ], + "nullValues": { + "nullValue": "zero" + }, + "platformOptions": { + "ignoreTimeRange": false + }, + "units": { + "unit": "MS" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Accumulated time per minute", + "layout": { + "column": 9, + "row": 4, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.area" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT rate(sum(logs.fb.total.send.time), 1 minute) AS 'Sending', rate(sum(logs.fb.packaging.time), 1 minute) AS 'Packaging' FROM Metric TIMESERIES max" + } + ], + "nullValues": { + "nullValue": "zero" + }, + "platformOptions": { + "ignoreTimeRange": false + }, + "units": { + "unit": "MS" + } + } + }, + { + "title": "Payload size", + "layout": { + "column": 5, + "row": 7, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT min(logs.fb.payload.size) AS 'Minimum', average(logs.fb.payload.size) AS 'Average', max(logs.fb.payload.size) AS 'Maximum' FROM Metric timeseries MAX " + } + ], + "nullValues": { + "nullValue": "default" + }, + "platformOptions": { + "ignoreTimeRange": false + }, + "units": { + "unit": "BYTES" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + }, + { + "title": "Payload packets per Fluent Bit chunk", + "layout": { + "column": 9, + "row": 7, + "width": 4, + "height": 3 + }, + "linkedEntityGuids": null, + "visualization": { + "id": "viz.line" + }, + "rawConfiguration": { + "facet": { + "showOtherSeries": false + }, + "legend": { + "enabled": true + }, + "nrqlQueries": [ + { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT min(logs.fb.payload.count) AS 'Minimum', average(logs.fb.payload.count) AS 'Average', max(logs.fb.payload.count) AS 'Maximum' FROM Metric timeseries MAX " + } + ], + "platformOptions": { + "ignoreTimeRange": false + }, + "units": { + "unit": "COUNT" + }, + "yAxisLeft": { + "zero": true + }, + "yAxisRight": { + "zero": true + } + } + } + ] + } + ], + "variables": [ + { + "name": "cluster_name", + "items": null, + "defaultValues": [ + { + "value": { + "string": "*" + } + } + ], + "nrqlQuery": { + "accountIds": [ + YOUR_ACCOUNT_ID + ], + "query": "SELECT uniques(cluster_name) FROM Metric where metricName = 'fluentbit_input_storage_overlimit'" + }, + "options": { + "ignoreTimeRange": false + }, + "title": "Cluster Name", + "type": "NRQL", + "isMultiSelection": true, + "replacementStrategy": "STRING" + } + ] +} \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/NOTES.txt new file mode 100644 index 0000000000..289f2157f3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/NOTES.txt @@ -0,0 +1,18 @@ +{{- if (include "newrelic-logging.areValuesValid" .) }} +Your deployment of the New Relic Kubernetes Logging is complete. You can check on the progress of this by running the following command: + + kubectl get daemonset -o wide -w --namespace {{ .Release.Namespace }} {{ template "newrelic-logging.fullname" . }} +{{- else -}} +############################################################################## +#### ERROR: You did not set a license key. #### +############################################################################## + +This deployment will be incomplete until you get your API key from New Relic. + +Then run: + + helm upgrade {{ .Release.Name }} \ + --set licenseKey=(your-license-key) \ + newrelic/newrelic-logging + +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/_helpers.tpl new file mode 100644 index 0000000000..439d25cae4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/_helpers.tpl @@ -0,0 +1,215 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "newrelic-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). +*/}} +{{- define "newrelic-logging.fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if ne $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s" $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + + +{{/* Generate basic labels */}} +{{- define "newrelic-logging.labels" }} +app: {{ template "newrelic-logging.name" . }} +chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} +heritage: {{.Release.Service }} +release: {{.Release.Name }} +app.kubernetes.io/name: {{ template "newrelic-logging.name" . }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "newrelic-logging.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + + +{{/* +Create the name of the fluent bit config +*/}} +{{- define "newrelic-logging.fluentBitConfig" -}} +{{ template "newrelic-logging.fullname" . }}-fluent-bit-config +{{- end -}} + +{{/* +Return the licenseKey +*/}} +{{- define "newrelic-logging.licenseKey" -}} +{{- if .Values.global}} + {{- if .Values.global.licenseKey }} + {{- .Values.global.licenseKey -}} + {{- else -}} + {{- .Values.licenseKey | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.licenseKey | default "" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the cluster name +*/}} +{{- define "newrelic-logging.cluster" -}} +{{- if .Values.global}} + {{- if .Values.global.cluster }} + {{- .Values.global.cluster -}} + {{- else -}} + {{- .Values.cluster | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.cluster | default "" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the customSecretName +*/}} +{{- define "newrelic-logging.customSecretName" -}} +{{- if .Values.global }} + {{- if .Values.global.customSecretName }} + {{- .Values.global.customSecretName -}} + {{- else -}} + {{- .Values.customSecretName | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.customSecretName | default "" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the customSecretLicenseKey +*/}} +{{- define "newrelic-logging.customSecretKey" -}} +{{- if .Values.global }} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- else -}} + {{- if .Values.global.customSecretKey }} + {{- .Values.global.customSecretKey -}} + {{- else -}} + {{- .Values.customSecretKey | default "" -}} + {{- end -}} + {{- end -}} +{{- else -}} + {{- if .Values.customSecretLicenseKey }} + {{- .Values.customSecretLicenseKey -}} + {{- else -}} + {{- .Values.customSecretKey | default "" -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Returns nrStaging +*/}} +{{- define "newrelic.nrStaging" -}} +{{- if .Values.global }} + {{- if .Values.global.nrStaging }} + {{- .Values.global.nrStaging -}} + {{- end -}} +{{- else if .Values.nrStaging }} + {{- .Values.nrStaging -}} +{{- end -}} +{{- end -}} + +{{/* +Returns fargate +*/}} +{{- define "newrelic.fargate" -}} +{{- if .Values.global }} + {{- if .Values.global.fargate }} + {{- .Values.global.fargate -}} + {{- end -}} +{{- else if .Values.fargate }} + {{- .Values.fargate -}} +{{- end -}} +{{- end -}} + +{{/* +Returns lowDataMode +*/}} +{{- define "newrelic-logging.lowDataMode" -}} +{{/* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{/* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic-logging.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{/* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Returns logsEndpoint +*/}} +{{- define "newrelic-logging.logsEndpoint" -}} +{{- if (include "newrelic.nrStaging" .) -}} +https://staging-log-api.newrelic.com/log/v1 +{{- else if .Values.endpoint -}} +{{ .Values.endpoint -}} +{{- else if eq (substr 0 2 (include "newrelic-logging.licenseKey" .)) "eu" -}} +https://log-api.eu.newrelic.com/log/v1 +{{- else -}} +https://log-api.newrelic.com/log/v1 +{{- end -}} +{{- end -}} + +{{/* +Returns metricsHost +*/}} +{{- define "newrelic-logging.metricsHost" -}} +{{- if (include "newrelic.nrStaging" .) -}} +staging-metric-api.newrelic.com +{{- else if eq (substr 0 2 (include "newrelic-logging.licenseKey" .)) "eu" -}} +metric-api.eu.newrelic.com +{{- else -}} +metric-api.newrelic.com +{{- end -}} +{{- end -}} + +{{/* +Returns if the template should render, it checks if the required values are set. +*/}} +{{- define "newrelic-logging.areValuesValid" -}} +{{- $licenseKey := include "newrelic-logging.licenseKey" . -}} +{{- $customSecretName := include "newrelic-logging.customSecretName" . -}} +{{- $customSecretKey := include "newrelic-logging.customSecretKey" . -}} +{{- and (or $licenseKey (and $customSecretName $customSecretKey))}} +{{- end -}} + +{{/* +If additionalEnvVariables is set, renames to extraEnv. Returns extraEnv. +*/}} +{{- define "newrelic-logging.extraEnv" -}} +{{- if .Values.fluentBit }} + {{- if .Values.fluentBit.additionalEnvVariables }} + {{- toYaml .Values.fluentBit.additionalEnvVariables -}} + {{- else if .Values.fluentBit.extraEnv }} + {{- toYaml .Values.fluentBit.extraEnv -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrole.yaml new file mode 100644 index 0000000000..b36340fe63 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrole.yaml @@ -0,0 +1,23 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fullname" . }} +rules: + - apiGroups: [""] + resources: + - namespaces + - pods + verbs: ["get", "list", "watch"] +{{- if .Values.rbac.pspEnabled }} + - apiGroups: + - extensions + resources: + - podsecuritypolicies + resourceNames: + - privileged-{{ template "newrelic-logging.fullname" . }} + verbs: + - use +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..6b258f6973 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/clusterrolebinding.yaml @@ -0,0 +1,15 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ template "newrelic-logging.fullname" . }} +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/configmap.yaml new file mode 100644 index 0000000000..4b1d890144 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/configmap.yaml @@ -0,0 +1,38 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ .Release.Namespace }} + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fluentBitConfig" . }} +data: + fluent-bit.conf: | + {{- if .Values.fluentBit.config.service }} + {{- .Values.fluentBit.config.service | nindent 4 }} + {{- end }} + {{- if .Values.fluentBit.config.inputs }} + {{- .Values.fluentBit.config.inputs | nindent 4 }} + {{- end }} + {{- if .Values.fluentBit.config.extraInputs }} + {{- .Values.fluentBit.config.extraInputs | nindent 4}} + {{- end }} + {{- if and (include "newrelic-logging.lowDataMode" .) (.Values.fluentBit.config.lowDataModeFilters) }} + {{- .Values.fluentBit.config.lowDataModeFilters | nindent 4 }} + {{- else }} + {{- .Values.fluentBit.config.filters | nindent 4 }} + {{- end }} + {{- if .Values.fluentBit.config.extraFilters }} + {{- .Values.fluentBit.config.extraFilters | nindent 4}} + {{- end }} + {{- if .Values.fluentBit.config.outputs }} + {{- .Values.fluentBit.config.outputs | nindent 4 }} + {{- end }} + {{- if .Values.fluentBit.config.extraOutputs }} + {{- .Values.fluentBit.config.extraOutputs | nindent 4}} + {{- end }} + {{- if and (.Values.fluentBit.sendMetrics) (.Values.fluentBit.config.metricInstrumentation) }} + {{- .Values.fluentBit.config.metricInstrumentation | nindent 4}} + {{- end }} + parsers.conf: | + {{- if .Values.fluentBit.config.parsers }} + {{- .Values.fluentBit.config.parsers | nindent 4}} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset-windows.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset-windows.yaml new file mode 100644 index 0000000000..6a3145d134 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset-windows.yaml @@ -0,0 +1,174 @@ +{{- if and (include "newrelic-logging.areValuesValid" $) $.Values.enableWindows }} +{{- range .Values.windowsOsList }} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + namespace: {{ $.Release.Namespace }} + labels: + kubernetes.io/os: windows +{{ include "newrelic-logging.labels" $ | indent 4 }} + name: {{ template "newrelic-logging.fullname" $ }}-windows-{{ .version }} + annotations: + {{- if $.Values.daemonSet.annotations }} +{{ toYaml $.Values.daemonSet.annotations | indent 4 }} + {{- end }} +spec: + updateStrategy: + type: {{ $.Values.updateStrategy }} + selector: + matchLabels: + app: {{ template "newrelic-logging.name" $ }} + release: {{ $.Release.Name }} + kubernetes.io/os: windows + template: + metadata: + annotations: + checksum/fluent-bit-config: {{ include (print $.Template.BasePath "/configmap.yaml") $ | sha256sum }} + {{- if $.Values.podAnnotations }} +{{ toYaml $.Values.podAnnotations | indent 8}} + {{- end }} + labels: + app: {{ template "newrelic-logging.name" $ }} + release: {{ $.Release.Name }} + kubernetes.io/os: windows + app.kubernetes.io/name: {{ template "newrelic-logging.name" $ }} + {{- if $.Values.podLabels}} +{{ toYaml $.Values.podLabels | indent 8 }} + {{- end }} + spec: + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" $ }} + {{- with include "newrelic.common.dnsConfig" $ }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + dnsPolicy: ClusterFirst + terminationGracePeriodSeconds: 10 + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list $.Values.image.pullSecrets) "context" $) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + {{- if $.Values.hostNetwork }} + hostNetwork: {{ $.Values.hostNetwork }} + {{- end }} + {{- if $.Values.windows.initContainers }} + initContainers: +{{ toYaml $.Values.windows.initContainers | indent 8 }} + {{- end }} + containers: + - name: {{ template "newrelic-logging.name" $ }} + # We have to use 'replace' to remove the double-quotes that "newrelic.common.images.image" has, so that + # we can append the Windows image tag suffix (and then re-quote that value) + image: "{{ include "newrelic.common.images.image" ( dict "imageRoot" $.Values.image "context" $) | replace "\"" ""}}-{{ .imageTagSuffix }}" + imagePullPolicy: "{{ $.Values.image.pullPolicy }}" + securityContext: {} + env: + - name: ENDPOINT + value: {{ include "newrelic-logging.logsEndpoint" $ | quote }} + - name: SOURCE + value: {{ if (include "newrelic-logging.lowDataMode" $) }} "k8s" {{- else }} "kubernetes" {{- end }} + - name: LICENSE_KEY + valueFrom: + secretKeyRef: + {{- if (include "newrelic-logging.licenseKey" $) }} + name: {{ template "newrelic-logging.fullname" $ }}-config + key: license + {{- else }} + name: {{ include "newrelic-logging.customSecretName" $ }} + key: {{ include "newrelic-logging.customSecretKey" $ }} + {{- end }} + - name: CLUSTER_NAME + value: {{ include "newrelic-logging.cluster" $ }} + - name: LOG_LEVEL + value: {{ $.Values.fluentBit.logLevel | quote }} + - name: LOG_PARSER + {{- if $.Values.fluentBit.criEnabled }} + value: "cri,docker" + {{- else }} + value: "docker,cri" + {{- end }} + {{- if or (not $.Values.fluentBit.persistence) (eq $.Values.fluentBit.persistence.mode "hostPath") }} + - name: FB_DB + value: {{ $.Values.fluentBit.windowsDb | quote }} + {{- else }} + - name: FB_DB + value: "" + {{- end }} + - name: PATH + value: {{ $.Values.fluentBit.windowsPath | quote }} + - name: K8S_BUFFER_SIZE + value: {{ $.Values.fluentBit.k8sBufferSize | quote }} + - name: K8S_LOGGING_EXCLUDE + value: {{ $.Values.fluentBit.k8sLoggingExclude | default "false" | quote }} + - name: LOW_DATA_MODE + value: {{ include "newrelic-logging.lowDataMode" $ | default "false" | quote }} + - name: RETRY_LIMIT + value: {{ $.Values.fluentBit.retryLimit | quote }} + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: HOSTNAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: SEND_OUTPUT_PLUGIN_METRICS + value: {{ $.Values.fluentBit.sendMetrics | default "false" | quote }} + - name: METRICS_HOST + value: {{ include "newrelic-logging.metricsHost" $ | quote }} + {{- include "newrelic-logging.extraEnv" $ | nindent 12 }} + command: + - C:\fluent-bit\bin\fluent-bit.exe + - -c + - c:\fluent-bit\etc\fluent-bit.conf + - -e + - C:\fluent-bit\bin\out_newrelic.dll + {{- if $.Values.exposedPorts }} + ports: {{ toYaml $.Values.exposedPorts | nindent 12 }} + {{- end }} + volumeMounts: + - mountPath: C:\fluent-bit\etc + name: fluent-bit-config + - mountPath: C:\var\log + name: logs + {{- if and ($.Values.fluentBit.persistence) (ne $.Values.fluentBit.persistence.mode "hostPath") }} + readOnly: true + {{- end }} + # We need to also mount this because the logs in C:\var\logs are actually symlinks to C:\ProgramData. + # So, in order to be able to read these logs, the reading process needs to also have access to C:\ProgramData. + - mountPath: C:\ProgramData + name: progdata + {{- if and ($.Values.fluentBit.persistence) (ne $.Values.fluentBit.persistence.mode "hostPath") }} + readOnly: true + {{- end }} + {{- if $.Values.resources }} + resources: +{{ toYaml $.Values.resources | indent 12 }} + {{- end }} + volumes: + - name: fluent-bit-config + configMap: + name: {{ template "newrelic-logging.fluentBitConfig" $ }} + - name: logs + hostPath: + path: C:\var\log + - name: progdata + hostPath: + path: C:\ProgramData + {{- if $.Values.priorityClassName }} + priorityClassName: {{ $.Values.priorityClassName }} + {{- end }} + nodeSelector: + {{- if $.Values.windowsNodeSelector }} +{{ toYaml $.Values.windowsNodeSelector | indent 8 }} + {{- else }} + kubernetes.io/os: windows + # Windows containers can only be executed on hosts running the exact same Windows version and build number + node.kubernetes.io/windows-build: {{ .buildNumber }} + {{- end }} + {{- if $.Values.tolerations }} + tolerations: +{{ toYaml $.Values.tolerations | indent 8 }} + {{- end }} +--- +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset.yaml new file mode 100644 index 0000000000..7dac9e07e3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/daemonset.yaml @@ -0,0 +1,211 @@ +{{- if and (include "newrelic-logging.areValuesValid" .) .Values.enableLinux }} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + namespace: {{ .Release.Namespace }} + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fullname" . }} + annotations: + {{- if .Values.daemonSet.annotations }} +{{ toYaml .Values.daemonSet.annotations | indent 4 }} + {{- end }} +spec: + updateStrategy: + type: {{ .Values.updateStrategy }} + selector: + matchLabels: + app: {{ template "newrelic-logging.name" . }} + release: {{.Release.Name }} + template: + metadata: + annotations: + checksum/fluent-bit-config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + {{- if .Values.podAnnotations }} +{{ toYaml .Values.podAnnotations | indent 8}} + {{- end }} + labels: + app: {{ template "newrelic-logging.name" . }} + release: {{.Release.Name }} + kubernetes.io/os: linux + app.kubernetes.io/name: {{ template "newrelic-logging.name" . }} + {{- if .Values.podLabels}} +{{ toYaml .Values.podLabels | indent 8 }} + {{- end }} + spec: + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + dnsPolicy: ClusterFirstWithHostNet + terminationGracePeriodSeconds: 10 + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.image.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + {{- if .Values.hostNetwork }} + hostNetwork: {{ .Values.hostNetwork }} + {{- end }} + initContainers: + {{- if and (.Values.fluentBit.persistence) (eq .Values.fluentBit.persistence.mode "persistentVolume") }} + - name: init + image: busybox:1.36 + command: ["/bin/sh", "-c"] + args: ["/bin/find /db -type f -mtime +1 -delete"] # Delete all db files not updated in the last 24h + volumeMounts: + - name: fb-db-pvc + mountPath: /db + {{- end }} + {{- if .Values.initContainers }} +{{ toYaml .Values.initContainers | indent 8 }} + {{- end }} + containers: + - name: {{ template "newrelic-logging.name" . }} + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 12 }} + {{- end }} + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.image "context" .) }} + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + env: + - name: ENDPOINT + value: {{ include "newrelic-logging.logsEndpoint" . | quote }} + - name: SOURCE + value: {{ if (include "newrelic-logging.lowDataMode" .) }} "k8s" {{- else }} "kubernetes" {{- end }} + - name: LICENSE_KEY + valueFrom: + secretKeyRef: + {{- if (include "newrelic-logging.licenseKey" .) }} + name: {{ template "newrelic-logging.fullname" . }}-config + key: license + {{- else }} + name: {{ include "newrelic-logging.customSecretName" . }} + key: {{ include "newrelic-logging.customSecretKey" . }} + {{- end }} + - name: CLUSTER_NAME + value: {{ include "newrelic-logging.cluster" . }} + - name: LOG_LEVEL + value: {{ .Values.fluentBit.logLevel | quote }} + - name: LOG_PARSER + {{- if .Values.fluentBit.criEnabled }} + value: "cri,docker" + {{- else }} + value: "docker,cri" + {{- end }} + {{- if or (not .Values.fluentBit.persistence) (eq .Values.fluentBit.persistence.mode "hostPath") }} + - name: FB_DB + value: {{ .Values.fluentBit.db | quote }} + {{- else if eq .Values.fluentBit.persistence.mode "persistentVolume" }} + - name: FB_DB + value: "/db/$(NODE_NAME)-fb.db" + {{- else }} + - name: FB_DB + value: "" + {{- end }} + - name: PATH + value: {{ .Values.fluentBit.path | quote }} + - name: K8S_BUFFER_SIZE + value: {{ .Values.fluentBit.k8sBufferSize | quote }} + - name: K8S_LOGGING_EXCLUDE + value: {{ .Values.fluentBit.k8sLoggingExclude | default "false" | quote }} + - name: LOW_DATA_MODE + value: {{ include "newrelic-logging.lowDataMode" . | default "false" | quote }} + - name: RETRY_LIMIT + value: {{ .Values.fluentBit.retryLimit | quote }} + - name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: HOSTNAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: SEND_OUTPUT_PLUGIN_METRICS + value: {{ $.Values.fluentBit.sendMetrics | default "false" | quote }} + - name: METRICS_HOST + value: {{ include "newrelic-logging.metricsHost" . | quote }} + {{- include "newrelic-logging.extraEnv" . | nindent 12 }} + command: + - /fluent-bit/bin/fluent-bit + - -c + - /fluent-bit/etc/fluent-bit.conf + - -e + - /fluent-bit/bin/out_newrelic.so + volumeMounts: + - name: fluent-bit-config + mountPath: /fluent-bit/etc + - name: logs + # We mount /var by default because container logs could be on /var/log or /var/lib/docker/containers (symlinked to /var/log) + mountPath: {{ .Values.fluentBit.linuxMountPath | default "/var" }} + {{- if and (.Values.fluentBit.persistence) (ne .Values.fluentBit.persistence.mode "hostPath") }} + readOnly: true + {{- end }} + {{- if and (.Values.fluentBit.persistence) (eq .Values.fluentBit.persistence.mode "persistentVolume") }} + - name: fb-db-pvc + mountPath: /db + {{- end }} + {{- if .Values.extraVolumeMounts }} + {{- toYaml .Values.extraVolumeMounts | nindent 12 }} + {{- end }} + {{- if .Values.exposedPorts }} + ports: {{ toYaml .Values.exposedPorts | nindent 12 }} + {{- end }} + {{- if .Values.resources }} + resources: +{{ toYaml .Values.resources | indent 12 }} + {{- end }} + volumes: + - name: fluent-bit-config + configMap: + name: {{ template "newrelic-logging.fluentBitConfig" . }} + - name: logs + hostPath: + path: {{ .Values.fluentBit.linuxMountPath | default "/var" }} + {{- if and (.Values.fluentBit.persistence) (eq .Values.fluentBit.persistence.mode "persistentVolume") }} + - name: fb-db-pvc + persistentVolumeClaim: + {{- if .Values.fluentBit.persistence.persistentVolume.existingVolumeClaim }} + claimName: {{ .Values.fluentBit.persistence.persistentVolume.existingVolumeClaim }} + {{- else }} + claimName: {{ template "newrelic-logging.fullname" . }}-pvc + {{- end }} + {{- end }} + {{- if .Values.extraVolumes }} + {{- toYaml .Values.extraVolumes | nindent 8 }} + {{- end }} + {{- if $.Values.priorityClassName }} + priorityClassName: {{ $.Values.priorityClassName }} + {{- end }} + {{- if .Values.nodeAffinity }} + affinity: + nodeAffinity: {{ .Values.nodeAffinity | toYaml | nindent 10 }} + {{- else if include "newrelic.fargate" . }} + affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: eks.amazonaws.com/compute-type + operator: NotIn + values: + - fargate + {{- end }} + nodeSelector: + {{- if .Values.nodeSelector }} +{{ toYaml .Values.nodeSelector | indent 8 }} + {{- else if $.Values.enableWindows }} + # We add this only if Windows is enabled to keep backwards-compatibility. Prior to version 1.14, this label was + # named beta.kubernetes.io/os. In version 1.14, it was deprecated and replaced by this one. Version 1.14 also + # introduces Windows support. Therefore, anyone wishing to use Windows containers must bet at version >=1.14 and + # are going to need this label, in order to avoid placing a linux container on a windows node, and vice-versa. + kubernetes.io/os: linux + {{- end }} + {{- if .Values.tolerations }} + tolerations: +{{ toYaml .Values.tolerations | indent 8 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/persistentvolume.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/persistentvolume.yaml new file mode 100644 index 0000000000..f2fb93d77a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/persistentvolume.yaml @@ -0,0 +1,57 @@ +{{- if (not (empty .Values.fluentBit.persistence)) }} + +{{- if and (eq .Values.fluentBit.persistence.mode "persistentVolume") (not .Values.fluentBit.persistence.persistentVolume.storageClass) (not .Values.fluentBit.persistence.persistentVolume.existingVolumeClaim) }} +{{ fail "You should provide a ReadWriteMany storageClass or an existingVolumeClaim if using persitentVolume as Fluent Bit persistence mode." }} +{{- end }} + +{{- if and (eq .Values.fluentBit.persistence.mode "persistentVolume") (not .Values.fluentBit.persistence.persistentVolume.existingVolumeClaim) }} +{{- if and (not .Values.fluentBit.persistence.persistentVolume.dynamicProvisioning) (not .Values.fluentBit.persistence.persistentVolume.existingVolume) }} +apiVersion: v1 +kind: PersistentVolume +metadata: + namespace: {{ .Release.Namespace }} + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fullname" . }}-pv + annotations: + {{- if .Values.fluentBit.persistence.persistentVolume.annotations.volume }} +{{ toYaml .Values.fluentBit.persistence.persistentVolume.annotations.volume | indent 4 }} + {{- end }} +spec: + accessModes: + - ReadWriteMany + capacity: + storage: {{ .Values.fluentBit.persistence.persistentVolume.size }} + storageClassName: {{ .Values.fluentBit.persistence.persistentVolume.storageClass }} + persistentVolumeReclaimPolicy: Delete + {{- if .Values.fluentBit.persistence.persistentVolume.extra.volume }} +{{ toYaml .Values.fluentBit.persistence.persistentVolume.extra.volume | indent 2 }} + {{- end }} +--- +{{- end }} +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + namespace: {{ .Release.Namespace }} + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fullname" . }}-pvc + annotations: + {{- if .Values.fluentBit.persistence.persistentVolume.annotations.claim }} +{{ toYaml .Values.fluentBit.persistence.persistentVolume.annotations.claim | indent 4 }} + {{- end }} +spec: + storageClassName: {{ .Values.fluentBit.persistence.persistentVolume.storageClass }} + accessModes: + - ReadWriteMany +{{- if .Values.fluentBit.persistence.persistentVolume.existingVolume }} + volumeName: {{ .Values.fluentBit.persistence.persistentVolume.existingVolume }} +{{- else if not .Values.fluentBit.persistence.persistentVolume.dynamicProvisioning }} + volumeName: {{ template "newrelic-logging.fullname" . }}-pv +{{- end }} + resources: + requests: + storage: {{ .Values.fluentBit.persistence.persistentVolume.size }} + {{- if .Values.fluentBit.persistence.persistentVolume.extra.claim }} +{{ toYaml .Values.fluentBit.persistence.persistentVolume.extra.claim | indent 2 }} + {{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/podsecuritypolicy.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/podsecuritypolicy.yaml new file mode 100644 index 0000000000..2c8c598e24 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/podsecuritypolicy.yaml @@ -0,0 +1,24 @@ +{{- if .Values.rbac.pspEnabled }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: privileged-{{ template "newrelic-logging.fullname" . }} +spec: + allowedCapabilities: + - '*' + fsGroup: + rule: RunAsAny + runAsUser: + rule: RunAsAny + seLinux: + rule: RunAsAny + supplementalGroups: + rule: RunAsAny + volumes: + - '*' + hostPID: true + hostIPC: true + hostPorts: + - min: 1 + max: 65536 +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/secret.yaml new file mode 100644 index 0000000000..47a56e573e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/secret.yaml @@ -0,0 +1,12 @@ +{{- $licenseKey := include "newrelic-logging.licenseKey" . -}} +{{- if $licenseKey }} +apiVersion: v1 +kind: Secret +metadata: + namespace: {{ .Release.Namespace }} + labels: {{ include "newrelic-logging.labels" . | indent 4 }} + name: {{ template "newrelic-logging.fullname" . }}-config +type: Opaque +data: + license: {{ $licenseKey | b64enc }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/serviceaccount.yaml new file mode 100644 index 0000000000..51da56a3e7 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/templates/serviceaccount.yaml @@ -0,0 +1,17 @@ +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- if include "newrelic.common.serviceAccount.annotations" . }} + annotations: + {{- include "newrelic.common.serviceAccount.annotations" . | nindent 4 }} + {{- end }} + labels: + app: {{ template "newrelic-logging.name" . }} + chart: {{ template "newrelic-logging.chart" . }} + heritage: "{{ .Release.Service }}" + release: "{{ .Release.Name }}" + {{- /*include "newrelic.common.labels" . | nindent 4 /!\ Breaking change /!\ */}} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/cri_parser_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/cri_parser_test.yaml new file mode 100644 index 0000000000..f4a1d01d07 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/cri_parser_test.yaml @@ -0,0 +1,37 @@ +suite: test cri, docker parser options in daemonsets +templates: + - templates/configmap.yaml + - templates/daemonset.yaml + - templates/daemonset-windows.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: cri enabled by default and docker as fallback + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: LOG_PARSER + value: "cri,docker" + - it: docker is set if enabled by and cri as fallback + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + fluentBit: + criEnabled: false + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: LOG_PARSER + value: "docker,cri" \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/dns_config_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/dns_config_test.yaml new file mode 100644 index 0000000000..76d24eac5f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/dns_config_test.yaml @@ -0,0 +1,62 @@ +suite: test dnsConfig options in daemonsets +templates: + - templates/configmap.yaml + - templates/daemonset.yaml + - templates/daemonset-windows.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: daemonsets contain dnsConfig block when provided + set: + licenseKey: nr_license_key + enableWindows: true + dnsConfig: + nameservers: + - 192.0.2.1 + asserts: + - exists: + path: spec.template.spec.dnsConfig + template: templates/daemonset.yaml + - exists: + path: spec.template.spec.dnsConfig + template: templates/daemonset-windows.yaml + + - it: daemonsets do not contain dnsConfig block when not provided + set: + licenseKey: nr_license_key + enableWindows: true + dnsConfig: {} + asserts: + - notExists: + path: spec.template.spec.dnsConfig + template: templates/daemonset.yaml + - notExists: + path: spec.template.spec.dnsConfig + template: templates/daemonset-windows.yaml + + - it: daemonsets contain provided dnsConfig options + set: + licenseKey: nr_license_key + enableWindows: true + dnsConfig: + options: + - name: ndots + value: "1" + asserts: + - equal: + path: spec.template.spec.dnsConfig.options[0].name + value: ndots + template: templates/daemonset.yaml + - equal: + path: spec.template.spec.dnsConfig.options[0].value + value: "1" + template: templates/daemonset.yaml + - equal: + path: spec.template.spec.dnsConfig.options[0].name + value: ndots + template: templates/daemonset-windows.yaml + - equal: + path: spec.template.spec.dnsConfig.options[0].value + value: "1" + template: templates/daemonset-windows.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/endpoint_region_selection_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/endpoint_region_selection_test.yaml new file mode 100644 index 0000000000..82e700d930 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/endpoint_region_selection_test.yaml @@ -0,0 +1,128 @@ +suite: test endpoint selection based on region settings +templates: + - templates/configmap.yaml + - templates/daemonset.yaml + - templates/daemonset-windows.yaml +release: + name: endpoint-selection-release + namespace: endpoint-selection-namespace +tests: + + - it: selects staging endpoints if nrStaging is enabled + set: + licenseKey: nr_license_key + nrStaging: true + enableWindows: true + asserts: + # Linux + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "https://staging-log-api.newrelic.com/log/v1" + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "staging-metric-api.newrelic.com" + template: templates/daemonset.yaml + # Windows + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "https://staging-log-api.newrelic.com/log/v1" + template: templates/daemonset-windows.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "staging-metric-api.newrelic.com" + template: templates/daemonset-windows.yaml + + - it: selects US endpoints for a US license key + set: + licenseKey: aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaFFFFNRAL + enableWindows: true + asserts: + # Linux + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "https://log-api.newrelic.com/log/v1" + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "metric-api.newrelic.com" + template: templates/daemonset.yaml + # Windows + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "https://log-api.newrelic.com/log/v1" + template: templates/daemonset-windows.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "metric-api.newrelic.com" + template: templates/daemonset-windows.yaml + + - it: selects EU endpoints for a EU license key + set: + licenseKey: euaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaFFFFNRAL + enableWindows: true + asserts: + # Linux + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "https://log-api.eu.newrelic.com/log/v1" + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "metric-api.eu.newrelic.com" + template: templates/daemonset.yaml + # Windows + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "https://log-api.eu.newrelic.com/log/v1" + template: templates/daemonset-windows.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "metric-api.eu.newrelic.com" + template: templates/daemonset-windows.yaml + + + - it: selects custom logs endpoint if provided + set: + licenseKey: euaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaFFFFNRAL + endpoint: custom + enableWindows: true + asserts: + # Linux + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "custom" + template: templates/daemonset.yaml + # Windows + - contains: + path: spec.template.spec.containers[0].env + content: + name: ENDPOINT + value: "custom" + template: templates/daemonset-windows.yaml \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_k8logging_exclude_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_k8logging_exclude_test.yaml new file mode 100644 index 0000000000..446f829b00 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_k8logging_exclude_test.yaml @@ -0,0 +1,45 @@ +suite: test fluent-bit exclude logging +templates: + - templates/daemonset.yaml + - templates/configmap.yaml + - templates/persistentvolume.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: K8S_LOGGING_EXCLUDE set true + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + fluentBit: + k8sLoggingExclude: true + asserts: + - equal: + path: spec.template.spec.containers[0].env[?(@.name=="K8S_LOGGING_EXCLUDE")].value + value: "true" + template: templates/daemonset.yaml + - it: K8S_LOGGING_EXCLUDE set false + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + fluentBit: + k8sLoggingExclude: false + asserts: + - equal: + path: spec.template.spec.containers[0].env[?(@.name=="K8S_LOGGING_EXCLUDE")].value + value: "false" + template: templates/daemonset.yaml + - it: K8S_LOGGING_EXCLUDE set value xyz and expect it to be set + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + fluentBit: + k8sLoggingExclude: xyz + asserts: + - equal: + path: spec.template.spec.containers[0].env[?(@.name=="K8S_LOGGING_EXCLUDE")].value + value: "xyz" + template: templates/daemonset.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_persistence_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_persistence_test.yaml new file mode 100644 index 0000000000..67d14c7955 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_persistence_test.yaml @@ -0,0 +1,317 @@ +suite: test fluent-bit persistence options +templates: + - templates/daemonset.yaml + - templates/configmap.yaml + - templates/persistentvolume.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: default persistence is hostPath, DB is set properly and logs volume is read/write + set: + licenseKey: nr_license_key + asserts: + - contains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: logs + mountPath: /var + template: templates/daemonset.yaml + - notContains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: fb-db-pvc + mountPath: /db + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.volumes + content: + name: logs + hostPath: + path: /var + template: templates/daemonset.yaml + - notContains: + path: spec.template.spec.volumes + content: + name: fb-db-pvc + persistentVolumeClaim: + claimName: my-release-newrelic-logging-pvc + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: FB_DB + value: /var/log/flb_kube.db + template: templates/daemonset.yaml + - hasDocuments: + count: 0 + template: templates/persistentvolume.yaml + - it: fluentBit.persistence set to none should keep FB_DB env empty and mount logs volume as read-only + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: none + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: FB_DB + value: "" + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: logs + mountPath: /var + readOnly: true + template: templates/daemonset.yaml + - notContains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: fb-db-pvc + mountPath: /db + template: templates/daemonset.yaml + - notContains: + path: spec.template.spec.volumes + content: + name: fb-db-pvc + persistentVolumeClaim: + claimName: my-release-newrelic-logging-pvc + template: templates/daemonset.yaml + - hasDocuments: + count: 0 + template: templates/persistentvolume.yaml + - it: fluentBit.persistence set to persistentVolume should create volume, add it to daemonset, add an initContainer to cleanup and set the FB_DB. Dynamic provisioning is enabled by default. + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: FB_DB + value: "/db/$(NODE_NAME)-fb.db" + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: logs + mountPath: /var + readOnly: true + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: fb-db-pvc + mountPath: /db + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.volumes + content: + name: fb-db-pvc + persistentVolumeClaim: + claimName: my-release-newrelic-logging-pvc + template: templates/daemonset.yaml + - isNotNullOrEmpty: + path: spec.template.spec.initContainers + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.initContainers[0].volumeMounts + content: + name: fb-db-pvc + mountPath: /db + template: templates/daemonset.yaml + - hasDocuments: + count: 1 + template: templates/persistentvolume.yaml + - isKind: + of: PersistentVolumeClaim + template: templates/persistentvolume.yaml + - equal: + path: spec.accessModes + value: + - ReadWriteMany + template: templates/persistentvolume.yaml + - it: fluentBit.persistence.persistentVolume with non dynamic provisioning should create the PV and PVC + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + dynamicProvisioning: false + asserts: + - hasDocuments: + count: 2 + template: templates/persistentvolume.yaml + - isKind: + of: PersistentVolume + documentIndex: 0 + template: templates/persistentvolume.yaml + - isKind: + of: PersistentVolumeClaim + documentIndex: 1 + template: templates/persistentvolume.yaml + - equal: + path: spec.accessModes + value: + - ReadWriteMany + documentIndex: 0 + template: templates/persistentvolume.yaml + - equal: + path: spec.accessModes + value: + - ReadWriteMany + documentIndex: 1 + template: templates/persistentvolume.yaml + - it: fluentBit.persistence storage class should be set properly on PV and PVC + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + dynamicProvisioning: false + storageClass: sample-storage-rwx + asserts: + - equal: + path: spec.storageClassName + value: sample-storage-rwx + documentIndex: 0 + template: templates/persistentvolume.yaml + - equal: + path: spec.storageClassName + value: sample-storage-rwx + documentIndex: 1 + template: templates/persistentvolume.yaml + - it: fluentBit.persistence.persistentVolume size should be set properly on PV and PVC + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + dynamicProvisioning: false + size: 100Gi + asserts: + - equal: + path: spec.capacity.storage + value: 100Gi + documentIndex: 0 + template: templates/persistentvolume.yaml + - equal: + path: spec.resources.requests.storage + value: 100Gi + documentIndex: 1 + template: templates/persistentvolume.yaml + - it: fluentBit.persistence.persistentVolume not dynamic provisioned but volumeName provided should use the volumeName and do not create a PV + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + dynamicProvisioning: false + existingVolume: existing-volume + asserts: + - hasDocuments: + count: 1 + template: templates/persistentvolume.yaml + - isKind: + of: PersistentVolumeClaim + template: templates/persistentvolume.yaml + - equal: + path: spec.volumeName + value: existing-volume + template: templates/persistentvolume.yaml + - it: fluentBit.persistence.persistentVolume if a existing claim is provided it's used and PV/PVC are not created + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + dynamicProvisioning: false + existingVolumeClaim: existing-claim + asserts: + - hasDocuments: + count: 0 + template: templates/persistentvolume.yaml + - contains: + path: spec.template.spec.volumes + content: + name: fb-db-pvc + persistentVolumeClaim: + claimName: existing-claim + template: templates/daemonset.yaml + - it: fluentBit.persistence.persistentVolume annotations for PV and PVC are used + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + annotations: + volume: + foo: bar + claim: + baz: qux + dynamicProvisioning: false + asserts: + - equal: + path: metadata.annotations.foo + value: bar + documentIndex: 0 + template: templates/persistentvolume.yaml + - equal: + path: metadata.annotations.baz + value: qux + documentIndex: 1 + template: templates/persistentvolume.yaml + - it: fluentBit.persistence.persistentVolume extra for PV and PVC are used + set: + licenseKey: nr_license_key + fluentBit: + persistence: + mode: persistentVolume + persistentVolume: + storageClass: sample-rwx + extra: + volume: + nfs: + path: /tmp/ + server: 1.1.1.1 + claim: + some: property + dynamicProvisioning: false + asserts: + - equal: + path: spec.nfs + value: + path: /tmp/ + server: 1.1.1.1 + documentIndex: 0 + template: templates/persistentvolume.yaml + - equal: + path: spec.some + value: property + documentIndex: 1 + template: templates/persistentvolume.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_pod_label_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_pod_label_test.yaml new file mode 100644 index 0000000000..86edd7ccd6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_pod_label_test.yaml @@ -0,0 +1,48 @@ +suite: test fluent-bit pods labels +templates: + - templates/daemonset.yaml + - templates/configmap.yaml + - templates/persistentvolume.yaml +release: + name: my-release + namespace: my-namespace +tests: +- it: multiple pod labels are set properly + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + podLabels: + key1: value1 + key2: value2 + asserts: + - equal: + path: spec.template.metadata.labels.key1 + value: value1 + template: templates/daemonset.yaml + - equal: + path: spec.template.metadata.labels.key2 + value: value2 + template: templates/daemonset.yaml +- it: single pod label set properly + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + podLabels: + key1: value1 + asserts: + - equal: + path: spec.template.metadata.labels.key1 + value: value1 + template: templates/daemonset.yaml +- it: pod labels are not set + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + asserts: + - notExists: + path: spec.template.metadata.labels.key1 + - notExists: + path: spec.template.metadata.labels.key2 \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_sendmetrics_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_sendmetrics_test.yaml new file mode 100644 index 0000000000..f320172cbc --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/fluentbit_sendmetrics_test.yaml @@ -0,0 +1,74 @@ +suite: test fluentbit send metrics +templates: + - templates/configmap.yaml + - templates/daemonset.yaml + - templates/daemonset-windows.yaml +release: + name: sendmetrics-release + namespace: sendmetrics-namespace +tests: + + - it: sets requirement environment variables to send metrics + set: + licenseKey: nr_license_key + enableWindows: true + fluentBit.sendMetrics: true + asserts: + # Linux + - contains: + path: spec.template.spec.containers[0].env + content: + name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: HOSTNAME + valueFrom: + fieldRef: + fieldPath: metadata.name + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: SEND_OUTPUT_PLUGIN_METRICS + value: "true" + template: templates/daemonset.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "metric-api.newrelic.com" + template: templates/daemonset.yaml + # Windows + - contains: + path: spec.template.spec.containers[0].env + content: + name: NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + template: templates/daemonset-windows.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: HOSTNAME + valueFrom: + fieldRef: + fieldPath: metadata.name + template: templates/daemonset-windows.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: SEND_OUTPUT_PLUGIN_METRICS + value: "true" + template: templates/daemonset-windows.yaml + - contains: + path: spec.template.spec.containers[0].env + content: + name: METRICS_HOST + value: "metric-api.newrelic.com" + template: templates/daemonset-windows.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/host_network_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/host_network_test.yaml new file mode 100644 index 0000000000..612d1d9a5f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/host_network_test.yaml @@ -0,0 +1,46 @@ +suite: test hostNetwork options in fluent-bit pods +templates: + - templates/configmap.yaml + - templates/daemonset.yaml + - templates/daemonset-windows.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: daemonsets does not contain hostNetwork block when not provided + set: + licenseKey: nr_license_key + enableWindows: true + asserts: + - notExists: + path: spec.template.spec.hostNetwork + template: templates/daemonset.yaml + - notExists: + path: spec.template.spec.hostNetwork + template: templates/daemonset-windows.yaml + - it: daemonsets does not contain hostNetwork block when provided as false + set: + licenseKey: nr_license_key + enableWindows: true + hostNetwork: false + asserts: + - notExists: + path: spec.template.spec.hostNetwork + template: templates/daemonset.yaml + - notExists: + path: spec.template.spec.hostNetwork + template: templates/daemonset-windows.yaml + - it: daemonsets does contain hostNetwork=true when provided as true + set: + licenseKey: nr_license_key + enableWindows: true + hostNetwork: true + asserts: + - equal: + path: spec.template.spec.hostNetwork + value: true + template: templates/daemonset.yaml + - equal: + path: spec.template.spec.hostNetwork + value: true + template: templates/daemonset-windows.yaml \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/images_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/images_test.yaml new file mode 100644 index 0000000000..533a367c51 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/images_test.yaml @@ -0,0 +1,96 @@ +suite: test images settings +templates: + - templates/configmap.yaml + - templates/daemonset.yaml + - templates/daemonset-windows.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: image names are correct + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + asserts: + - equal: + path: spec.template.spec.containers[0].image + value: newrelic/newrelic-fluentbit-output:2.0.2 + template: templates/daemonset.yaml + - equal: + path: spec.template.spec.containers[0].image + value: newrelic/newrelic-fluentbit-output:2.0.2-windows-ltsc-2019 + template: templates/daemonset-windows.yaml + documentIndex: 0 + - equal: + path: spec.template.spec.containers[0].image + value: newrelic/newrelic-fluentbit-output:2.0.2-windows-ltsc-2022 + template: templates/daemonset-windows.yaml + documentIndex: 1 + - it: global registry is used if set + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + global: + images: + registry: global_registry + asserts: + - matchRegex: + path: spec.template.spec.containers[0].image + pattern: global_registry/.* + - it: local registry overrides global + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + global: + images: + registry: global_registry + image: + registry: local_registry + asserts: + - matchRegex: + path: spec.template.spec.containers[0].image + pattern: local_registry/.* + - it: pullSecrets is used if defined + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + image: + pullSecrets: + - name: regsecret + asserts: + - equal: + path: spec.template.spec.imagePullSecrets[0].name + value: regsecret + - it: pullSecrets are merged + templates: + - templates/daemonset.yaml + - templates/daemonset-windows.yaml + set: + licenseKey: nr_license_key + enableWindows: true + global: + images: + pullSecrets: + - name: global_regsecret + image: + pullSecrets: + - name: regsecret + asserts: + - equal: + path: spec.template.spec.imagePullSecrets[0].name + value: global_regsecret + - equal: + path: spec.template.spec.imagePullSecrets[1].name + value: regsecret diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/linux_volume_mount_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/linux_volume_mount_test.yaml new file mode 100644 index 0000000000..83d2a2c118 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/linux_volume_mount_test.yaml @@ -0,0 +1,37 @@ +suite: test fluent-bit linux mount for logs +templates: + - templates/configmap.yaml + - templates/daemonset.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: is set to /var by default an + set: + licenseKey: nr_license_key + asserts: + - equal: + path: spec.template.spec.containers[0].volumeMounts[1].mountPath + value: /var + template: templates/daemonset.yaml + - equal: + path: spec.template.spec.volumes[1].hostPath.path + value: /var + template: templates/daemonset.yaml + documentIndex: 0 + - it: is set to linuxMountPath if set + templates: + - templates/daemonset.yaml + set: + licenseKey: nr_license_key + fluentBit.linuxMountPath: /var/log + asserts: + - equal: + path: spec.template.spec.containers[0].volumeMounts[1].mountPath + value: /var/log + template: templates/daemonset.yaml + - equal: + path: spec.template.spec.volumes[1].hostPath.path + value: /var/log + template: templates/daemonset.yaml + documentIndex: 0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/rbac_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/rbac_test.yaml new file mode 100644 index 0000000000..a8d85da98a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/tests/rbac_test.yaml @@ -0,0 +1,48 @@ +suite: test RBAC creation +templates: + - templates/clusterrolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: template rbac if it is configured to do it + set: + rbac.create: true + asserts: + - hasDocuments: + count: 1 + + - it: don't template rbac if it is disabled + set: + rbac.create: false + asserts: + - hasDocuments: + count: 0 + + - it: RBAC points to the service account that is created by default + set: + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: subjects[0].name + value: my-release-newrelic-logging + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: subjects[0].name + value: sa-test + + - it: RBAC points to the default service account when serviceAccount is disabled + set: + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: subjects[0].name + value: default diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/values.yaml new file mode 100644 index 0000000000..c5d85b43f4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-logging/values.yaml @@ -0,0 +1,361 @@ +# IMPORTANT: Specify your New Relic API key here. +# licenseKey: +# +# Optionally, specify a cluster name and log records can +# be filtered by cluster. +# cluster: +# +# or Specify secret which contains New Relic API key +# customSecretName: secret_name +# customSecretLicenseKey: secret_key +# +# The previous values can also be set as global so that they +# can be shared by other newrelic product's charts +# +# global: +# licenseKey: +# cluster: +# customSecretName: +# customSecretLicenseKey: +# +# IMPORTANT: if you use a kubernetes secret to specify the license, +# you have to manually provide the correct endpoint depending on +# whether your account is for the EU region or not. +# +# endpoint: https://log-api.newrelic.com/log/v1 + +fluentBit: + logLevel: "info" + path: "/var/log/containers/*.log" + linuxMountPath: /var + windowsPath: "C:\\var\\log\\containers\\*.log" + db: "/var/log/flb_kube.db" + windowsDb: "C:\\var\\log\\flb_kube.db" + criEnabled: true + k8sBufferSize: "32k" + k8sLoggingExclude: "false" + retryLimit: 5 + sendMetrics: false + extraEnv: [] + # extraEnv: + # - name: HTTPS_PROXY + # value: http://example.com:3128 + # - name: METADATA_NAME + # valueFrom: + # fieldRef: + # fieldPath: metadata.name + + # Indicates how fluent-bit database is persisted + persistence: + # Define the persistent mode for fluent-bit db, allowed options are `hostPath` (default), `none`, `persistentVolume`. + # - `hostPath` will use hostPath to store the db file on the node disk. + # - `none` will disable the fluent-bit db file, this could cause log duplication or data loss in case fluent-bit gets restarted. + # - `persistentVolume` will use a ReadWriteMany persistent volume to store the db file. This will override `fluentBit.db` path and use `/db/${NODE_NAME}-fb.db` file instead. + mode: "hostPath" + + # In case persistence.mode is set to persistentVolume this will be needed + persistentVolume: + # The storage class should allow ReadWriteMany mode + storageClass: + # Volume and claim size. + size: 10Gi + # If dynamicProvisioning is enabled the chart will create only the PersistentVolumeClaim + dynamicProvisioning: true + # If an existingVolume is provided, we'll use it instead creating a new one + existingVolume: + # If an existingVolumeClaim is provided, we'll use it instead creating a new one + existingVolumeClaim: + # In case you need to add annotations to the created volume or claim + annotations: + volume: {} + claim: {} + # In case you need to specify any other option to your volume or claim + extra: + volume: + # nfs: + # path: /tmp/ + # server: 1.1.1.1 + claim: {} + + + # New Relic default configuration for fluent-bit.conf (service, inputs, filters, outputs) + # and parsers.conf (parsers). The configuration below is not configured for lowDataMode and will + # send all attributes. If custom configuration is required, update these variables. + config: + # Note that Prometheus metric collection needs the HTTP server to be online at port 2020 (see fluentBit.config.metricInstrumentation) + service: | + [SERVICE] + Flush 1 + Log_Level ${LOG_LEVEL} + Daemon off + Parsers_File parsers.conf + HTTP_Server On + HTTP_Listen 0.0.0.0 + HTTP_Port 2020 + + inputs: | + [INPUT] + Name tail + Alias pod-logs-tailer + Tag kube.* + Path ${PATH} + multiline.parser ${LOG_PARSER} + DB ${FB_DB} + Mem_Buf_Limit 7MB + Skip_Long_Lines On + Refresh_Interval 10 + +# extraInputs: | +# [INPUT] +# Name dummy +# Tag dummy.log + + filters: | + [FILTER] + Name kubernetes + Alias kubernetes-enricher + Match kube.* + # We need the full DNS suffix as Windows only supports resolving names with this suffix + # See: https://kubernetes.io/docs/setup/production-environment/windows/intro-windows-in-kubernetes/#dns-limitations + Kube_URL https://kubernetes.default.svc.cluster.local:443 + Buffer_Size ${K8S_BUFFER_SIZE} + K8S-Logging.Exclude ${K8S_LOGGING_EXCLUDE} + + [FILTER] + Name record_modifier + Alias node-attributes-enricher + Match * + Record cluster_name "${CLUSTER_NAME}" + +# extraFilters: | +# [FILTER] +# Name grep +# Match * +# Exclude log lvl=debug* + + lowDataModeFilters: | + [FILTER] + Name kubernetes + Match kube.* + Alias kubernetes-enricher + # We need the full DNS suffix as Windows only supports resolving names with this suffix + # See: https://kubernetes.io/docs/setup/production-environment/windows/intro-windows-in-kubernetes/#dns-limitations + Kube_URL https://kubernetes.default.svc.cluster.local:443 + Buffer_Size ${K8S_BUFFER_SIZE} + K8S-Logging.Exclude ${K8S_LOGGING_EXCLUDE} + Labels Off + Annotations Off + + [FILTER] + Name nest + Match * + Alias kubernetes-attribute-lifter + Operation lift + Nested_under kubernetes + + [FILTER] + Name record_modifier + Match * + Alias node-attributes-enricher-filter + Record cluster_name "${CLUSTER_NAME}" + Allowlist_key container_name + Allowlist_key namespace_name + Allowlist_key pod_name + Allowlist_key stream + Allowlist_key message + Allowlist_key log + + outputs: | + [OUTPUT] + Name newrelic + Match * + Alias newrelic-logs-forwarder + licenseKey ${LICENSE_KEY} + endpoint ${ENDPOINT} + lowDataMode ${LOW_DATA_MODE} + sendMetrics ${SEND_OUTPUT_PLUGIN_METRICS} + Retry_Limit ${RETRY_LIMIT} + +# extraOutputs: | +# [OUTPUT] +# Name null +# Match * + +# parsers: | +# [PARSER] +# Name my_custom_parser +# Format json +# Time_Key time +# Time_Format %Y-%m-%dT%H:%M:%S.%L +# Time_Keep On + metricInstrumentation: | + [INPUT] + name prometheus_scrape + Alias fb-metrics-collector + host 127.0.0.1 + port 2020 + tag fb_metrics + metrics_path /api/v2/metrics/prometheus + scrape_interval 10s + + [OUTPUT] + Name prometheus_remote_write + Match fb_metrics + Alias fb-metrics-forwarder + Host ${METRICS_HOST} + Port 443 + Uri /prometheus/v1/write?prometheus_server=${CLUSTER_NAME} + Header Authorization Bearer ${LICENSE_KEY} + Tls On + # Windows pods using prometheus_remote_write currently have issues if TLS verify is On + Tls.verify Off + # User-defined labels + add_label app fluent-bit + add_label cluster_name "${CLUSTER_NAME}" + add_label pod_name ${HOSTNAME} + add_label node_name ${NODE_NAME} + add_label source kubernetes + +image: + repository: newrelic/newrelic-fluentbit-output +# registry: my_registry + tag: "" + pullPolicy: IfNotPresent + ## See https://kubernetes.io/docs/concepts/containers/images/#specifying-imagepullsecrets-on-a-pod + pullSecrets: [] +# - name: regsecret + +# By default, the Linux DaemonSet will always be deployed, while the Windows DaemonSet(s) won't. +enableLinux: true +enableWindows: false +# For every entry in this Windows OS list, we will create an independent DaemonSet which will get deployed +# on Windows nodes running each specific Windows version and build number. Note that +# Windows containers can only be executed on hosts running the exact same Windows version and build number, +# because Kubernetes only supports process isolation and not Hyper-V isolation (as of September 2021) +windowsOsList: + # We aim to support (limited to LTSC2019/LTSC2022 using GitHub actions, see https://github.com/actions/runner-images/tree/main/images/win): + # https://kubernetes.io/docs/setup/production-environment/windows/intro-windows-in-kubernetes/#windows-os-version-support + - version: ltsc2019 + imageTagSuffix: windows-ltsc-2019 + buildNumber: 10.0.17763 + - version: ltsc2022 + imageTagSuffix: windows-ltsc-2022 + buildNumber: 10.0.20348 + +# Default set of resources assigned to the DaemonSet pods +resources: + limits: + cpu: 500m + memory: 128Mi + requests: + cpu: 250m + memory: 64Mi + +rbac: + # Specifies whether RBAC resources should be created + create: true + pspEnabled: false + +serviceAccount: + # Specifies whether a ServiceAccount should be created + create: + # The name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template + name: + # Specify any annotations to add to the ServiceAccount + annotations: {} + +# Optionally configure ports to expose metrics on /api/v1/metrics/prometheus +# See - https://docs.fluentbit.io/manual/administration/monitoring +exposedPorts: [] +# - containerPort: 2020 +# hostPort: 2020 +# name: metrics +# protocol: TCP + +# If you wish to provide additional labels to apply to the pod(s), specify +# them here +# podLabels: + +# Pod scheduling priority +# Ref: https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ +# priorityClassName: high-priority + +# Node affinity rules +# Ref: https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#affinity-and-anti-affinity +# +# IMPORTANT # +# ######### # +# When .Values.global.fargate == true, the chart will automatically add the required affinity rules to exclude +# the DaemonSet from Fargate nodes. There is no need to manually touch this property achieve this. +# This automatic exclusion will, however, not take place if this value is overridden: Setting this to a +# non-empty value WHEN deploying in EKS Fargate (global.fargate == true) requires the user to manually +# include in their custom ruleset an exclusion for nodes with "eks.amazonaws.com/compute-type: fargate", as +# the New Relic DaemonSet MUST NOT be deployed on fargate nodes, as the operator takes care of injecting it +# as a sidecar instead. +# Please refer to the daemonset.yaml template for more details on how to achieve this. +nodeAffinity: {} + +# Node labels for pod assignment +# Ref: https://kubernetes.io/docs/user-guide/node-selection/ +# Note that the Linux DaemonSet already contains a node selector label based on their OS (kubernetes.io/os: linux). +nodeSelector: {} + +# Note that the Windows DaemonSet already contains a node selector label based on their OS (kubernetes.io/os: windows). +# and build number (node.kubernetes.io/windows-build: {{ .buildNumber }}, to ensure that each version of the DaemonSet +# gets deployed only on those Windows nodes running the exact same Windows version and build number. Note that +# Windows containers can only be executed on hosts running the exact same Windows version and build number. +windowsNodeSelector: {} + +# These are default tolerations to be able to run the New Relic Kubernetes integration. +tolerations: + - operator: "Exists" + effect: "NoSchedule" + - operator: "Exists" + effect: "NoExecute" + +updateStrategy: RollingUpdate + +# Sends data to staging, can be set as a global. +# global.nrStaging +nrStaging: false + +daemonSet: + # Annotations to add to the DaemonSet. + annotations: {} + +# Annotations to add to the resulting Pods of the DaemonSet. +podAnnotations: {} + +# If host network should be enabled for fluentbit pods. +# There are some inputs like UDP which will require this setting to be true as they need to bind to the host network. +hostNetwork: + +# When low data mode is enabled only minimal attributes are added to the logs. Kubernetes labels and +# annotations are not included. The plugin.type, plugin.version and plugin.source attributes are minified +# into the plugin.source attribute. +# Can be set as a global: global.lowDataMode +# lowDataMode: false + +extraVolumes: [] +# - name: systemdlog +# hostPath: +# path: /run/log/journal + +extraVolumeMounts: [] +# - name: systemdlog +# mountPath: /run/log/journal + +initContainers: +# - name: init +# image: busybox +# command: ["sh", "-c", 'echo "hello world"'] + +windows: + initContainers: +# - name: init +# image: ... +# command: [...] + +# -- Sets pod dnsConfig. Can also be configured with `global.dnsConfig` +dnsConfig: {} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/Chart.yaml new file mode 100644 index 0000000000..acd3077d4a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/Chart.yaml @@ -0,0 +1,18 @@ +apiVersion: v1 +appVersion: 2.1.4 +description: A Helm chart for the New Relic Pixie integration. +home: https://hub.docker.com/u/newrelic +icon: https://newrelic.com/assets/newrelic/source/NewRelic-logo-square.svg +keywords: +- newrelic +- pixie +- monitoring +maintainers: +- name: nserrino +- name: philkuz +- name: htroisi +- name: vuqtran88 +name: newrelic-pixie +sources: +- https://github.com/newrelic/ +version: 2.1.4 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/README.md new file mode 100644 index 0000000000..228a3676d3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/README.md @@ -0,0 +1,166 @@ +# newrelic-pixie + +## Chart Details + +This chart will deploy the New Relic Pixie Integration. + +IMPORTANT: In order to retrieve the Pixie cluster id from the `pl-cluster-secrets` the integration needs to be deployed in the same namespace as Pixie. By default, Pixie is installed in the `pl` namespace. Alternatively the `clusterId` can be configured manually when installing the chart. In this case the integration can be deployed to any namespace. + +## Configuration + +| Parameter | Description | Default | +| ---------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------- | +| `global.cluster` - `cluster` | The cluster name for the Kubernetes cluster. Required. | | +| `global.licenseKey` - `licenseKey` | The New Relic license key (stored in a secret). Required. | | +| `global.lowDataMode` - `lowDataMode` | If `true`, the integration performs heavier sampling on the Pixie span data and sets the collect interval to 15 seconds instead of 10 seconds. | false | +| `global.nrStaging` - `nrStaging` | Send data to staging (requires a staging license key). | false | +| `apiKey` | The Pixie API key (stored in a secret). Required. | | +| `clusterId` | The Pixie cluster id. Optional. Read from the `pl-cluster-secrets` secret if empty. | | +| `endpoint` | The Pixie endpoint. Required when using Pixie Open Source. | | +| `verbose` | Whether the integration should run in verbose mode or not. | false | +| `global.customSecretName` - `customSecretName` | Name of an existing Secret object, not created by this chart, where the New Relic license is stored | | +| `global.customSecretLicenseKey` - `customSecretLicenseKey` | Key in the existing Secret object, indicated by `customSecretName`, where the New Relic license key is stored. | | +| `image.pullSecrets` | Image pull secrets. | `nil` | +| `customSecretApiKeyName` | Name of an existing Secret object, not created by this chart, where the Pixie API key is stored. | | +| `customSecretApiKeyKey` | Key in the existing Secret object, indicated by `customSecretApiKeyName`, where the Pixie API key is stored. | | +| `podLabels` | Labels added to each Job pod | `{}` | +| `podAnnotations` | Annotations added to each Job pod | `{}` | +| `job.annotations` | Annotations added to the `newrelic-pixie` Job resource | `{}` | +| `job.labels` | Annotations added to the `newrelic-pixie` Job resource | `{}` | +| `nodeSelector` | Node label to use for scheduling. | `{}` | +| `tolerations` | List of node taints to tolerate (requires Kubernetes >= 1.6). | `[]` | +| `affinity` | Node affinity to use for scheduling. | `{}` | +| `proxy` | Set proxy to connect to Pixie Cloud and New Relic. | | +| `customScripts` | YAML containing custom scripts for long-term data retention. The results of the custom scripts will be stored in New Relic. See [custom scripts](#custom-scripts) for YAML format. | `{}` | +| `customScriptsConfigMap` | Name of an existing ConfigMap object containing custom script for long-term data retention. This configuration takes precedence over `customScripts`. | | +| `excludeNamespacesRegex` | Observability data for namespaces matching this RE2 regex is not sent to New Relic. If empty, observability data for all namespaces is sent to New Relic. | | +| `excludePodsRegex` | Observability data for pods (across all namespaces) matching this RE2 regex is not sent to New Relic. If empty, observability data for all pods (in non-excluded namespaces) is sent to New Relic. | | + +## Example + +Make sure you have [added the New Relic chart repository.](../../README.md#installing-charts) + +Then, to install this chart, run the following command: + +```sh +helm install newrelic/newrelic-pixie \ + --set cluster= \ + --set licenseKey= \ + --set apiKey= \ + --namespace pl \ + --generate-name +``` + +## Globals + +**Important:** global parameters have higher precedence than locals with the same name. + +These are meant to be used when you are writing a chart with subcharts. It helps to avoid +setting values multiple times on different subcharts. + +More information on globals and subcharts can be found at [Helm's official documentation](https://helm.sh/docs/topics/chart_template_guide/subcharts_and_globals/). + +| Parameter | +| ------------------------------- | +| `global.cluster` | +| `global.licenseKey` | +| `global.customSecretName` | +| `global.customSecretLicenseKey` | +| `global.lowDataMode` | +| `global.nrStaging` | + +## Custom scripts + +Custom scripts can either be configured directly in `customScripts` or be provided through an existing ConfigMap `customScriptsConfigMap`. + +The entries in the ConfigMap should contain file-like keys with the `.yaml` extension. Each file in the ConfigMap should be valid YAML and contain the following keys: + + * name (string): the name of the script + * description (string): description of the script + * frequencyS (int): frequency to execute the script in seconds + * scripts (string): the actual PXL script to execute + * addExcludes (optional boolean, `false` by default): add pod and namespace excludes to the custom script + +For more detailed information about the custom scripts see [the New Relic Pixie integration repo](https://github.com/newrelic/newrelic-pixie-integration/). + +```yaml +customScripts: + custom1.yaml: | + name: "custom1" + description: "Custom script 1" + frequencyS: 60 + script: | + import px + + df = px.DataFrame(table='http_events', start_time=px.plugin.start_time) + + ns_prefix = df.ctx['namespace'] + '/' + df.container = df.ctx['container_name'] + df.pod = px.strip_prefix(ns_prefix, df.ctx['pod']) + df.service = px.strip_prefix(ns_prefix, df.ctx['service']) + df.namespace = df.ctx['namespace'] + + df.status_code = df.resp_status + + df = df.groupby(['status_code', 'pod', 'container','service', 'namespace']).agg( + latency_min=('latency', px.min), + latency_max=('latency', px.max), + latency_sum=('latency', px.sum), + latency_count=('latency', px.count), + time_=('time_', px.max), + ) + + df.latency_min = df.latency_min / 1000000 + df.latency_max = df.latency_max / 1000000 + df.latency_sum = df.latency_sum / 1000000 + + df.cluster_name = px.vizier_name() + df.cluster_id = px.vizier_id() + df.pixie = 'pixie' + + px.export( + df, px.otel.Data( + resource={ + 'service.name': df.service, + 'k8s.container.name': df.container, + 'service.instance.id': df.pod, + 'k8s.pod.name': df.pod, + 'k8s.namespace.name': df.namespace, + 'px.cluster.id': df.cluster_id, + 'k8s.cluster.name': df.cluster_name, + 'instrumentation.provider': df.pixie, + }, + data=[ + px.otel.metric.Summary( + name='http.server.duration', + description='measures the duration of the inbound HTTP request', + # Unit is not supported yet + # unit='ms', + count=df.latency_count, + sum=df.latency_sum, + quantile_values={ + 0.0: df.latency_min, + 1.0: df.latency_max, + }, + attributes={ + 'http.status_code': df.status_code, + }, + )], + ), + ) +``` + + +## Resources + +The default set of resources assigned to the pods is shown below: + +```yaml +resources: + limits: + memory: 250M + requests: + cpu: 100m + memory: 250M +``` + diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/ci/test-values.yaml new file mode 100644 index 0000000000..580f9b0ba1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/ci/test-values.yaml @@ -0,0 +1,5 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + apiKey: 1234567890abcdef + cluster: test-cluster +clusterId: foobar diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/NOTES.txt new file mode 100644 index 0000000000..d542838894 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/NOTES.txt @@ -0,0 +1,27 @@ +{{- if (include "newrelic-pixie.areValuesValid" .) }} + +Your deployment of the New Relic Pixie integration is complete. + +Please ensure this integration is deployed in the same namespace +as Pixie or manually specify the clusterId. +{{- else -}} +############################################################### +#### ERROR: You did not set all the required values. #### +############################################################### + +This deployment will be incomplete until you set all the required values: + +* Cluster name +* New Relic license key +* Pixie API key + +For a simple installation to be fixed, run: + + helm upgrade {{ .Release.Name }} \ + --set cluster=YOUR-CLUSTER-NAME \ + --set licenseKey=YOUR-LICENSE-KEY \ + --set apiKey=YOUR-API-KEY \ + -n {{ .Release.Namespace }} \ + newrelic/newrelic-pixie + +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/_helpers.tpl new file mode 100644 index 0000000000..40b9c68df1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/_helpers.tpl @@ -0,0 +1,172 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "newrelic-pixie.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{- define "newrelic-pixie.namespace" -}} +{{- if .Values.namespace -}} + {{- .Values.namespace -}} +{{- else -}} + {{- .Release.Namespace | default "pl" -}} +{{- 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). +*/}} +{{- define "newrelic-pixie.fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- if ne $name .Release.Name -}} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}} +{{- else -}} +{{- printf "%s" $name | trunc 63 | trimSuffix "-" -}} +{{- end -}} +{{- end -}} + +{{/* Generate basic labels */}} +{{- define "newrelic-pixie.labels" }} +app: {{ template "newrelic-pixie.name" . }} +app.kubernetes.io/name: {{ include "newrelic-pixie.name" . }} +chart: {{ .Chart.Name }}-{{ .Chart.Version | replace "+" "_" }} +heritage: {{.Release.Service }} +release: {{.Release.Name }} +{{- end }} + +{{- define "newrelic-pixie.cluster" -}} +{{- if .Values.global -}} + {{- if .Values.global.cluster -}} + {{- .Values.global.cluster -}} + {{- else -}} + {{- .Values.cluster | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.cluster | default "" -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-pixie.nrStaging" -}} +{{- if .Values.global }} + {{- if .Values.global.nrStaging }} + {{- .Values.global.nrStaging -}} + {{- end -}} +{{- else if .Values.nrStaging }} + {{- .Values.nrStaging -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-pixie.licenseKey" -}} +{{- if .Values.global}} + {{- if .Values.global.licenseKey }} + {{- .Values.global.licenseKey -}} + {{- else -}} + {{- .Values.licenseKey | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.licenseKey | default "" -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-pixie.apiKey" -}} +{{- if .Values.global}} + {{- if .Values.global.apiKey }} + {{- .Values.global.apiKey -}} + {{- else -}} + {{- .Values.apiKey | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.apiKey | default "" -}} +{{- end -}} +{{- end -}} + +{{- /* +adapted from https://github.com/newrelic/helm-charts/blob/af747af93fb5b912374196adc59b552965b6e133/library/common-library/templates/_low-data-mode.tpl +TODO: actually use common-library chart dep +*/ -}} +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic-pixie.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the customSecretName where the New Relic license is being stored. +*/}} +{{- define "newrelic-pixie.customSecretName" -}} +{{- if .Values.global }} + {{- if .Values.global.customSecretName }} + {{- .Values.global.customSecretName -}} + {{- else -}} + {{- .Values.customSecretName | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.customSecretName | default "" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the customSecretApiKeyName where the Pixie API key is being stored. +*/}} +{{- define "newrelic-pixie.customSecretApiKeyName" -}} + {{- .Values.customSecretApiKeyName | default "" -}} +{{- end -}} + +{{/* +Return the customSecretLicenseKey +*/}} +{{- define "newrelic-pixie.customSecretLicenseKey" -}} +{{- if .Values.global }} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- else -}} + {{- .Values.customSecretLicenseKey | default "" -}} + {{- end -}} +{{- else -}} + {{- .Values.customSecretLicenseKey | default "" -}} +{{- end -}} +{{- end -}} + +{{/* +Return the customSecretApiKeyKey +*/}} +{{- define "newrelic-pixie.customSecretApiKeyKey" -}} + {{- .Values.customSecretApiKeyKey | default "" -}} +{{- end -}} + +{{/* +Returns if the template should render, it checks if the required values +licenseKey and cluster are set. +*/}} +{{- define "newrelic-pixie.areValuesValid" -}} +{{- $cluster := include "newrelic-pixie.cluster" . -}} +{{- $licenseKey := include "newrelic-pixie.licenseKey" . -}} +{{- $apiKey := include "newrelic-pixie.apiKey" . -}} +{{- $customSecretName := include "newrelic-pixie.customSecretName" . -}} +{{- $customSecretLicenseKey := include "newrelic-pixie.customSecretLicenseKey" . -}} +{{- $customSecretApiKeyKey := include "newrelic-pixie.customSecretApiKeyKey" . -}} +{{- and (or (and $licenseKey $apiKey) (and $customSecretName $customSecretLicenseKey $customSecretApiKeyKey)) $cluster}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/configmap.yaml new file mode 100644 index 0000000000..19f7fe61ac --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/configmap.yaml @@ -0,0 +1,12 @@ +{{- if (include "newrelic-pixie.areValuesValid" .) }} +{{- if .Values.customScripts }} +apiVersion: v1 +kind: ConfigMap +metadata: + namespace: {{ template "newrelic-pixie.namespace" . }} + labels: {{ include "newrelic-pixie.labels" . | indent 4 }} + name: {{ template "newrelic-pixie.fullname" . }}-scripts +data: +{{- toYaml .Values.customScripts | nindent 2 }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/job.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/job.yaml new file mode 100644 index 0000000000..89b97514fa --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/job.yaml @@ -0,0 +1,164 @@ +{{- if (include "newrelic-pixie.areValuesValid" .) }} +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ template "newrelic-pixie.fullname" . }} + namespace: {{ template "newrelic-pixie.namespace" . }} + labels: + {{- include "newrelic-pixie.labels" . | trim | nindent 4}} + {{- if ((.Values.job).labels) }} + {{- toYaml .Values.job.labels | nindent 4 }} + {{- end }} + {{- if ((.Values.job).annotations) }} + annotations: + {{ toYaml .Values.job.annotations | nindent 4 | trim }} + {{- end }} +spec: + backoffLimit: 4 + ttlSecondsAfterFinished: 600 + template: + metadata: + labels: + app.kubernetes.io/name: {{ template "newrelic-pixie.name" . }} + release: {{.Release.Name }} + {{- if .Values.podLabels }} + {{- toYaml .Values.podLabels | nindent 8 }} + {{- end }} + {{- if .Values.podAnnotations }} + annotations: + {{- toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + spec: + {{- if .Values.image.pullSecrets }} + imagePullSecrets: + {{- toYaml .Values.image.pullSecrets | nindent 8 }} + {{- end }} + restartPolicy: Never + initContainers: + - name: cluster-registration-wait + image: gcr.io/pixie-oss/pixie-dev-public/curl:1.0 + command: ['sh', '-c', 'set -x; + URL="https://${SERVICE_NAME}:${SERVICE_PORT}/readyz"; + until [ $(curl -m 0.5 -s -o /dev/null -w "%{http_code}" -k ${URL}) -eq 200 ]; do + echo "Waiting for cluster registration. If this takes too long check the vizier-cloud-connector logs." + sleep 2; + done; + '] + env: + # The name of the Pixie service which connects to Pixie Cloud for cluster registration. + - name: SERVICE_NAME + value: "vizier-cloud-connector-svc" + - name: SERVICE_PORT + value: "50800" + containers: + - name: {{ template "newrelic-pixie.name" . }} + image: "{{ .Values.image.repository }}:{{ .Values.image.tag | default .Chart.AppVersion }}" + imagePullPolicy: "{{ .Values.image.pullPolicy }}" + env: + - name: CLUSTER_NAME + value: {{ template "newrelic-pixie.cluster" . }} + - name: NR_LICENSE_KEY + valueFrom: + secretKeyRef: + {{- if (include "newrelic-pixie.licenseKey" .) }} + name: {{ template "newrelic-pixie.fullname" . }}-secrets + key: newrelicLicenseKey + {{- else }} + name: {{ include "newrelic-pixie.customSecretName" . }} + key: {{ include "newrelic-pixie.customSecretLicenseKey" . }} + {{- end }} + - name: PIXIE_API_KEY + valueFrom: + secretKeyRef: + {{- if (include "newrelic-pixie.apiKey" .) }} + name: {{ template "newrelic-pixie.fullname" . }}-secrets + key: pixieApiKey + {{- else }} + name: {{ include "newrelic-pixie.customSecretApiKeyName" . }} + key: {{ include "newrelic-pixie.customSecretApiKeyKey" . }} + {{- end }} + - name: PIXIE_CLUSTER_ID + {{- if .Values.clusterId }} + value: {{ .Values.clusterId -}} + {{- else }} + valueFrom: + secretKeyRef: + key: cluster-id + name: pl-cluster-secrets + {{- end }} + {{- if .Values.verbose }} + - name: VERBOSE + value: "true" + {{- end }} + {{- if (include "newrelic-pixie.lowDataMode" .) }} + - name: COLLECT_INTERVAL_SEC + value: "15" + - name: HTTP_SPAN_LIMIT + value: "750" + - name: DB_SPAN_LIMIT + value: "250" + {{- else }} + - name: COLLECT_INTERVAL_SEC + value: "10" + - name: HTTP_SPAN_LIMIT + value: "1500" + - name: DB_SPAN_LIMIT + value: "500" + {{- end }} + {{- if (include "newrelic-pixie.nrStaging" .) }} + - name: NR_OTLP_HOST + value: "staging-otlp.nr-data.net:4317" + {{- end }} + {{- if or .Values.endpoint (include "newrelic-pixie.nrStaging" .) }} + - name: PIXIE_ENDPOINT + {{- if .Values.endpoint }} + value: {{ .Values.endpoint | quote }} + {{- else }} + value: "staging.withpixie.dev:443" + {{- end }} + {{- end }} + {{- if .Values.proxy }} + - name: HTTP_PROXY + value: {{ .Values.proxy | quote }} + - name: HTTPS_PROXY + value: {{ .Values.proxy | quote }} + {{- end }} + {{- if .Values.excludePodsRegex }} + - name: EXCLUDE_PODS_REGEX + value: {{ .Values.excludePodsRegex | quote }} + {{- end }} + {{- if .Values.excludeNamespacesRegex }} + - name: EXCLUDE_NAMESPACES_REGEX + value: {{ .Values.excludeNamespacesRegex | quote }} + {{- end }} + {{- if .Values.resources }} + resources: + {{- toYaml .Values.resources | nindent 10 }} + {{- end }} + {{- if or .Values.customScriptsConfigMap .Values.customScripts }} + volumeMounts: + - name: scripts + mountPath: "/scripts" + readOnly: true + volumes: + - name: scripts + configMap: + {{- if .Values.customScriptsConfigMap }} + name: {{ .Values.customScriptsConfigMap }} + {{- else }} + name: {{ template "newrelic-pixie.fullname" . }}-scripts + {{- end}} + {{- end }} + {{- if $.Values.nodeSelector }} + nodeSelector: + {{- toYaml $.Values.nodeSelector | nindent 8 }} + {{- end }} + {{- if .Values.tolerations }} + tolerations: + {{- toYaml .Values.tolerations | nindent 8 }} + {{- end }} + {{- if .Values.affinity }} + affinity: + {{- toYaml .Values.affinity | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/secret.yaml new file mode 100644 index 0000000000..4d95618772 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/templates/secret.yaml @@ -0,0 +1,20 @@ +{{- if (include "newrelic-pixie.areValuesValid" .) }} +{{- $licenseKey := include "newrelic-pixie.licenseKey" . -}} +{{- $apiKey := include "newrelic-pixie.apiKey" . -}} +{{- if or $apiKey $licenseKey}} +apiVersion: v1 +kind: Secret +metadata: + namespace: {{ template "newrelic-pixie.namespace" . }} + labels: {{ include "newrelic-pixie.labels" . | indent 4 }} + name: {{ template "newrelic-pixie.fullname" . }}-secrets +type: Opaque +data: + {{- if $licenseKey }} + newrelicLicenseKey: {{ $licenseKey | b64enc }} + {{- end }} + {{- if $apiKey }} + pixieApiKey: {{ include "newrelic-pixie.apiKey" . | b64enc -}} + {{- end }} +{{- end }} +{{- end}} \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/configmap.yaml new file mode 100644 index 0000000000..ecba6363bd --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/configmap.yaml @@ -0,0 +1,44 @@ +suite: test custom scripts ConfigMap +templates: + - templates/configmap.yaml +tests: + - it: ConfigMap is created + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + customScripts: + custom1.yaml: | + name: "custom1" + description: "Custom script 1" + frequencyS: 60 + script: | + import px + df = px.DataFrame(table='http_events', start_time=px.plugin.start_time) + asserts: + - isKind: + of: ConfigMap + - equal: + path: data.custom1\.yaml + value: |- + name: "custom1" + description: "Custom script 1" + frequencyS: 60 + script: | + import px + df = px.DataFrame(table='http_events', start_time=px.plugin.start_time) + - equal: + path: metadata.name + value: RELEASE-NAME-newrelic-pixie-scripts + - equal: + path: metadata.namespace + value: NAMESPACE + - it: ConfigMap is empty + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + customScripts: {} + asserts: + - hasDocuments: + count: 0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/jobs.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/jobs.yaml new file mode 100644 index 0000000000..03a3d86b8f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/tests/jobs.yaml @@ -0,0 +1,138 @@ +suite: test job +templates: + - templates/job.yaml +tests: + - it: Test primary fields of job + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + image: + tag: "latest" + asserts: + - isKind: + of: Job + - equal: + path: "metadata.name" + value: "RELEASE-NAME-newrelic-pixie" + - equal: + path: "metadata.namespace" + value: "NAMESPACE" + - equal: + path: "spec.template.spec.containers[0].image" + value: "newrelic/newrelic-pixie-integration:latest" + - equal: + path: "spec.template.spec.containers[0].env" + value: + - name: CLUSTER_NAME + value: test-cluster + - name: NR_LICENSE_KEY + valueFrom: + secretKeyRef: + key: newrelicLicenseKey + name: RELEASE-NAME-newrelic-pixie-secrets + - name: PIXIE_API_KEY + valueFrom: + secretKeyRef: + key: pixieApiKey + name: RELEASE-NAME-newrelic-pixie-secrets + - name: PIXIE_CLUSTER_ID + valueFrom: + secretKeyRef: + key: cluster-id + name: pl-cluster-secrets + - isEmpty: + path: "spec.template.spec.containers[0].volumeMounts" + - isEmpty: + path: "spec.template.spec.volumes" + - it: Job with clusterId + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + clusterId: "cid123" + asserts: + - equal: + path: "spec.template.spec.containers[0].env" + value: + - name: CLUSTER_NAME + value: test-cluster + - name: NR_LICENSE_KEY + valueFrom: + secretKeyRef: + key: newrelicLicenseKey + name: RELEASE-NAME-newrelic-pixie-secrets + - name: PIXIE_API_KEY + valueFrom: + secretKeyRef: + key: pixieApiKey + name: RELEASE-NAME-newrelic-pixie-secrets + - name: PIXIE_CLUSTER_ID + value: "cid123" + - it: Job with Pixie endpoint + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + clusterId: "cid123" + endpoint: "withpixie.ai:443" + asserts: + - equal: + path: "spec.template.spec.containers[0].env" + value: + - name: CLUSTER_NAME + value: test-cluster + - name: NR_LICENSE_KEY + valueFrom: + secretKeyRef: + key: newrelicLicenseKey + name: RELEASE-NAME-newrelic-pixie-secrets + - name: PIXIE_API_KEY + valueFrom: + secretKeyRef: + key: pixieApiKey + name: RELEASE-NAME-newrelic-pixie-secrets + - name: PIXIE_CLUSTER_ID + value: "cid123" + - name: PIXIE_ENDPOINT + value: "withpixie.ai:443" + - it: Job with custom scripts + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + customScripts: + custom1.yaml: | + name: "custom1" + asserts: + - equal: + path: "spec.template.spec.containers[0].volumeMounts" + value: + - name: scripts + mountPath: "/scripts" + readOnly: true + - equal: + path: "spec.template.spec.volumes[0]" + value: + name: scripts + configMap: + name: RELEASE-NAME-newrelic-pixie-scripts + - it: Job with custom script in defined ConfigMap + set: + cluster: "test-cluster" + licenseKey: "license123" + apiKey: "api123" + customScriptsConfigMap: "myconfigmap" + asserts: + - equal: + path: "spec.template.spec.containers[0].volumeMounts" + value: + - name: scripts + mountPath: "/scripts" + readOnly: true + - equal: + path: "spec.template.spec.volumes[0]" + value: + name: scripts + configMap: + name: myconfigmap diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/values.yaml new file mode 100644 index 0000000000..4103d54e91 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-pixie/values.yaml @@ -0,0 +1,70 @@ +# IMPORTANT: The Kubernetes cluster name +# https://docs.newrelic.com/docs/kubernetes-monitoring-integration +# cluster: "" + +# The New Relic license key +# licenseKey: "" + +# The Pixie API key +# apiKey: "" + +# The Pixie Cluster Id +# clusterId: + +# The Pixie endpoint +# endpoint: + +# If you already have a secret where the New Relic license key is stored, indicate its name here +# customSecretName: +# The key in the customSecretName secret that contains the New Relic license key +# customSecretLicenseKey: +# If you already have a secret where the Pixie API key is stored, indicate its name here +# customSecretApiKeyName: +# The key in the customSecretApiKeyName secret that contains the Pixie API key +# customSecretApiKeyKey: + +image: + repository: newrelic/newrelic-pixie-integration + tag: "" + pullPolicy: IfNotPresent + pullSecrets: [] + # - name: regsecret + +resources: + limits: + memory: 250M + requests: + cpu: 100m + memory: 250M + +# -- Annotations to add to the pod. +podAnnotations: {} +# -- Additional labels for chart pods +podLabels: {} + +job: + # job.annotations -- Annotations to add to the Job. + annotations: {} + # job.labels -- Labels to add to the Job. + labels: {} + +proxy: {} + +nodeSelector: {} + +tolerations: [] + +affinity: {} + +customScripts: {} +# Optionally the scripts can be provided in an already existing ConfigMap: +# customScriptsConfigMap: + +excludeNamespacesRegex: +excludePodsRegex: + +# When low data mode is enabled the integration performs heavier sampling on the Pixie span data +# and sets the collect interval to 15 seconds instead of 10 seconds. +# Can be set as a global: global.lowDataMode or locally as newrelic-pixie.lowDataMode +# @default -- false +lowDataMode: diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/.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/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.lock new file mode 100644 index 0000000000..1dc01d3a1d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-08-30T00:20:40.371047222Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.yaml new file mode 100644 index 0000000000..14e3dcee3e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/Chart.yaml @@ -0,0 +1,22 @@ +annotations: + configuratorVersion: 1.17.4 +apiVersion: v2 +appVersion: v2.37.8 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy Prometheus with New Relic Prometheus Configurator. +keywords: +- newrelic +- prometheus +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: newrelic-prometheus-agent +type: application +version: 1.14.4 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md new file mode 100644 index 0000000000..069b9a79b6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md @@ -0,0 +1,244 @@ +# newrelic-prometheus-agent + +A Helm chart to deploy Prometheus with New Relic Prometheus Configurator. + +# Description + +This chart deploys Prometheus Server in Agent mode configured by the `newrelic-prometheus-configurator`. + +The solution is deployed as a StatefulSet for sharding proposes. +Each Pod will execute the `newrelic-prometheus-configurator` init container which will convert the provided config to a config file in the Prometheus format. Once the init container finishes and saves the config in a shared volume, the container running Prometheus in Agent mode will start. + +```mermaid +graph LR + subgraph pod[Pod] + direction TB + subgraph volume[shared volume] + plain[Prometheus Config] + end + + subgraph init-container[init Container] + configurator[Configurator] --> plain[Prometheus Config] + end + + subgraph container[Main Container] + plain[Prometheus Config] --> prom-agent[Prometheus-Agent] + end + + end + + subgraph configMap + NewRelic-Config --> configurator[Configurator] + end + +classDef plain fill:#ddd,stroke:#fff,stroke-width:4px,color:#000; +classDef k8s fill:#326ce5,stroke:#fff,stroke-width:4px,color:#fff; +classDef pod fill:#fff,stroke:#bbb,stroke-width:2px,color:#326ce5; +class configurator,init-container,container,prom-agent k8s; +class volume plain; +class pod pod; + +``` + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add newrelic-prometheus https://newrelic.github.io/newrelic-prometheus-configurator +helm upgrade --install newrelic newrelic-prometheus/newrelic-prometheus-agent -f your-custom-values.yaml +``` + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Chart particularities + +### Configuration + +The configuration used is similar to the [Prometheus configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/), but it includes some syntactic sugar to make easy to set up some special use-cases like Kubernetes targets, sharding and some New Relic related settings like remote write endpoints. + +The configurator will create [scrape_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#scrape_config), [relabel_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config), [remote_write](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write) and other entries based on the defined configuration. + +As general rules: +- Configs parameters having the same name as the [Prometheus configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/) should have similar behavior. For example, the `tls_config` defined inside a `Kubernetes.jobs` will have the same definition as [tls_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#tls_config) of Prometheus and will affect all targets scraped by that job. +- Configs starting with `extra_` prefix will be appended to the ones created by the Configurator. For example, the relabel configs defined in `extra_relabel_config` on the Kubernetes section will be appended to the end of the list that is already being generated by the Configurator for filtering, sharding, metadata decoration, etc. + +### Default Kubernetes jobs configuration + +By default, some Kubernetes objects are discovered and scraped by Prometheus. Taking into account the snippet from `values.yaml` below: + +```yaml + integrations_filter: + enabled: true + source_labels: ["app.kubernetes.io/name", "app.newrelic.io/name", "k8s-app"] + app_values: ["redis", "traefik", "calico", "nginx", "coredns", "etcd", "cockroachdb", "velero", "harbor", "argocd"] + jobs: + - job_name_prefix: default + target_discovery: + pod: true + endpoints: true + filter: + annotations: + prometheus.io/scrape: true + - job_name_prefix: newrelic + integrations_filter: + enabled: false + target_discovery: + pod: true + endpoints: true + filter: + annotations: + newrelic.io/scrape: true +``` + +All pods and endpoints with the `newrelic.io/scrape: true` annotation will be scraped by default. + +Moreover, the solution will scrape as well all pods and endpoints with the `prometheus.io/scrape: true` annotations and +having one of the labels matching the integrations_filter configuration. + +Notice that at any point you can turn off the integrations filters and scrape all pods and services annotated with +`prometheus.io/scrape: true` by setting `config.kubernetes.integrations_filter.integrations_filter: false` or turning +it off in any specific job. + +### Kubernetes job examples + +#### API Server metrics +By default, the API Server Service named `kubernetes` is created in the `default` namespace. The following configuration will scrape metrics from all endpoints behind the mentioned service using the Prometheus Pod bearer token as Authorization Header: + +```yaml +config: + kubernetes: + jobs: + - job_name_prefix: apiserver + target_discovery: + endpoints: true + extra_relabel_config: + # Filter endpoints on `default` namespace associated to `kubernetes` service. + - source_labels: [__meta_kubernetes_namespace, __meta_kubernetes_service_name] + action: keep + regex: default;kubernetes + + scheme: https + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + insecure_skip_verify: true + authorization: + credentials_file: /var/run/secrets/kubernetes.io/serviceaccount/token +``` + +### Metrics Filtering + +Check [docs](https://github.com/newrelic/newrelic-prometheus-configurator/blob/main/docs/MetricsFilters.md) for a detailed explanation and examples of how to filter metrics and labels. + +### Self metrics + +By default, it is defined as a job in `static_target.jobs` to obtain self-metrics. Particularly, a snippet like the one +below is used. If you define your own static_targets jobs, it is important to also include this kind of job in order +to keep getting self-metrics. + +```yaml +config: + static_targets: + jobs: + - job_name: self-metrics + targets: + - "localhost:9090" + extra_metric_relabel_config: + - source_labels: [__name__] + regex: "" + action: keep +``` + +### Low data mode + +There are two mechanisms to reduce the amount of data that this integration sends to New Relic. See this snippet from the `values.yaml` file: +```yaml +lowDataMode: false + +config: + common: + scrape_interval: 30s +``` + +You might set `lowDataMode` flag to `true` (it will filter some metrics which can also be collected using New Relic Kubernetes integration), check +`values.yaml` for details. + +It is also possible to adjust how frequently Prometheus scrapes the targets by setting up the` config.common.scrape_interval` value. + +### Affinities and tolerations + +The New Relic common library allows you to set affinities, tolerations, and node selectors globally using e.g. `.global.affinity` to ease the configuration +when you use this chart using `nri-bundle`. This chart has an extra level of granularity to the components that it deploys: +control plane, ksm, and kubelet. + +Take this snippet as an example: +```yaml +global: + affinity: {} +affinity: {} +``` + +The order to set the affinity is to set `affinity` field (at root level), if that value is empty, the chart fallbacks to `global.affinity`. + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Sets pod/node affinities set almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) | +| cluster | string | `""` | Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster`. Note it will be set as an external label in prometheus configuration, it will have precedence over `config.common.external_labels.cluster_name` and `customAttributes.cluster_name``. | +| config | object | See `values.yaml` | It holds the New Relic Prometheus configuration. Here you can easily set up Prometheus to get set metrics, discover ponds and endpoints Kubernetes and send metrics to New Relic using remote-write. | +| config.common | object | See `values.yaml` | Include global configuration for Prometheus agent. | +| config.common.scrape_interval | string | `"30s"` | How frequently to scrape targets by default, unless a different value is specified on the job. | +| config.extra_remote_write | object | `nil` | It includes additional remote-write configuration. Note this configuration is not parsed, so valid [prometheus remote_write configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write) should be provided. | +| config.extra_scrape_configs | list | `[]` | It is possible to include extra scrape configuration in [prometheus format](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#scrape_config). Please note, it should be a valid Prometheus configuration which will not be parsed by the chart. WARNING extra_scrape_configs is a raw Prometheus config. Therefore, the metrics collected thanks to it will not have by default the metadata (pod_name, service_name, ...) added by the configurator for the static or kubernetes jobs. This configuration should be used as a workaround whenever kubernetes and static job do not cover a particular use-case. | +| config.kubernetes | object | See `values.yaml` | It allows defining scrape jobs for Kubernetes in a simple way. | +| config.kubernetes.integrations_filter.app_values | list | `["redis","traefik","calico","nginx","coredns","kube-dns","etcd","cockroachdb","velero","harbor","argocd"]` | app_values used to create the regex used in the relabel config added by the integration filters configuration. Note that a single regex will be created from this list, example: '.*(?i)(app1|app2|app3).*' | +| config.kubernetes.integrations_filter.enabled | bool | `true` | enabling the integration filters, merely the targets having one of the specified labels matching one of the values of app_values are scraped. Each job configuration can override this default. | +| config.kubernetes.integrations_filter.source_labels | list | `["app.kubernetes.io/name","app.newrelic.io/name","k8s-app"]` | source_labels used to fetch label values in the relabel config added by the integration filters configuration | +| config.newrelic_remote_write | object | See `values.yaml` | Newrelic remote-write configuration settings. | +| config.static_targets | object | See `values.yaml`. | It allows defining scrape jobs for targets with static URLs. | +| config.static_targets.jobs | list | See `values.yaml`. | List of static target jobs. By default, it defines a job to get self-metrics. Please note, if you define `static_target.jobs` and would like to keep self-metrics you need to include a job like the one defined by default. | +| containerSecurityContext | object | `{}` | Sets security context (at container level). Can be configured also with `global.containerSecurityContext` | +| customAttributes | object | `{}` | Adds extra attributes to prometheus external labels. Can be configured also with `global.customAttributes`. Please note, values defined in `common.config.externar_labels` will have precedence over `customAttributes`. | +| customSecretLicenseKey | string | `""` | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` | +| customSecretName | string | `""` | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` | +| dnsConfig | object | `{}` | Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` | +| extraVolumeMounts | list | `[]` | Defines where to mount volumes specified with `extraVolumes` | +| extraVolumes | list | `[]` | Volumes to mount in the containers | +| fullnameOverride | string | `""` | Override the full name of the release | +| hostNetwork | bool | `false` | Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` | +| images.configurator | object | See `values.yaml` | Image for New Relic configurator. | +| images.prometheus | object | See `values.yaml` | Image for prometheus which is executed in agent mode. | +| images.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| labels | object | `{}` | Additional labels for chart objects. Can be configured also with `global.labels` | +| licenseKey | string | `""` | This set this license key to use. Can be configured also with `global.licenseKey` | +| lowDataMode | bool | false | Reduces the number of metrics sent in order to reduce costs. It can be configured also with `global.lowDataMode`. Specifically, it makes Prometheus stop reporting some Kubernetes cluster-specific metrics, you can see details in `static/lowdatamodedefaults.yaml`. | +| metric_type_override | object | `{"enabled":true}` | It holds the configuration for metric type override. If enabled, a series of metric relabel configs will be added to `config.newrelic_remote_write.extra_write_relabel_configs`, you can check the whole list in `static/metrictyperelabeldefaults.yaml` | +| nameOverride | string | `""` | Override the name of the chart | +| nodeSelector | object | `{}` | Sets pod's node selector almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) | +| nrStaging | bool | `false` | Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` | +| podAnnotations | object | `{}` | Annotations to be added to all pods created by the integration. | +| podLabels | object | `{}` | Additional labels for chart pods. Can be configured also with `global.podLabels` | +| podSecurityContext | object | `{}` | Sets security context (at pod level). Can be configured also with `global.podSecurityContext` | +| priorityClassName | string | `""` | Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` | +| rbac.create | bool | `true` | Whether the chart should automatically create the RBAC objects required to run. | +| rbac.pspEnabled | bool | `false` | Whether the chart should create Pod Security Policy objects. | +| resources | object | `{}` | Resource limits to be added to all pods created by the integration. | +| serviceAccount | object | See `values.yaml` | Settings controlling ServiceAccount creation. | +| serviceAccount.create | bool | `true` | Whether the chart should automatically create the ServiceAccount objects required to run. | +| sharding | string | See `values.yaml` | Set up Prometheus replicas to allow horizontal scalability. | +| tolerations | list | `[]` | Sets pod's tolerations to node taints almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) | +| verboseLog | bool | `false` | Sets the debug log to Prometheus and prometheus-configurator or all integrations if it is set globally. Can be configured also with `global.verboseLog` | + +## Maintainers + +* [juanjjaramillo](https://github.com/juanjjaramillo) +* [csongnr](https://github.com/csongnr) +* [dbudziwojskiNR](https://github.com/dbudziwojskiNR) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md.gotmpl new file mode 100644 index 0000000000..8738b73297 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/README.md.gotmpl @@ -0,0 +1,209 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +# Description + +This chart deploys Prometheus Server in Agent mode configured by the `newrelic-prometheus-configurator`. + +The solution is deployed as a StatefulSet for sharding proposes. +Each Pod will execute the `newrelic-prometheus-configurator` init container which will convert the provided config to a config file in the Prometheus format. Once the init container finishes and saves the config in a shared volume, the container running Prometheus in Agent mode will start. + +```mermaid +graph LR + subgraph pod[Pod] + direction TB + subgraph volume[shared volume] + plain[Prometheus Config] + end + + subgraph init-container[init Container] + configurator[Configurator] --> plain[Prometheus Config] + end + + subgraph container[Main Container] + plain[Prometheus Config] --> prom-agent[Prometheus-Agent] + end + + end + + subgraph configMap + NewRelic-Config --> configurator[Configurator] + end + +classDef plain fill:#ddd,stroke:#fff,stroke-width:4px,color:#000; +classDef k8s fill:#326ce5,stroke:#fff,stroke-width:4px,color:#fff; +classDef pod fill:#fff,stroke:#bbb,stroke-width:2px,color:#326ce5; +class configurator,init-container,container,prom-agent k8s; +class volume plain; +class pod pod; + +``` + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add newrelic-prometheus https://newrelic.github.io/newrelic-prometheus-configurator +helm upgrade --install newrelic newrelic-prometheus/newrelic-prometheus-agent -f your-custom-values.yaml +``` + +{{ template "chart.sourcesSection" . }} + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Chart particularities + +### Configuration + +The configuration used is similar to the [Prometheus configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/), but it includes some syntactic sugar to make easy to set up some special use-cases like Kubernetes targets, sharding and some New Relic related settings like remote write endpoints. + +The configurator will create [scrape_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#scrape_config), [relabel_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config), [remote_write](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write) and other entries based on the defined configuration. + +As general rules: +- Configs parameters having the same name as the [Prometheus configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/) should have similar behavior. For example, the `tls_config` defined inside a `Kubernetes.jobs` will have the same definition as [tls_config](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#tls_config) of Prometheus and will affect all targets scraped by that job. +- Configs starting with `extra_` prefix will be appended to the ones created by the Configurator. For example, the relabel configs defined in `extra_relabel_config` on the Kubernetes section will be appended to the end of the list that is already being generated by the Configurator for filtering, sharding, metadata decoration, etc. + +### Default Kubernetes jobs configuration + +By default, some Kubernetes objects are discovered and scraped by Prometheus. Taking into account the snippet from `values.yaml` below: + +```yaml + integrations_filter: + enabled: true + source_labels: ["app.kubernetes.io/name", "app.newrelic.io/name", "k8s-app"] + app_values: ["redis", "traefik", "calico", "nginx", "coredns", "etcd", "cockroachdb", "velero", "harbor", "argocd"] + jobs: + - job_name_prefix: default + target_discovery: + pod: true + endpoints: true + filter: + annotations: + prometheus.io/scrape: true + - job_name_prefix: newrelic + integrations_filter: + enabled: false + target_discovery: + pod: true + endpoints: true + filter: + annotations: + newrelic.io/scrape: true +``` + +All pods and endpoints with the `newrelic.io/scrape: true` annotation will be scraped by default. + +Moreover, the solution will scrape as well all pods and endpoints with the `prometheus.io/scrape: true` annotations and +having one of the labels matching the integrations_filter configuration. + +Notice that at any point you can turn off the integrations filters and scrape all pods and services annotated with +`prometheus.io/scrape: true` by setting `config.kubernetes.integrations_filter.integrations_filter: false` or turning +it off in any specific job. + +### Kubernetes job examples + +#### API Server metrics +By default, the API Server Service named `kubernetes` is created in the `default` namespace. The following configuration will scrape metrics from all endpoints behind the mentioned service using the Prometheus Pod bearer token as Authorization Header: + +```yaml +config: + kubernetes: + jobs: + - job_name_prefix: apiserver + target_discovery: + endpoints: true + extra_relabel_config: + # Filter endpoints on `default` namespace associated to `kubernetes` service. + - source_labels: [__meta_kubernetes_namespace, __meta_kubernetes_service_name] + action: keep + regex: default;kubernetes + + scheme: https + tls_config: + ca_file: /var/run/secrets/kubernetes.io/serviceaccount/ca.crt + insecure_skip_verify: true + authorization: + credentials_file: /var/run/secrets/kubernetes.io/serviceaccount/token +``` + +### Metrics Filtering + +Check [docs](https://github.com/newrelic/newrelic-prometheus-configurator/blob/main/docs/MetricsFilters.md) for a detailed explanation and examples of how to filter metrics and labels. + +### Self metrics + +By default, it is defined as a job in `static_target.jobs` to obtain self-metrics. Particularly, a snippet like the one +below is used. If you define your own static_targets jobs, it is important to also include this kind of job in order +to keep getting self-metrics. + +```yaml +config: + static_targets: + jobs: + - job_name: self-metrics + targets: + - "localhost:9090" + extra_metric_relabel_config: + - source_labels: [__name__] + regex: "" + action: keep +``` + +### Low data mode + +There are two mechanisms to reduce the amount of data that this integration sends to New Relic. See this snippet from the `values.yaml` file: +```yaml +lowDataMode: false + +config: + common: + scrape_interval: 30s +``` + +You might set `lowDataMode` flag to `true` (it will filter some metrics which can also be collected using New Relic Kubernetes integration), check +`values.yaml` for details. + +It is also possible to adjust how frequently Prometheus scrapes the targets by setting up the` config.common.scrape_interval` value. + + +### Affinities and tolerations + +The New Relic common library allows you to set affinities, tolerations, and node selectors globally using e.g. `.global.affinity` to ease the configuration +when you use this chart using `nri-bundle`. This chart has an extra level of granularity to the components that it deploys: +control plane, ksm, and kubelet. + +Take this snippet as an example: +```yaml +global: + affinity: {} +affinity: {} +``` + +The order to set the affinity is to set `affinity` field (at root level), if that value is empty, the chart fallbacks to `global.affinity`. + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/ci/test-values.yaml new file mode 100644 index 0000000000..ac5ed6bb0c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/ci/test-values.yaml @@ -0,0 +1,6 @@ +licenseKey: fakeLicenseKey +cluster: test-cluster-name +images: + configurator: + repository: ct/prometheus-configurator + tag: ct diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/lowdatamodedefaults.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/lowdatamodedefaults.yaml new file mode 100644 index 0000000000..726815755d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/lowdatamodedefaults.yaml @@ -0,0 +1,6 @@ +# This file contains an entry of the array `extra_write_relabel_configs` to filter +# metrics on Low Data Mode. These metrics are already collected by the New Relic Kubernetes Integration. +low_data_mode: +- action: drop + source_labels: [__name__] + regex: "kube_.+|container_.+|machine_.+|cadvisor_.+" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/metrictyperelabeldefaults.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/metrictyperelabeldefaults.yaml new file mode 100644 index 0000000000..c0a2774094 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/static/metrictyperelabeldefaults.yaml @@ -0,0 +1,17 @@ +# This file contains an entry of the array `extra_write_relabel_configs` to override metric types. +# https://docs.newrelic.com/docs/infrastructure/prometheus-integrations/install-configure-remote-write/set-your-prometheus-remote-write-integration#override-mapping +metrics_type_relabel: +- source_labels: [__name__] + separator: ; + regex: timeseries_write_(.*) # Cockroach + target_label: newrelic_metric_type + replacement: counter + action: replace +- source_labels: [__name__] + separator: ; + regex: sql_byte(.*) # Cockroach + target_label: newrelic_metric_type + replacement: counter + action: replace +# Note that adding more elements to this list could cause a possible breaking change to users already leveraging affected metrics. +# Therefore, before adding new entries check if any users is relying already on those metrics and warn them. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/_helpers.tpl new file mode 100644 index 0000000000..6cc58e251d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/_helpers.tpl @@ -0,0 +1,165 @@ +{{- /* Return the newrelic-prometheus configuration */ -}} + +{{- /* it builds the common configuration from configurator config, cluster name and custom attributes */ -}} +{{- define "newrelic-prometheus.configurator.common" -}} +{{- $tmp := dict "external_labels" (dict "cluster_name" (include "newrelic.common.cluster" . )) -}} + +{{- if .Values.config -}} + {{- if .Values.config.common -}} + {{- $_ := mustMerge $tmp .Values.config.common -}} + {{- end -}} +{{- end -}} + +{{- $tmpCustomAttribute := dict "external_labels" (include "newrelic.common.customAttributes" . | fromYaml ) -}} +{{- $tmp = mustMerge $tmp $tmpCustomAttribute -}} + +common: +{{- $tmp | toYaml | nindent 2 -}} + +{{- end -}} + + +{{- /* it builds the newrelic_remote_write configuration from configurator config */ -}} +{{- define "newrelic-prometheus.configurator.newrelic_remote_write" -}} +{{- $tmp := dict -}} + +{{- if include "newrelic.common.nrStaging" . -}} + {{- $_ := set $tmp "staging" true -}} +{{- end -}} + +{{- if include "newrelic.common.fedramp.enabled" . -}} + {{- $_ := set $tmp "fedramp" (dict "enabled" true) -}} +{{- end -}} + +{{- $extra_write_relabel_configs :=(include "newrelic-prometheus.configurator.extra_write_relabel_configs" . | fromYaml) -}} +{{- if ne (len $extra_write_relabel_configs.list) 0 -}} + {{- $_ := set $tmp "extra_write_relabel_configs" $extra_write_relabel_configs.list -}} +{{- end -}} + +{{- if .Values.config -}} +{{- if .Values.config.newrelic_remote_write -}} + {{- $tmp = mustMerge $tmp .Values.config.newrelic_remote_write -}} +{{- end -}} +{{- end -}} + +{{- if not (empty $tmp) -}} + {{- dict "newrelic_remote_write" $tmp | toYaml -}} +{{- end -}} + +{{- end -}} + +{{- /* it builds the extra_write_relabel_configs configuration merging: lowdatamode, user ones, and metrictyperelabeldefaults */ -}} +{{- define "newrelic-prometheus.configurator.extra_write_relabel_configs" -}} + +{{- $extra_write_relabel_configs := list -}} +{{- if (include "newrelic.common.lowDataMode" .) -}} + {{- $lowDataModeRelabelConfig := .Files.Get "static/lowdatamodedefaults.yaml" | fromYaml -}} + {{- $extra_write_relabel_configs = concat $extra_write_relabel_configs $lowDataModeRelabelConfig.low_data_mode -}} +{{- end -}} + +{{- if .Values.metric_type_override -}} + {{- if .Values.metric_type_override.enabled -}} + {{- $metricTypeOverride := .Files.Get "static/metrictyperelabeldefaults.yaml" | fromYaml -}} + {{- $extra_write_relabel_configs = concat $extra_write_relabel_configs $metricTypeOverride.metrics_type_relabel -}} + {{- end -}} +{{- end -}} + +{{- if .Values.config -}} +{{- if .Values.config.newrelic_remote_write -}} + {{- /* it concatenates the defined 'extra_write_relabel_configs' to the ones defined in lowDataMode */ -}} + {{- if .Values.config.newrelic_remote_write.extra_write_relabel_configs -}} + {{- $extra_write_relabel_configs = concat $extra_write_relabel_configs .Values.config.newrelic_remote_write.extra_write_relabel_configs -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- /* sadly in helm we cannot pass back a list without putting it into a tmp dict */ -}} +{{ dict "list" $extra_write_relabel_configs | toYaml}} + +{{- end -}} + + +{{- /* it builds the extra_remote_write configuration from configurator config */ -}} +{{- define "newrelic-prometheus.configurator.extra_remote_write" -}} +{{- if .Values.config -}} + {{- if .Values.config.extra_remote_write -}} +extra_remote_write: + {{- .Values.config.extra_remote_write | toYaml | nindent 2 -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-prometheus.configurator.static_targets" -}} +{{- if .Values.config -}} + {{- if .Values.config.static_targets -}} +static_targets: + {{- .Values.config.static_targets | toYaml | nindent 2 -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-prometheus.configurator.extra_scrape_configs" -}} +{{- if .Values.config -}} + {{- if .Values.config.extra_scrape_configs -}} +extra_scrape_configs: + {{- .Values.config.extra_scrape_configs | toYaml | nindent 2 -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-prometheus.configurator.kubernetes" -}} +{{- if .Values.config -}} +{{- if .Values.config.kubernetes -}} +kubernetes: + {{- if .Values.config.kubernetes.jobs }} + jobs: + {{- .Values.config.kubernetes.jobs | toYaml | nindent 2 -}} + {{- end -}} + + {{- if .Values.config.kubernetes.integrations_filter }} + integrations_filter: + {{- .Values.config.kubernetes.integrations_filter | toYaml | nindent 4 -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + +{{- define "newrelic-prometheus.configurator.sharding" -}} + {{- if .Values.sharding -}} +sharding: + total_shards_count: {{ include "newrelic-prometheus.configurator.replicas" . }} + {{- end -}} +{{- end -}} + +{{- define "newrelic-prometheus.configurator.replicas" -}} + {{- if .Values.sharding -}} +{{- .Values.sharding.total_shards_count | default 1 }} + {{- else -}} +1 + {{- end -}} +{{- end -}} + +{{- /* +Return the proper configurator image name +{{ include "newrelic-prometheus.configurator.images.configurator_image" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic-prometheus.configurator.configurator_image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "context" .context) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic-prometheus.configurator.configurator_image.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + +{{- /* +Return the proper image tag for the configurator image +{{ include "newrelic-prometheus.configurator.configurator_image.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic-prometheus.configurator.configurator_image.tag" -}} + {{- .imageRoot.tag | default .context.Chart.Annotations.configuratorVersion | toString -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrole.yaml new file mode 100644 index 0000000000..e9d4208e2f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrole.yaml @@ -0,0 +1,24 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - endpoints + - services + - pods + - services + verbs: + - get + - list + - watch + - nonResourceURLs: + - "/metrics" + verbs: + - get +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..44244653ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic.common.naming.fullname" . }} +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/configmap.yaml new file mode 100644 index 0000000000..b775aca746 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/configmap.yaml @@ -0,0 +1,31 @@ +kind: ConfigMap +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +apiVersion: v1 +data: + config.yaml: |- + # Configuration for newrelic-prometheus-configurator + {{- with (include "newrelic-prometheus.configurator.newrelic_remote_write" . ) -}} + {{- . | nindent 4 }} + {{- end -}} + {{- with (include "newrelic-prometheus.configurator.extra_remote_write" . ) -}} + {{- . | nindent 4 }} + {{- end -}} + {{- with (include "newrelic-prometheus.configurator.static_targets" . ) -}} + {{- . | nindent 4 }} + {{- end -}} + {{- with (include "newrelic-prometheus.configurator.extra_scrape_configs" . ) -}} + {{- . | nindent 4 }} + {{- end -}} + {{- with (include "newrelic-prometheus.configurator.common" . ) -}} + {{- . | nindent 4 }} + {{- end -}} + {{- with (include "newrelic-prometheus.configurator.kubernetes" . ) -}} + {{- . | nindent 4 }} + {{- end -}} + {{- with (include "newrelic-prometheus.configurator.sharding" . ) -}} + {{- . | nindent 4 }} + {{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/secret.yaml new file mode 100644 index 0000000000..f558ee86c9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/secret.yaml @@ -0,0 +1,2 @@ +{{- /* Common library will take care of creating the secret or not. */}} +{{- include "newrelic.common.license.secret" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/serviceaccount.yaml new file mode 100644 index 0000000000..b1e74523e5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- if include "newrelic.common.serviceAccount.annotations" . }} + annotations: + {{- include "newrelic.common.serviceAccount.annotations" . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/statefulset.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/statefulset.yaml new file mode 100644 index 0000000000..846c41c233 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/templates/statefulset.yaml @@ -0,0 +1,157 @@ +apiVersion: apps/v1 +kind: StatefulSet +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} +spec: + serviceName: {{ include "newrelic.common.naming.fullname" . }}-headless + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + replicas: {{ include "newrelic-prometheus.configurator.replicas" . }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.images.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} + {{- if include "newrelic.common.hostNetwork" . }} + dnsPolicy: ClusterFirstWithHostNet + {{- end }} + + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + + initContainers: + - name: configurator + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 12 }} + {{- end }} + image: {{ include "newrelic-prometheus.configurator.configurator_image" ( dict "imageRoot" .Values.images.configurator "context" .) }} + imagePullPolicy: {{ .Values.images.configurator.pullPolicy }} + args: + - --input=/etc/configurator/config.yaml + - --output=/etc/prometheus/config/config.yaml + {{- if include "newrelic.common.verboseLog" . }} + - --verbose=true + {{- end }} + {{- with .Values.resources.configurator }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: configurator-config + mountPath: /etc/configurator/ + - name: prometheus-config + mountPath: /etc/prometheus/config + env: + - name: NR_PROM_DATA_SOURCE_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NR_PROM_LICENSE_KEY + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + - name: NR_PROM_CHART_VERSION + value: {{ .Chart.Version }} + + containers: + - name: prometheus + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 12 }} + {{- end }} + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.prometheus "context" .) }} + imagePullPolicy: {{ .Values.images.prometheus.pullPolicy }} + ports: + - containerPort: 9090 + protocol: TCP + livenessProbe: + httpGet: + path: /-/healthy + port: 9090 + scheme: HTTP + initialDelaySeconds: 10 + periodSeconds: 15 + timeoutSeconds: 10 + failureThreshold: 3 + successThreshold: 1 + readinessProbe: + httpGet: + path: /-/ready + port: 9090 + scheme: HTTP + initialDelaySeconds: 10 + periodSeconds: 5 + timeoutSeconds: 4 + failureThreshold: 3 + successThreshold: 1 + args: + - --config.file=/etc/prometheus/config/config.yaml + - --enable-feature=agent,expand-external-labels + - --storage.agent.retention.max-time=30m + - --storage.agent.wal-truncate-frequency=30m + - --storage.agent.path=/etc/prometheus/storage + {{- if include "newrelic.common.verboseLog" . }} + - --log.level=debug + {{- end }} + {{- with .Values.resources.prometheus }} + resources: {{ toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: prometheus-config + mountPath: /etc/prometheus/config + - name: prometheus-storage + mountPath: /etc/prometheus/storage + {{- with .Values.extraVolumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + + volumes: + - name: configurator-config + configMap: + name: {{ include "newrelic.common.naming.fullname" . }} + - name: prometheus-config + emptyDir: {} + - name: prometheus-storage + emptyDir: {} + {{- with .Values.extraVolumes }} + {{- toYaml . | nindent 8 }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configmap_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configmap_test.yaml new file mode 100644 index 0000000000..f2dd0468ea --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configmap_test.yaml @@ -0,0 +1,572 @@ +suite: test configmap +templates: + - templates/configmap.yaml +tests: + - it: config with defaults + set: + licenseKey: license-key-test + cluster: cluster-test + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: replace + regex: timeseries_write_(.*) + replacement: counter + separator: ; + source_labels: + - __name__ + target_label: newrelic_metric_type + - action: replace + regex: sql_byte(.*) + replacement: counter + separator: ; + source_labels: + - __name__ + target_label: newrelic_metric_type + static_targets: + jobs: + - extra_metric_relabel_config: + - action: keep + regex: prometheus_agent_active_series|prometheus_target_interval_length_seconds|prometheus_target_scrape_pool_targets|prometheus_remote_storage_samples_pending|prometheus_remote_storage_samples_in_total|prometheus_remote_storage_samples_retried_total|prometheus_agent_corruptions_total|prometheus_remote_storage_shards|prometheus_sd_kubernetes_events_total|prometheus_agent_checkpoint_creations_failed_total|prometheus_agent_checkpoint_deletions_failed_total|prometheus_remote_storage_samples_dropped_total|prometheus_remote_storage_samples_failed_total|prometheus_sd_kubernetes_http_request_total|prometheus_agent_truncate_duration_seconds_sum|prometheus_build_info|process_resident_memory_bytes|process_virtual_memory_bytes|process_cpu_seconds_total|prometheus_remote_storage_bytes_total + source_labels: + - __name__ + job_name: self-metrics + skip_sharding: true + targets: + - localhost:9090 + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + kubernetes: + jobs: + - job_name_prefix: default + target_discovery: + endpoints: true + filter: + annotations: + prometheus.io/scrape: true + pod: true + - integrations_filter: + enabled: false + job_name_prefix: newrelic + target_discovery: + endpoints: true + filter: + annotations: + newrelic.io/scrape: true + pod: true + integrations_filter: + app_values: + - redis + - traefik + - calico + - nginx + - coredns + - kube-dns + - etcd + - cockroachdb + - velero + - harbor + - argocd + enabled: true + source_labels: + - app.kubernetes.io/name + - app.newrelic.io/name + - k8s-app + + - it: staging is enabled + set: + licenseKey: license-key-test + cluster: cluster-test + nrStaging: true + metric_type_override: + enabled: false + config: + static_targets: # Set empty to make this test simple + asserts: + - matchRegex: + path: data["config.yaml"] + pattern: "newrelic_remote_write:\n staging: true" # We do not want to test the whole YAML + + - it: fedramp is enabled + set: + licenseKey: license-key-test + cluster: cluster-test + fedramp: + enabled: true + metric_type_override: + enabled: false + config: + static_targets: # Set empty to make this test simple + asserts: + - matchRegex: + path: data["config.yaml"] + pattern: "newrelic_remote_write:\n fedramp:\n enabled: true" # We do not want to test the whole YAML + + - it: config including remote_write most possible sections + set: + licenseKey: license-key-test + cluster: cluster-test + nrStaging: true + config: + newrelic_remote_write: + proxy_url: http://proxy.url + remote_timeout: 30s + tls_config: + insecure_skip_verify: true + queue_config: + retry_on_http_429: false + extra_write_relabel_configs: + - source_labels: + - __name__ + - instance + regex: node_memory_active_bytes;localhost:9100 + action: drop + extra_remote_write: + - url: "https://second.remote.write" + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: replace + regex: timeseries_write_(.*) + replacement: counter + separator: ; + source_labels: + - __name__ + target_label: newrelic_metric_type + - action: replace + regex: sql_byte(.*) + replacement: counter + separator: ; + source_labels: + - __name__ + target_label: newrelic_metric_type + - action: drop + regex: node_memory_active_bytes;localhost:9100 + source_labels: + - __name__ + - instance + proxy_url: http://proxy.url + queue_config: + retry_on_http_429: false + remote_timeout: 30s + staging: true + tls_config: + insecure_skip_verify: true + extra_remote_write: + - url: https://second.remote.write + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: config including remote_write.extra_write_relabel_configs and not metric relabels + set: + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: false + config: + newrelic_remote_write: + extra_write_relabel_configs: + - source_labels: + - __name__ + - instance + regex: node_memory_active_bytes;localhost:9100 + action: drop + + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: drop + regex: node_memory_active_bytes;localhost:9100 + source_labels: + - __name__ + - instance + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: cluster_name is set from global + set: + licenseKey: license-key-test + global: + cluster: "test" + metric_type_override: + enabled: false + config: + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: test + scrape_interval: 30s + - it: cluster_name local value has precedence over global precedence + set: + licenseKey: license-key-test + global: + cluster: "test" + cluster: "test2" + metric_type_override: + enabled: false + config: + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: test2 + scrape_interval: 30s + - it: cluster_name is not overwritten from customAttributes + set: + licenseKey: license-key-test + global: + cluster: "test" + cluster: "test2" + customAttributes: + cluster_name: "test3" + metric_type_override: + enabled: false + config: + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: test2 + scrape_interval: 30s + + - it: cluster_name has precedence over extra labels has precedence over customAttributes + set: + licenseKey: license-key-test + cluster: test + customAttributes: + attribute: "value" + one: error + cluster_name: "different" + metric_type_override: + enabled: false + config: + common: + external_labels: + one: two + cluster_name: "different" + scrape_interval: 15 + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + attribute: value + cluster_name: test + one: two + scrape_interval: 15 + + - it: config including static_targets overwritten with most possible sections + set: + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: false + config: + static_targets: + jobs: + - job_name: my-custom-target-authorization-full + targets: + - "192.168.3.1:2379" + params: + q: [ "puppies" ] + oe: [ "utf8" ] + scheme: "https" + body_size_limit: 100MiB + sample_limit: 2000 + target_limit: 2000 + label_limit: 2000 + label_name_length_limit: 2000 + label_value_length_limit: 2000 + scrape_interval: 15s + scrape_timeout: 15s + tls_config: + insecure_skip_verify: true + ca_file: /path/to/ca.crt + key_file: /path/to/key.crt + cert_file: /path/to/cert.crt + server_name: server.name + min_version: TLS12 + authorization: + type: Bearer + credentials: "fancy-credentials" + extra_relabel_config: + - source_labels: [ '__name__', 'instance' ] + regex: node_memory_active_bytes;localhost:9100 + action: drop + extra_metric_relabel_config: + - source_labels: [ '__name__', 'instance' ] + regex: node_memory_active_bytes;localhost:9100 + action: drop + extra_scrape_configs: + - job_name: extra-scrape-config + static_configs: + - targets: + - "192.168.3.1:2379" + labels: + label1: value1 + label2: value2 + scrape_interval: 15s + scrape_timeout: 15s + tls_config: + insecure_skip_verify: true + ca_file: /path/to/ca.crt + key_file: /path/to/key.crt + cert_file: /path/to/cert.crt + server_name: server.name + min_version: TLS12 + authorization: + type: Bearer + credentials: "fancy-credentials" + relabel_configs: + - source_labels: [ '__name__', 'instance' ] + regex: node_memory_active_bytes;localhost:9100 + action: drop + metric_relabel_configs: + - source_labels: [ '__name__', 'instance' ] + regex: node_memory_active_bytes;localhost:9100 + action: drop + # Set empty to make this test simple + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + static_targets: + jobs: + - authorization: + credentials: fancy-credentials + type: Bearer + body_size_limit: 100MiB + extra_metric_relabel_config: + - action: drop + regex: node_memory_active_bytes;localhost:9100 + source_labels: + - __name__ + - instance + extra_relabel_config: + - action: drop + regex: node_memory_active_bytes;localhost:9100 + source_labels: + - __name__ + - instance + job_name: my-custom-target-authorization-full + label_limit: 2000 + label_name_length_limit: 2000 + label_value_length_limit: 2000 + params: + oe: + - utf8 + q: + - puppies + sample_limit: 2000 + scheme: https + scrape_interval: 15s + scrape_timeout: 15s + target_limit: 2000 + targets: + - 192.168.3.1:2379 + tls_config: + ca_file: /path/to/ca.crt + cert_file: /path/to/cert.crt + insecure_skip_verify: true + key_file: /path/to/key.crt + min_version: TLS12 + server_name: server.name + extra_scrape_configs: + - authorization: + credentials: fancy-credentials + type: Bearer + job_name: extra-scrape-config + metric_relabel_configs: + - action: drop + regex: node_memory_active_bytes;localhost:9100 + source_labels: + - __name__ + - instance + relabel_configs: + - action: drop + regex: node_memory_active_bytes;localhost:9100 + source_labels: + - __name__ + - instance + scrape_interval: 15s + scrape_timeout: 15s + static_configs: + - labels: + label1: value1 + label2: value2 + targets: + - 192.168.3.1:2379 + tls_config: + ca_file: /path/to/ca.crt + cert_file: /path/to/cert.crt + insecure_skip_verify: true + key_file: /path/to/key.crt + min_version: TLS12 + server_name: server.name + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: kubernetes config section custom values + set: + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: false + config: + kubernetes: + integrations_filter: + enabled: false + jobs: + - job_name_prefix: pod-job + target_discovery: + pod: true + endpoints: false + filter: + annotations: + custom/scrape-pod: true + - job_name_prefix: endpoints-job + target_discovery: + pod: false + endpoints: true + filter: + annotations: + custom/scrape-endpoints: true + # Set empty to make this test simple + static_targets: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + kubernetes: + jobs: + - job_name_prefix: pod-job + target_discovery: + endpoints: false + filter: + annotations: + custom/scrape-pod: true + pod: true + - job_name_prefix: endpoints-job + target_discovery: + endpoints: true + filter: + annotations: + custom/scrape-endpoints: true + pod: false + integrations_filter: + app_values: + - redis + - traefik + - calico + - nginx + - coredns + - kube-dns + - etcd + - cockroachdb + - velero + - harbor + - argocd + enabled: false + source_labels: + - app.kubernetes.io/name + - app.newrelic.io/name + - k8s-app + + - it: sharding empty not propagated + set: + licenseKey: license-key-test + cluster: cluster-test + sharding: + metric_type_override: + enabled: false + config: + kubernetes: + static_targets: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: sharding config custom values + set: + licenseKey: license-key-test + cluster: cluster-test + sharding: + total_shards_count: 2 + metric_type_override: + enabled: false + config: + kubernetes: + static_targets: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + sharding: + total_shards_count: 2 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configurator_image_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configurator_image_test.yaml new file mode 100644 index 0000000000..0f5da69bf8 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/configurator_image_test.yaml @@ -0,0 +1,57 @@ +suite: test image +templates: + - templates/statefulset.yaml + - templates/configmap.yaml +tests: + - it: configurator image is set + set: + licenseKey: license-key-test + cluster: cluster-test + images: + configurator: + tag: "test" + pullPolicy: Never + prometheus: + tag: "test-2" + asserts: + - template: templates/statefulset.yaml + equal: + path: spec.template.spec.initContainers[0].image + value: "newrelic/newrelic-prometheus-configurator:test" + - equal: + path: spec.template.spec.initContainers[0].imagePullPolicy + value: "Never" + template: templates/statefulset.yaml + - template: templates/statefulset.yaml + equal: + path: spec.template.spec.containers[0].image + value: "quay.io/prometheus/prometheus:test-2" + - equal: + path: spec.template.spec.containers[0].imagePullPolicy + value: "IfNotPresent" + template: templates/statefulset.yaml + + - it: has a linux node selector by default + set: + licenseKey: license-key-test + cluster: my-cluster + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + template: templates/statefulset.yaml + + - it: has a linux node selector and additional selectors + set: + licenseKey: license-key-test + cluster: my-cluster + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue + template: templates/statefulset.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/integration_filters_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/integration_filters_test.yaml new file mode 100644 index 0000000000..d1813f1352 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/integration_filters_test.yaml @@ -0,0 +1,119 @@ +suite: test configmap with IntegrationFilter +templates: + - templates/configmap.yaml +tests: + - it: config with IntegrationFilter true + set: + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: false + config: + kubernetes: + integrations_filter: + enabled: true + # Set empty to make this test simple + static_targets: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + kubernetes: + jobs: + - job_name_prefix: default + target_discovery: + endpoints: true + filter: + annotations: + prometheus.io/scrape: true + pod: true + - integrations_filter: + enabled: false + job_name_prefix: newrelic + target_discovery: + endpoints: true + filter: + annotations: + newrelic.io/scrape: true + pod: true + integrations_filter: + app_values: + - redis + - traefik + - calico + - nginx + - coredns + - kube-dns + - etcd + - cockroachdb + - velero + - harbor + - argocd + enabled: true + source_labels: + - app.kubernetes.io/name + - app.newrelic.io/name + - k8s-app + + - it: config with IntegrationFilter false + set: + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: false + config: + kubernetes: + integrations_filter: + enabled: false + # Set empty to make this test simple + static_targets: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + kubernetes: + jobs: + - job_name_prefix: default + target_discovery: + endpoints: true + filter: + annotations: + prometheus.io/scrape: true + pod: true + - integrations_filter: + enabled: false + job_name_prefix: newrelic + target_discovery: + endpoints: true + filter: + annotations: + newrelic.io/scrape: true + pod: true + integrations_filter: + app_values: + - redis + - traefik + - calico + - nginx + - coredns + - kube-dns + - etcd + - cockroachdb + - velero + - harbor + - argocd + enabled: false + source_labels: + - app.kubernetes.io/name + - app.newrelic.io/name + - k8s-app diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/lowdatamode_configmap_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/lowdatamode_configmap_test.yaml new file mode 100644 index 0000000000..ac3953df6d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/tests/lowdatamode_configmap_test.yaml @@ -0,0 +1,138 @@ +suite: test configmap with LowDataMode +templates: + - templates/configmap.yaml +tests: + - it: config with lowDataMode true + set: + licenseKey: license-key-test + cluster: cluster-test + lowDataMode: true + metric_type_override: + enabled: false + config: + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: drop + regex: kube_.+|container_.+|machine_.+|cadvisor_.+ + source_labels: + - __name__ + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: config with lowDataMode and nrStaging true + set: + licenseKey: license-key-test + cluster: cluster-test + lowDataMode: true + nrStaging: true + metric_type_override: + enabled: false + config: + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: drop + regex: kube_.+|container_.+|machine_.+|cadvisor_.+ + source_labels: + - __name__ + staging: true + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: config with lowDataMode true from global config + set: + global: + lowDataMode: true + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: false + config: + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: drop + regex: kube_.+|container_.+|machine_.+|cadvisor_.+ + source_labels: + - __name__ + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s + + - it: existing relabel configs are appended to low data mode and metric_type_override relabel configs. + set: + lowDataMode: true + licenseKey: license-key-test + cluster: cluster-test + metric_type_override: + enabled: true + config: + newrelic_remote_write: + extra_write_relabel_configs: + - action: drop + regex: my_custom_metric_relabel_config + source_labels: + - __name__ + # Set empty to make this test simple + static_targets: + kubernetes: + asserts: + - equal: + path: data["config.yaml"] + value: |- + # Configuration for newrelic-prometheus-configurator + newrelic_remote_write: + extra_write_relabel_configs: + - action: drop + regex: kube_.+|container_.+|machine_.+|cadvisor_.+ + source_labels: + - __name__ + - action: replace + regex: timeseries_write_(.*) + replacement: counter + separator: ; + source_labels: + - __name__ + target_label: newrelic_metric_type + - action: replace + regex: sql_byte(.*) + replacement: counter + separator: ; + source_labels: + - __name__ + target_label: newrelic_metric_type + - action: drop + regex: my_custom_metric_relabel_config + source_labels: + - __name__ + common: + external_labels: + cluster_name: cluster-test + scrape_interval: 30s diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/values.yaml new file mode 100644 index 0000000000..2fb3ed7bca --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/newrelic-prometheus-agent/values.yaml @@ -0,0 +1,473 @@ +# -- Override the name of the chart +nameOverride: "" +# -- Override the full name of the release +fullnameOverride: "" + +# -- Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster`. +# Note it will be set as an external label in prometheus configuration, it will have precedence over `config.common.external_labels.cluster_name` +# and `customAttributes.cluster_name``. +cluster: "" +# -- This set this license key to use. Can be configured also with `global.licenseKey` +licenseKey: "" +# -- In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` +customSecretName: "" +# -- In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` +customSecretLicenseKey: "" + +# -- Adds extra attributes to prometheus external labels. Can be configured also with `global.customAttributes`. Please note, values defined +# in `common.config.externar_labels` will have precedence over `customAttributes`. +customAttributes: {} + +# Images used by the chart for prometheus and New Relic configurator. +# @default See `values.yaml` +images: + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + + # -- Image for New Relic configurator. + # @default -- See `values.yaml` + configurator: + registry: "" + repository: newrelic/newrelic-prometheus-configurator + pullPolicy: IfNotPresent + # @default It defaults to `annotation.configuratorVersion` in `Chart.yaml`. + tag: "" + # -- Image for prometheus which is executed in agent mode. + # @default -- See `values.yaml` + prometheus: + registry: "" + repository: quay.io/prometheus/prometheus + pullPolicy: IfNotPresent + # @default It defaults to `appVersion` in `Chart.yaml`. + tag: "" + +# -- Volumes to mount in the containers +extraVolumes: [] +# -- Defines where to mount volumes specified with `extraVolumes` +extraVolumeMounts: [] + +# -- Settings controlling ServiceAccount creation. +# @default -- See `values.yaml` +serviceAccount: + # -- Whether the chart should automatically create the ServiceAccount objects required to run. + create: true + annotations: {} + # If not set and create is true, a name is generated using the full name template + name: "" + +# -- Additional labels for chart objects. Can be configured also with `global.labels` +labels: {} +# -- Annotations to be added to all pods created by the integration. +podAnnotations: {} +# -- Additional labels for chart pods. Can be configured also with `global.podLabels` +podLabels: {} + +# -- Resource limits to be added to all pods created by the integration. +# @default -- `{}` +resources: + prometheus: {} + +# -- Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` +priorityClassName: "" +# -- (bool) Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` +# @default -- `false` +hostNetwork: +# -- Sets security context (at pod level). Can be configured also with `global.podSecurityContext` +podSecurityContext: {} +# -- Sets security context (at container level). Can be configured also with `global.containerSecurityContext` +containerSecurityContext: {} + +# -- Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` +dnsConfig: {} + +# Settings controlling RBAC objects creation. +rbac: + # -- Whether the chart should automatically create the RBAC objects required to run. + create: true + # -- Whether the chart should create Pod Security Policy objects. + pspEnabled: false + +# -- Sets pod/node affinities set almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) +affinity: {} +# -- Sets pod's node selector almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) +nodeSelector: {} +# -- Sets pod's tolerations to node taints almost globally. (See [Affinities and tolerations](README.md#affinities-and-tolerations)) +tolerations: [] + +# -- (bool) Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` +# @default -- `false` +nrStaging: + +# -- (bool) Reduces the number of metrics sent in order to reduce costs. It can be configured also with `global.lowDataMode`. +# Specifically, it makes Prometheus stop reporting some Kubernetes cluster-specific metrics, you can see details in `static/lowdatamodedefaults.yaml`. +# @default -- false +lowDataMode: + +# -- It holds the configuration for metric type override. If enabled, a series of metric relabel configs will be added to +# `config.newrelic_remote_write.extra_write_relabel_configs`, you can check the whole list in `static/metrictyperelabeldefaults.yaml` +metric_type_override: + enabled: true + +# -- Set up Prometheus replicas to allow horizontal scalability. +# @default -- See `values.yaml` +sharding: + # -- Sets the number of Prometheus instances running on sharding mode. + # @default -- `1` + # total_shards_count: + +# -- (bool) Sets the debug log to Prometheus and prometheus-configurator or all integrations if it is set globally. Can be configured also with `global.verboseLog` +# @default -- `false` +verboseLog: + +# -- It holds the New Relic Prometheus configuration. Here you can easily set up Prometheus to get set metrics, discover +# ponds and endpoints Kubernetes and send metrics to New Relic using remote-write. +# @default -- See `values.yaml` +config: + # -- Include global configuration for Prometheus agent. + # @default -- See `values.yaml` + common: + # -- The labels to add to any timeseries that this Prometheus instance scrapes. + # @default -- `{}` + # external_labels: + # label_key_example: foo-bar + # -- How frequently to scrape targets by default, unless a different value is specified on the job. + scrape_interval: 30s + # -- The default timeout when scraping targets. + # @default -- `10s` + # scrape_timeout: + + # -- (object) Newrelic remote-write configuration settings. + # @default -- See `values.yaml` + newrelic_remote_write: + # # -- Includes additional [relabel configs](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#relabel_config) + # # for the New Relic remote write. + # # @default -- `[]` + # extra_write_relabel_configs: [] + + # # Enable the extra_write_relabel_configs below for backwards compatibility with legacy POMI labels. + # # This helpful when migrating from POMI to ensure that Prometheus metrics will contain both labels (e.g. cluster_name and clusterName). + # # For more migration info, please visit the [migration guide](https://docs.newrelic.com/docs/infrastructure/prometheus-integrations/install-configure-prometheus-agent/migration-guide/). + # - source_labels: [namespace] + # action: replace + # target_label: namespaceName + # - source_labels: [node] + # action: replace + # target_label: nodeName + # - source_labels: [pod] + # action: replace + # target_label: podName + # - source_labels: [service] + # action: replace + # target_label: serviceName + # - source_labels: [cluster_name] + # action: replace + # target_label: clusterName + # - source_labels: [job] + # action: replace + # target_label: scrapedTargetKind + # - source_labels: [instance] + # action: replace + # target_label: scrapedTargetInstance + + # -- Set up the proxy used to send metrics to New Relic. + # @default -- `""` + # proxy_url: + + # -- # Timeout for requests to the remote write endpoint. + # @default -- `30s` + # remote_timeout: + + # -- Fine-tune remote-write behavior: . + # queue_config: + # -- Remote Write shard capacity. + # @default -- `2500` + # capacity: + # -- Maximum number of shards. + # @default -- `200` + # max_shards: + # -- Minimum number of shards. + # @default -- `1` + # min_shards: + # -- Maximum number of samples per send. + # @default -- `500` + # max_samples_per_send: + # -- Maximum time a sample will wait in the buffer. + # @default -- `5s` + # batch_send_deadline: + # -- Initial retry delay. Gets doubled for every retry. + # @default -- `30ms` + # min_backoff: + # -- Maximum retry delay. + # @default -- `5s` + # max_backoff: + # -- Retry upon receiving a 429 status code from the remote-write storage. + # @default -- `false` + # retry_on_http_429: + + # -- (object) It includes additional remote-write configuration. Note this configuration is not parsed, so valid + # [prometheus remote_write configuration](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write) + # should be provided. + extra_remote_write: + + # -- It allows defining scrape jobs for Kubernetes in a simple way. + # @default -- See `values.yaml` + kubernetes: + # NewRelic provides a list of Dashboards, alerts and entities for several Services. The integrations_filter configuration + # allows to scrape only the targets having this experience out of the box. + # If integrations_filter is enabled, then the jobs scrape merely the targets having one of the specified labels matching + # one of the values of app_values. + # Under the hood, a relabel_configs with 'action=keep' are generated, consider it in case any custom extra_relabel_config is needed. + integrations_filter: + # -- enabling the integration filters, merely the targets having one of the specified labels matching + # one of the values of app_values are scraped. Each job configuration can override this default. + enabled: true + # -- source_labels used to fetch label values in the relabel config added by the integration filters configuration + source_labels: ["app.kubernetes.io/name", "app.newrelic.io/name", "k8s-app"] + # -- app_values used to create the regex used in the relabel config added by the integration filters configuration. + # Note that a single regex will be created from this list, example: '.*(?i)(app1|app2|app3).*' + app_values: ["redis", "traefik", "calico", "nginx", "coredns", "kube-dns", "etcd", "cockroachdb", "velero", "harbor", "argocd"] + + # Kubernetes jobs define [kubernetes_sd_configs](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#kubernetes_sd_config) + # to discover and scrape Kubernetes objects. Besides, a set of relabel_configs are included in order to include some Kubernetes metadata as + # Labels. For example, address, metrics_path, URL scheme, prometheus_io_parameters, namespace, pod name, service name and labels are taken + # to set the corresponding labels. + # Please note, the relabeling allows configuring the pod/endpoints scrape using the following annotations: + # - `prometheus.io/scheme`: If the metrics endpoint is secured then you will need to set this to `https` + # - `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 for service endpoints or to + # the default 9102 for pods. + # - `prometheus.io/param_`: To include additional parameters in the scrape URL. + jobs: + # 'default' scrapes all targets having 'prometheus.io/scrape: true'. + # Out of the box, since kubernetes.integrations_filter.enabled=true then only targets selected by the integration filters are considered. + - job_name_prefix: default + target_discovery: + pod: true + endpoints: true + filter: + annotations: + prometheus.io/scrape: true + # -- integrations_filter configuration for this specific job. It overrides kubernetes.integrations_filter configuration + # integrations_filter: + + # 'newrelic' scrapes all targets having 'newrelic.io/scrape: true'. + # This is useful to extend the targets scraped by the 'default' job allowlisting services leveraging `newrelic.io/scrape` annotation + - job_name_prefix: newrelic + integrations_filter: + enabled: false + target_discovery: + pod: true + endpoints: true + filter: + annotations: + newrelic.io/scrape: true + + # -- Set up the job name prefix. The final Prometheus `job` name will be composed of + the target discovery kind. ie: `default-pod` + # @default -- `""` + # - job_name_prefix: + + # -- The target discovery field allows customizing how Kubernetes discovery works. + # target_discovery: + + # -- Whether pods should be discovered. + # @default -- `false` + # pod: + + # -- Whether endpoints should be discovered. + # @default -- `false` + # endpoints: + + # -- Defines filtering criteria, it is possible to set labels and/or annotations. All filters will apply (defined + # filters are taken into account as an "AND operation"). + # @default -- `{}` + # filter: + # -- Map of annotations that the targets should have. If only the annotation name is defined, the filter only checks if exists. + # @default -- `{}` + # annotations: + + # -- Map of labels that the targets should have. If only the label name is defined, the filter only checks if exists. + # @default -- `{}` + # labels: + + # -- Advanced configs of the Kubernetes service discovery `kuberentes_sd_config` options, + # check [prometheus documentation](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#kubernetes_sd_config) for details. + # Notice that using `filter` is the recommended way to filter targets to avoid adding load to the API Server. + # additional_config: + # kubeconfig_file: "" + # namespaces: {} + # selectors: {} + # attach_metadata: {} + + + # -- The HTTP resource path on which to fetch metrics from targets. + # Use `prometheus.io/path` pod/service annotation to override this or modify it here. + # @default -- `/metrics` + # metrics_path: + + # -- Optional HTTP URL parameters. + # Use `prometheus.io/param_` pod/service annotation to include additional parameters in the scrape url or modify it here. + # @default -- `{}` + # params: + + # -- Configures the protocol scheme used for requests. + # Annotate the service/pod with `prometheus.io/scheme=https` if the secured port is used or modify it here. + # @default -- `http` + # scheme: + + # -- How frequently to scrape targets from this job. + # @default -- defined in `common.scrape_interval` + # scrape_interval: + + # -- Per-scrape timeout when scraping this job. + # @default -- defined in `common.scrape_timeout` + # scrape_timeout: + + # -- Configures the scrape request's TLS settings. + # @default -- `{}` + # tls_config: + # -- CA certificate file path to validate API server certificate with. + # @default -- `""` + # ca_file: + + # -- Certificate and key files path for client cert authentication to the server. + # @default -- `""` + # cert_file: + # key_file: + + # Disable validation of the server certificate. + # @default -- `false` + # insecure_skip_verify: + + # -- Sets the `Authorization` Bearer token header on every scrape request + # @default -- `{}` + # authorization: + # Sets the credentials to the credentials read from the configured file. + # @default -- `""` + # credentials_file: + + # -- Sets the `Authorization` header on every scrape request with the configured username and password. + # @default -- `{}` + # basic_auth: + # username: + # password_file: + + # -- List of relabeling configurations. Used if needed to add any special filter or label manipulation before the scrape takes place. + # @default -- `[]` + # extra_relabel_config: + + # -- List of metric relabel configurations. Used it to filter metrics and labels after scrape. + # @default -- `[]` + # extra_metric_relabel_config: + + + # -- It allows defining scrape jobs for targets with static URLs. + # @default -- See `values.yaml`. + static_targets: + # -- List of static target jobs. By default, it defines a job to get self-metrics. Please note, if you define `static_target.jobs` and would like to keep + # self-metrics you need to include a job like the one defined by default. + # @default -- See `values.yaml`. + jobs: + - job_name: self-metrics + skip_sharding: true # sharding is skipped to obtain self-metrics from all Prometheus servers. + targets: + - "localhost:9090" + extra_metric_relabel_config: + - source_labels: [__name__] + regex: "\ + prometheus_agent_active_series|\ + prometheus_target_interval_length_seconds|\ + prometheus_target_scrape_pool_targets|\ + prometheus_remote_storage_samples_pending|\ + prometheus_remote_storage_samples_in_total|\ + prometheus_remote_storage_samples_retried_total|\ + prometheus_agent_corruptions_total|\ + prometheus_remote_storage_shards|\ + prometheus_sd_kubernetes_events_total|\ + prometheus_agent_checkpoint_creations_failed_total|\ + prometheus_agent_checkpoint_deletions_failed_total|\ + prometheus_remote_storage_samples_dropped_total|\ + prometheus_remote_storage_samples_failed_total|\ + prometheus_sd_kubernetes_http_request_total|\ + prometheus_agent_truncate_duration_seconds_sum|\ + prometheus_build_info|\ + process_resident_memory_bytes|\ + process_virtual_memory_bytes|\ + process_cpu_seconds_total|\ + prometheus_remote_storage_bytes_total" + action: keep + + # -- The job name assigned to scraped metrics by default. + # @default -- `""`. + # - job_name: + # -- List of target URLs to be scraped by this job. + # @default -- `[]`. + # targets: + + # -- Labels assigned to all metrics scraped from the targets. + # @default -- `{}`. + # labels: + + # -- The HTTP resource path on which to fetch metrics from targets. + # @default -- `/metrics` + # metrics_path: + + # -- Optional HTTP URL parameters. + # @default -- `{}` + # params: + + # -- Configures the protocol scheme used for requests. + # @default -- `http` + # scheme: + + # -- How frequently to scrape targets from this job. + # @default -- defined in `common.scrape_interval` + # scrape_interval: + + # -- Per-scrape timeout when scraping this job. + # @default -- defined in `common.scrape_timeout` + # scrape_timeout: + + # -- Configures the scrape request's TLS settings. + # @default -- `{}` + # tls_config: + # -- CA certificate file path to validate API server certificate with. + # @default -- `""` + # ca_file: + + # -- Certificate and key files path for client cert authentication to the server. + # @default -- `""` + # cert_file: + # key_file: + + # Disable validation of the server certificate. + # @default -- `false` + # insecure_skip_verify: + + # -- Sets the `Authorization` Bearer token header on every scrape request + # @default -- `{}` + # authorization: + # Sets the credentials to the credentials read from the configured file. + # @default -- `""` + # credentials_file: + + # -- Sets the `Authorization` header on every scrape request with the configured username and password. + # @default -- `{}` + # basic_auth: + # username: + # password_file: + + # -- List of relabeling configurations. Used if needed to add any special filter or label manipulation before the scrape takes place. + # @default -- `[]` + # extra_relabel_config: + + # -- List of metric relabel configurations. Used it to filter metrics and labels after scrape. + # @default -- `[]` + # extra_metric_relabel_config: + + + # -- It is possible to include extra scrape configuration in [prometheus format](https://prometheus.io/docs/prometheus/latest/configuration/configuration/#scrape_config). + # Please note, it should be a valid Prometheus configuration which will not be parsed by the chart. + # WARNING extra_scrape_configs is a raw Prometheus config. Therefore, the metrics collected thanks to it will not have by default the metadata (pod_name, service_name, ...) added by the configurator for the static or kubernetes jobs. + # This configuration should be used as a workaround whenever kubernetes and static job do not cover a particular use-case. + # @default -- `[]` + extra_scrape_configs: [] diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.lock new file mode 100644 index 0000000000..d524c92920 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-08-30T23:46:01.668441447Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.yaml new file mode 100644 index 0000000000..6f1031d968 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/Chart.yaml @@ -0,0 +1,26 @@ +apiVersion: v2 +appVersion: 2.10.8 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy the New Relic Kube Events router +home: https://docs.newrelic.com/docs/integrations/kubernetes-integration/kubernetes-events/install-kubernetes-events-integration +icon: https://newrelic.com/themes/custom/curio/assets/mediakit/NR_logo_Horizontal.svg +keywords: +- infrastructure +- newrelic +- monitoring +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: nri-kube-events +sources: +- https://github.com/newrelic/nri-kube-events/ +- https://github.com/newrelic/nri-kube-events/tree/main/charts/nri-kube-events +- https://github.com/newrelic/infrastructure-agent/ +version: 3.10.8 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md new file mode 100644 index 0000000000..a2236f984f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md @@ -0,0 +1,79 @@ +# nri-kube-events + +![Version: 3.10.8](https://img.shields.io/badge/Version-3.10.8-informational?style=flat-square) ![AppVersion: 2.10.8](https://img.shields.io/badge/AppVersion-2.10.8-informational?style=flat-square) + +A Helm chart to deploy the New Relic Kube Events router + +**Homepage:** + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-kube-events https://newrelic.github.io/nri-kube-events +helm upgrade --install nri-kube-events/nri-kube-events -f your-custom-values.yaml +``` + +## Source Code + +* +* +* + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Sets pod/node affinities. Can be configured also with `global.affinity` | +| agentHTTPTimeout | string | `"30s"` | Amount of time to wait until timeout to send metrics to the metric forwarder | +| cluster | string | `""` | Name of the Kubernetes cluster monitored. Mandatory. Can be configured also with `global.cluster` | +| containerSecurityContext | object | `{}` | Sets security context (at container level). Can be configured also with `global.containerSecurityContext` | +| customAttributes | object | `{}` | Adds extra attributes to the cluster and all the metrics emitted to the backend. Can be configured also with `global.customAttributes` | +| customSecretLicenseKey | string | `""` | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` | +| customSecretName | string | `""` | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` | +| deployment.annotations | object | `{}` | Annotations to add to the Deployment. | +| dnsConfig | object | `{}` | Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` | +| fedramp.enabled | bool | `false` | Enables FedRAMP. Can be configured also with `global.fedramp.enabled` | +| forwarder | object | `{"resources":{}}` | Resources for the forwarder sidecar container | +| fullnameOverride | string | `""` | Override the full name of the release | +| hostNetwork | bool | `false` | Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` | +| images | object | See `values.yaml` | Images used by the chart for the integration and agents | +| images.agent | object | See `values.yaml` | Image for the New Relic Infrastructure Agent sidecar | +| images.integration | object | See `values.yaml` | Image for the New Relic Kubernetes integration | +| images.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| labels | object | `{}` | Additional labels for chart objects | +| licenseKey | string | `""` | This set this license key to use. Can be configured also with `global.licenseKey` | +| nameOverride | string | `""` | Override the name of the chart | +| nodeSelector | object | `{}` | Sets pod's node selector. Can be configured also with `global.nodeSelector` | +| nrStaging | bool | `false` | Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` | +| podAnnotations | object | `{}` | Annotations to add to the pod. | +| podLabels | object | `{}` | Additional labels for chart pods | +| podSecurityContext | object | `{}` | Sets security context (at pod level). Can be configured also with `global.podSecurityContext` | +| priorityClassName | string | `""` | Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` | +| proxy | string | `""` | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port`. Can be configured also with `global.proxy` | +| rbac.create | bool | `true` | Specifies whether RBAC resources should be created | +| resources | object | `{}` | Resources for the integration container | +| scrapers | object | See `values.yaml` | Configure the various kinds of scrapers that should be run. | +| serviceAccount | object | See `values.yaml` | Settings controlling ServiceAccount creation | +| serviceAccount.create | bool | `true` | Specifies whether a ServiceAccount should be created | +| sinks | object | See `values.yaml` | Configure where will the metrics be written. Mostly for debugging purposes. | +| sinks.newRelicInfra | bool | `true` | The newRelicInfra sink sends all events to New Relic. | +| sinks.stdout | bool | `false` | Enable the stdout sink to also see all events in the logs. | +| tolerations | list | `[]` | Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` | +| verboseLog | bool | `false` | Sets the debug logs to this integration or all integrations if it is set globally. Can be configured also with `global.verboseLog` | + +## Maintainers + +* [juanjjaramillo](https://github.com/juanjjaramillo) +* [csongnr](https://github.com/csongnr) +* [dbudziwojskiNR](https://github.com/dbudziwojskiNR) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md.gotmpl new file mode 100644 index 0000000000..e77eb7f14a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/README.md.gotmpl @@ -0,0 +1,43 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.badgesSection" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-kube-events https://newrelic.github.io/nri-kube-events +helm upgrade --install nri-kube-events/nri-kube-events -f your-custom-values.yaml +``` + +{{ template "chart.sourcesSection" . }} + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-bare-minimum-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-bare-minimum-values.yaml new file mode 100644 index 0000000000..3fb7df0506 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-bare-minimum-values.yaml @@ -0,0 +1,3 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-map.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-map.yaml new file mode 100644 index 0000000000..9fec33dc63 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-map.yaml @@ -0,0 +1,12 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +customAttributes: + test_tag_label: test_tag_value + +image: + kubeEvents: + repository: e2e/nri-kube-events + tag: test + pullPolicy: IfNotPresent diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-string.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-string.yaml new file mode 100644 index 0000000000..e12cba339c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-custom-attributes-as-string.yaml @@ -0,0 +1,11 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +customAttributes: '{"test_tag_label": "test_tag_value"}' + +image: + kubeEvents: + repository: e2e/nri-kube-events + tag: test + pullPolicy: IfNotPresent diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-values.yaml new file mode 100644 index 0000000000..4e517d6667 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/ci/test-values.yaml @@ -0,0 +1,60 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +sinks: + # Enable the stdout sink to also see all events in the logs. + stdout: true + # The newRelicInfra sink sends all events to New relic. + newRelicInfra: true + +customAttributes: + test_tag_label: test_tag_value + +config: + accountID: 111 + region: EU + +rbac: + create: true + +serviceAccount: + create: true + +podAnnotations: + annotation1: "annotation" + +nodeSelector: + kubernetes.io/os: linux + +tolerations: + - key: "key1" + effect: "NoSchedule" + operator: "Exists" + +affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: kubernetes.io/os + operator: In + values: + - linux + +hostNetwork: true + +dnsConfig: + nameservers: + - 1.2.3.4 + searches: + - my.dns.search.suffix + options: + - name: ndots + value: "1" + +image: + kubeEvents: + repository: e2e/nri-kube-events + tag: test + pullPolicy: IfNotPresent diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/NOTES.txt new file mode 100644 index 0000000000..3fd06b4a29 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/NOTES.txt @@ -0,0 +1,3 @@ +{{ include "nri-kube-events.compatibility.message.securityContext.runAsUser" . }} + +{{ include "nri-kube-events.compatibility.message.images" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers.tpl new file mode 100644 index 0000000000..5d0b8d257d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers.tpl @@ -0,0 +1,45 @@ +{{/* vim: set filetype=mustache: */}} + +{{- define "nri-kube-events.securityContext.pod" -}} +{{- $defaults := fromYaml ( include "nriKubernetes.securityContext.podDefaults" . ) -}} +{{- $compatibilityLayer := include "nri-kube-events.compatibility.securityContext.pod" . | fromYaml -}} +{{- $commonLibrary := fromYaml ( include "newrelic.common.securityContext.pod" . ) -}} + +{{- $finalSecurityContext := dict -}} +{{- if $commonLibrary -}} + {{- $finalSecurityContext = mustMergeOverwrite $commonLibrary $compatibilityLayer -}} +{{- else -}} + {{- $finalSecurityContext = mustMergeOverwrite $defaults $compatibilityLayer -}} +{{- end -}} +{{- toYaml $finalSecurityContext -}} +{{- end -}} + + + +{{- /* These are the defaults that are used for all the containers in this chart */ -}} +{{- define "nriKubernetes.securityContext.podDefaults" -}} +runAsUser: 1000 +runAsNonRoot: true +{{- end -}} + + + +{{- define "nri-kube-events.securityContext.container" -}} +{{- if include "newrelic.common.securityContext.container" . -}} +{{- include "newrelic.common.securityContext.container" . -}} +{{- else -}} +privileged: false +allowPrivilegeEscalation: false +readOnlyRootFilesystem: true +{{- end -}} +{{- end -}} + + + +{{- /* */ -}} +{{- define "nri-kube-events.agentConfig" -}} +is_forward_only: true +http_server_enabled: true +http_server_port: 8001 +{{ include "newrelic.common.agentConfig.defaults" . }} +{{- end -}} \ No newline at end of file diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers_compatibility.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers_compatibility.tpl new file mode 100644 index 0000000000..059cfff12a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/_helpers_compatibility.tpl @@ -0,0 +1,262 @@ +{{/* +Returns a dictionary with legacy runAsUser config. +We know that it only has "one line" but it is separated from the rest of the helpers because it is a temporary things +that we should EOL. The EOL time of this will be marked when we GA the deprecation of Helm v2. +*/}} +{{- define "nri-kube-events.compatibility.securityContext.pod" -}} +{{- if .Values.runAsUser -}} +runAsUser: {{ .Values.runAsUser }} +{{- end -}} +{{- end -}} + + + +{{- /* +Functions to get values from the globals instead of the common library +We make this because there could be difficult to see what is going under +the hood if we use the common-library here. So it is easy to read something +like: +{{- $registry := $oldRegistry | default $newRegistry | default $globalRegistry -}} +*/ -}} +{{- define "nri-kube-events.compatibility.global.registry" -}} + {{- if .Values.global -}} + {{- if .Values.global.images -}} + {{- if .Values.global.images.registry -}} + {{- .Values.global.images.registry -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Functions to fetch integration image configuration from the old .Values.image */ -}} +{{- /* integration's old registry */ -}} +{{- define "nri-kube-events.compatibility.old.integration.registry" -}} + {{- if .Values.image -}} + {{- if .Values.image.kubeEvents -}} + {{- if .Values.image.kubeEvents.registry -}} + {{- .Values.image.kubeEvents.registry -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* integration's old repository */ -}} +{{- define "nri-kube-events.compatibility.old.integration.repository" -}} + {{- if .Values.image -}} + {{- if .Values.image.kubeEvents -}} + {{- if .Values.image.kubeEvents.repository -}} + {{- .Values.image.kubeEvents.repository -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* integration's old tag */ -}} +{{- define "nri-kube-events.compatibility.old.integration.tag" -}} + {{- if .Values.image -}} + {{- if .Values.image.kubeEvents -}} + {{- if .Values.image.kubeEvents.tag -}} + {{- .Values.image.kubeEvents.tag -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* integration's old imagePullPolicy */ -}} +{{- define "nri-kube-events.compatibility.old.integration.pullPolicy" -}} + {{- if .Values.image -}} + {{- if .Values.image.kubeEvents -}} + {{- if .Values.image.kubeEvents.pullPolicy -}} + {{- .Values.image.kubeEvents.pullPolicy -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Functions to fetch agent image configuration from the old .Values.image */ -}} +{{- /* agent's old registry */ -}} +{{- define "nri-kube-events.compatibility.old.agent.registry" -}} + {{- if .Values.image -}} + {{- if .Values.image.infraAgent -}} + {{- if .Values.image.infraAgent.registry -}} + {{- .Values.image.infraAgent.registry -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* agent's old repository */ -}} +{{- define "nri-kube-events.compatibility.old.agent.repository" -}} + {{- if .Values.image -}} + {{- if .Values.image.infraAgent -}} + {{- if .Values.image.infraAgent.repository -}} + {{- .Values.image.infraAgent.repository -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* agent's old tag */ -}} +{{- define "nri-kube-events.compatibility.old.agent.tag" -}} + {{- if .Values.image -}} + {{- if .Values.image.infraAgent -}} + {{- if .Values.image.infraAgent.tag -}} + {{- .Values.image.infraAgent.tag -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* agent's old imagePullPolicy */ -}} +{{- define "nri-kube-events.compatibility.old.agent.pullPolicy" -}} + {{- if .Values.image -}} + {{- if .Values.image.infraAgent -}} + {{- if .Values.image.infraAgent.pullPolicy -}} + {{- .Values.image.infraAgent.pullPolicy -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{/* +Creates the image string needed to pull the integration image respecting the breaking change we made in the values file +*/}} +{{- define "nri-kube-events.compatibility.images.integration" -}} +{{- $globalRegistry := include "nri-kube-events.compatibility.global.registry" . -}} +{{- $oldRegistry := include "nri-kube-events.compatibility.old.integration.registry" . -}} +{{- $newRegistry := .Values.images.integration.registry -}} +{{- $registry := $oldRegistry | default $newRegistry | default $globalRegistry -}} + +{{- $oldRepository := include "nri-kube-events.compatibility.old.integration.repository" . -}} +{{- $newRepository := .Values.images.integration.repository -}} +{{- $repository := $oldRepository | default $newRepository }} + +{{- $oldTag := include "nri-kube-events.compatibility.old.integration.tag" . -}} +{{- $newTag := .Values.images.integration.tag -}} +{{- $tag := $oldTag | default $newTag | default .Chart.AppVersion -}} + +{{- if $registry -}} + {{- printf "%s/%s:%s" $registry $repository $tag -}} +{{- else -}} + {{- printf "%s:%s" $repository $tag -}} +{{- end -}} +{{- end -}} + + + +{{/* +Creates the image string needed to pull the agent's image respecting the breaking change we made in the values file +*/}} +{{- define "nri-kube-events.compatibility.images.agent" -}} +{{- $globalRegistry := include "nri-kube-events.compatibility.global.registry" . -}} +{{- $oldRegistry := include "nri-kube-events.compatibility.old.agent.registry" . -}} +{{- $newRegistry := .Values.images.agent.registry -}} +{{- $registry := $oldRegistry | default $newRegistry | default $globalRegistry -}} + +{{- $oldRepository := include "nri-kube-events.compatibility.old.agent.repository" . -}} +{{- $newRepository := .Values.images.agent.repository -}} +{{- $repository := $oldRepository | default $newRepository }} + +{{- $oldTag := include "nri-kube-events.compatibility.old.agent.tag" . -}} +{{- $newTag := .Values.images.agent.tag -}} +{{- $tag := $oldTag | default $newTag -}} + +{{- if $registry -}} + {{- printf "%s/%s:%s" $registry $repository $tag -}} +{{- else -}} + {{- printf "%s:%s" $repository $tag -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the pull policy for the integration image taking into account that we made a breaking change on the values path. +*/}} +{{- define "nri-kube-events.compatibility.images.pullPolicy.integration" -}} +{{- $old := include "nri-kube-events.compatibility.old.integration.pullPolicy" . -}} +{{- $new := .Values.images.integration.pullPolicy -}} + +{{- $old | default $new -}} +{{- end -}} + + + +{{/* +Returns the pull policy for the agent image taking into account that we made a breaking change on the values path. +*/}} +{{- define "nri-kube-events.compatibility.images.pullPolicy.agent" -}} +{{- $old := include "nri-kube-events.compatibility.old.agent.pullPolicy" . -}} +{{- $new := .Values.images.agent.pullPolicy -}} + +{{- $old | default $new -}} +{{- end -}} + + + +{{/* +Returns a merged list of pull secrets ready to be used +*/}} +{{- define "nri-kube-events.compatibility.images.renderPullSecrets" -}} +{{- $list := list -}} + +{{- if .Values.image -}} + {{- if .Values.image.pullSecrets -}} + {{- $list = append $list .Values.image.pullSecrets }} + {{- end -}} +{{- end -}} + +{{- if .Values.images.pullSecrets -}} + {{- $list = append $list .Values.images.pullSecrets -}} +{{- end -}} + +{{- include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" $list "context" .) }} +{{- end -}} + + + +{{- /* Messege to show to the user saying that image value is not supported anymore */ -}} +{{- define "nri-kube-events.compatibility.message.images" -}} +{{- $oldIntegrationRegistry := include "nri-kube-events.compatibility.old.integration.registry" . -}} +{{- $oldIntegrationRepository := include "nri-kube-events.compatibility.old.integration.repository" . -}} +{{- $oldIntegrationTag := include "nri-kube-events.compatibility.old.integration.tag" . -}} +{{- $oldIntegrationPullPolicy := include "nri-kube-events.compatibility.old.integration.pullPolicy" . -}} +{{- $oldAgentRegistry := include "nri-kube-events.compatibility.old.agent.registry" . -}} +{{- $oldAgentRepository := include "nri-kube-events.compatibility.old.agent.repository" . -}} +{{- $oldAgentTag := include "nri-kube-events.compatibility.old.agent.tag" . -}} +{{- $oldAgentPullPolicy := include "nri-kube-events.compatibility.old.agent.pullPolicy" . -}} + +{{- if or $oldIntegrationRegistry $oldIntegrationRepository $oldIntegrationTag $oldIntegrationPullPolicy $oldAgentRegistry $oldAgentRepository $oldAgentTag $oldAgentPullPolicy }} +Configuring image repository an tag under 'image' is no longer supported. +This is the list values that we no longer support: + - image.kubeEvents.registry + - image.kubeEvents.repository + - image.kubeEvents.tag + - image.kubeEvents.pullPolicy + - image.infraAgent.registry + - image.infraAgent.repository + - image.infraAgent.tag + - image.infraAgent.pullPolicy + +Please set: + - images.agent.* to configure the infrastructure-agent forwarder. + - images.integration.* to configure the image in charge of scraping k8s data. + +------ +{{- end }} +{{- end -}} + + + +{{- /* Messege to show to the user saying that image value is not supported anymore */ -}} +{{- define "nri-kube-events.compatibility.message.securityContext.runAsUser" -}} +{{- if .Values.runAsUser }} +WARNING: `runAsUser` is deprecated +================================== + +We have automatically translated your `runAsUser` setting to the new format, but this shimming will be removed in the +future. Please migrate your configs to the new format in the `securityContext` key. +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/agent-configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/agent-configmap.yaml new file mode 100644 index 0000000000..02bf8306be --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/agent-configmap.yaml @@ -0,0 +1,12 @@ +{{- if .Values.sinks.newRelicInfra -}} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }}-agent-config + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: | + {{- include "nri-kube-events.agentConfig" . | nindent 4 }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrole.yaml new file mode 100644 index 0000000000..cbfd5d9cef --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrole.yaml @@ -0,0 +1,42 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }} +rules: +- apiGroups: + - "" + resources: + - events + - namespaces + - nodes + - jobs + - persistentvolumes + - persistentvolumeclaims + - pods + - services + verbs: + - get + - watch + - list +- apiGroups: + - apps + resources: + - daemonsets + - deployments + verbs: + - get + - watch + - list +- apiGroups: + - batch + resources: + - cronjobs + - jobs + verbs: + - get + - watch + - list +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..fc5dfb8dae --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic.common.naming.fullname" . }} +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/configmap.yaml new file mode 100644 index 0000000000..9e4e35f6b3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/configmap.yaml @@ -0,0 +1,23 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.naming.fullname" . }}-config + namespace: {{ .Release.Namespace }} +data: + config.yaml: |- + sinks: + {{- if .Values.sinks.stdout }} + - name: stdout + {{- end }} + {{- if .Values.sinks.newRelicInfra }} + - name: newRelicInfra + config: + agentEndpoint: http://localhost:8001/v1/data + clusterName: {{ include "newrelic.common.cluster" . }} + agentHTTPTimeout: {{ .Values.agentHTTPTimeout }} + {{- end }} + captureDescribe: {{ .Values.scrapers.descriptions.enabled }} + describeRefresh: {{ .Values.scrapers.descriptions.resyncPeriod | default "24h" }} + captureEvents: {{ .Values.scrapers.events.enabled }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/deployment.yaml new file mode 100644 index 0000000000..7ba9eaea9c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/deployment.yaml @@ -0,0 +1,124 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + annotations: + {{- if .Values.deployment.annotations }} + {{- toYaml .Values.deployment.annotations | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + app.kubernetes.io/name: {{ include "newrelic.common.naming.name" . }} + template: + metadata: + {{- if .Values.podAnnotations }} + annotations: + {{- toYaml .Values.podAnnotations | nindent 8}} + {{- end }} + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + {{- with include "nri-kube-events.compatibility.images.renderPullSecrets" . }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + {{- with include "nri-kube-events.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: kube-events + image: {{ include "nri-kube-events.compatibility.images.integration" . }} + imagePullPolicy: {{ include "nri-kube-events.compatibility.images.pullPolicy.integration" . }} + {{- with include "nri-kube-events.securityContext.container" . }} + securityContext: + {{- . | nindent 12 }} + {{- end }} + {{- if .Values.resources }} + resources: + {{- toYaml .Values.resources | nindent 12 }} + {{- end }} + args: ["-config", "/app/config/config.yaml", "-loglevel", "debug"] + volumeMounts: + - name: config-volume + mountPath: /app/config + {{- if .Values.sinks.newRelicInfra }} + - name: forwarder + image: {{ include "nri-kube-events.compatibility.images.agent" . }} + imagePullPolicy: {{ include "nri-kube-events.compatibility.images.pullPolicy.agent" . }} + {{- with include "nri-kube-events.securityContext.container" . }} + securityContext: + {{- . | nindent 12 }} + {{- end }} + ports: + - containerPort: {{ get (fromYaml (include "nri-kube-events.agentConfig" .)) "http_server_port" }} + env: + - name: NRIA_LICENSE_KEY + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + + - name: NRIA_OVERRIDE_HOSTNAME_SHORT + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: spec.nodeName + + volumeMounts: + - mountPath: /var/db/newrelic-infra/data + name: tmpfs-data + - mountPath: /var/db/newrelic-infra/user_data + name: tmpfs-user-data + - mountPath: /tmp + name: tmpfs-tmp + - name: config + mountPath: /etc/newrelic-infra.yml + subPath: newrelic-infra.yml + {{- if ((.Values.forwarder).resources) }} + resources: + {{- toYaml .Values.forwarder.resources | nindent 12 }} + {{- end }} + {{- end }} + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + volumes: + {{- if .Values.sinks.newRelicInfra }} + - name: config + configMap: + name: {{ include "newrelic.common.naming.fullname" . }}-agent-config + items: + - key: newrelic-infra.yml + path: newrelic-infra.yml + {{- end }} + - name: config-volume + configMap: + name: {{ include "newrelic.common.naming.fullname" . }}-config + - name: tmpfs-data + emptyDir: {} + - name: tmpfs-user-data + emptyDir: {} + - name: tmpfs-tmp + emptyDir: {} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/secret.yaml new file mode 100644 index 0000000000..f558ee86c9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/secret.yaml @@ -0,0 +1,2 @@ +{{- /* Common library will take care of creating the secret or not. */}} +{{- include "newrelic.common.license.secret" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/serviceaccount.yaml new file mode 100644 index 0000000000..07e818da00 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/templates/serviceaccount.yaml @@ -0,0 +1,11 @@ +{{- if include "newrelic.common.serviceAccount.create" . }} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} + annotations: +{{ include "newrelic.common.serviceAccount.annotations" . | indent 4 }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/agent_configmap_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/agent_configmap_test.yaml new file mode 100644 index 0000000000..831b0c5aa1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/agent_configmap_test.yaml @@ -0,0 +1,46 @@ +suite: test configmap for newrelic infra agent +templates: + - templates/agent-configmap.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: has the correct default values + set: + cluster: test-cluster + licenseKey: us-whatever + asserts: + - equal: + path: data["newrelic-infra.yml"] + value: | + is_forward_only: true + http_server_enabled: true + http_server_port: 8001 + + - it: integrates properly with the common library + set: + cluster: test-cluster + licenseKey: us-whatever + fedramp.enabled: true + verboseLog: true + asserts: + - equal: + path: data["newrelic-infra.yml"] + value: | + is_forward_only: true + http_server_enabled: true + http_server_port: 8001 + + log: + level: trace + fedramp: true + + - it: does not template if the http sink is disabled + set: + cluster: test-cluster + licenseKey: us-whatever + sinks: + newRelicInfra: false + asserts: + - hasDocuments: + count: 0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/configmap_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/configmap_test.yaml new file mode 100644 index 0000000000..68ad53a579 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/configmap_test.yaml @@ -0,0 +1,139 @@ +suite: test configmap for sinks +templates: + - templates/configmap.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: has the correct sinks when default values used + set: + licenseKey: us-whatever + cluster: a-cluster + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + - name: newRelicInfra + config: + agentEndpoint: http://localhost:8001/v1/data + clusterName: a-cluster + agentHTTPTimeout: 30s + captureDescribe: true + describeRefresh: 24h + captureEvents: true + + - it: honors agentHTTPTimeout + set: + licenseKey: us-whatever + cluster: a-cluster + agentHTTPTimeout: 10s + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + - name: newRelicInfra + config: + agentEndpoint: http://localhost:8001/v1/data + clusterName: a-cluster + agentHTTPTimeout: 10s + captureDescribe: true + describeRefresh: 24h + captureEvents: true + + - it: has the correct sinks defined in local values + set: + licenseKey: us-whatever + cluster: a-cluster + sinks: + stdout: true + newRelicInfra: false + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + - name: stdout + captureDescribe: true + describeRefresh: 24h + captureEvents: true + + - it: allows enabling/disabling event scraping + set: + licenseKey: us-whatever + cluster: a-cluster + scrapers: + events: + enabled: false + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + - name: newRelicInfra + config: + agentEndpoint: http://localhost:8001/v1/data + clusterName: a-cluster + agentHTTPTimeout: 30s + captureDescribe: true + describeRefresh: 24h + captureEvents: false + + - it: allows enabling/disabling description scraping + set: + licenseKey: us-whatever + cluster: a-cluster + scrapers: + descriptions: + enabled: false + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + - name: newRelicInfra + config: + agentEndpoint: http://localhost:8001/v1/data + clusterName: a-cluster + agentHTTPTimeout: 30s + captureDescribe: false + describeRefresh: 24h + captureEvents: true + + - it: allows changing description resync intervals + set: + licenseKey: us-whatever + cluster: a-cluster + scrapers: + descriptions: + resyncPeriod: 4h + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + - name: newRelicInfra + config: + agentEndpoint: http://localhost:8001/v1/data + clusterName: a-cluster + agentHTTPTimeout: 30s + captureDescribe: true + describeRefresh: 4h + captureEvents: true + + - it: has another document generated with the proper config set + set: + licenseKey: us-whatever + cluster: a-cluster + sinks: + stdout: false + newRelicInfra: false + asserts: + - equal: + path: data["config.yaml"] + value: |- + sinks: + captureDescribe: true + describeRefresh: 24h + captureEvents: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/deployment_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/deployment_test.yaml new file mode 100644 index 0000000000..702917bcea --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/deployment_test.yaml @@ -0,0 +1,104 @@ +suite: test deployment images +templates: + - templates/deployment.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: deployment image uses pullSecrets + set: + cluster: my-cluster + licenseKey: us-whatever + images: + pullSecrets: + - name: regsecret + asserts: + - equal: + path: spec.template.spec.imagePullSecrets + value: + - name: regsecret + + - it: deployment images use the proper image tag + set: + cluster: test-cluster + licenseKey: us-whatever + images: + integration: + repository: newrelic/nri-kube-events + tag: "latest" + agent: + repository: newrelic/k8s-events-forwarder + tag: "latest" + asserts: + - matchRegex: + path: spec.template.spec.containers[0].image + pattern: .*newrelic/nri-kube-events:latest$ + - matchRegex: + path: spec.template.spec.containers[1].image + pattern: .*newrelic/k8s-events-forwarder:latest$ + + + - it: by default the agent forwarder templates + set: + cluster: test-cluster + licenseKey: us-whatever + asserts: + - contains: + path: spec.template.spec.containers + any: true + content: + name: forwarder + - contains: + path: spec.template.spec.volumes + content: + name: config + configMap: + name: my-release-nri-kube-events-agent-config + items: + - key: newrelic-infra.yml + path: newrelic-infra.yml + + - it: agent does not template if the sink is disabled + set: + cluster: test-cluster + licenseKey: us-whatever + sinks: + newRelicInfra: false + asserts: + - notContains: + path: spec.template.spec.containers + any: true + content: + name: forwarder + - notContains: + path: spec.template.spec.volumes + content: + name: config + configMap: + name: my-release-nri-kube-events-agent-config + items: + - key: newrelic-infra.yml + path: newrelic-infra.yml + + - it: has a linux node selector by default + set: + cluster: my-cluster + licenseKey: us-whatever + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + + - it: has a linux node selector and additional selectors + set: + cluster: my-cluster + licenseKey: us-whatever + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/images_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/images_test.yaml new file mode 100644 index 0000000000..361be582b5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/images_test.yaml @@ -0,0 +1,168 @@ +suite: test image compatibility layer +templates: + - templates/deployment.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: by default the tag is not nil + set: + cluster: test-cluster + licenseKey: us-whatever + asserts: + - notMatchRegex: + path: spec.template.spec.containers[0].image + pattern: ".*nil.*" + - notMatchRegex: + path: spec.template.spec.containers[1].image + pattern: ".*nil.*" + + - it: templates image correctly from the new values + set: + cluster: test-cluster + licenseKey: us-whatever + images: + integration: + registry: ireg + repository: irep + tag: itag + agent: + registry: areg + repository: arep + tag: atag + asserts: + - equal: + path: spec.template.spec.containers[0].image + value: ireg/irep:itag + - equal: + path: spec.template.spec.containers[1].image + value: areg/arep:atag + + - it: templates image correctly from old values + set: + cluster: test-cluster + licenseKey: us-whatever + image: + kubeEvents: + registry: ireg + repository: irep + tag: itag + infraAgent: + registry: areg + repository: arep + tag: atag + asserts: + - equal: + path: spec.template.spec.containers[0].image + value: ireg/irep:itag + - equal: + path: spec.template.spec.containers[1].image + value: areg/arep:atag + + - it: old image values take precedence + set: + cluster: test-cluster + licenseKey: us-whatever + images: + integration: + registry: inew + repository: inew + tag: inew + agent: + registry: anew + repository: anew + tag: anew + image: + kubeEvents: + registry: iold + repository: iold + tag: iold + infraAgent: + registry: aold + repository: aold + tag: aold + asserts: + - equal: + path: spec.template.spec.containers[0].image + value: iold/iold:iold + - equal: + path: spec.template.spec.containers[1].image + value: aold/aold:aold + + - it: pullImagePolicy templates correctly from the new values + set: + cluster: test-cluster + licenseKey: us-whatever + images: + integration: + pullPolicy: new + agent: + pullPolicy: new + asserts: + - equal: + path: spec.template.spec.containers[0].imagePullPolicy + value: new + - equal: + path: spec.template.spec.containers[1].imagePullPolicy + value: new + + - it: pullImagePolicy templates correctly from old values + set: + cluster: test-cluster + licenseKey: us-whatever + image: + kubeEvents: + pullPolicy: old + infraAgent: + pullPolicy: old + asserts: + - equal: + path: spec.template.spec.containers[0].imagePullPolicy + value: old + - equal: + path: spec.template.spec.containers[1].imagePullPolicy + value: old + + - it: old imagePullPolicy values take precedence + set: + cluster: test-cluster + licenseKey: us-whatever + images: + integration: + pullPolicy: new + agent: + pullPolicy: new + image: + kubeEvents: + pullPolicy: old + infraAgent: + pullPolicy: old + asserts: + - equal: + path: spec.template.spec.containers[0].imagePullPolicy + value: old + - equal: + path: spec.template.spec.containers[1].imagePullPolicy + value: old + + - it: imagePullSecrets merge properly + set: + cluster: test-cluster + licenseKey: us-whatever + global: + images: + pullSecrets: + - global: global + images: + pullSecrets: + - images: images + image: + pullSecrets: + - image: image + asserts: + - equal: + path: spec.template.spec.imagePullSecrets + value: + - global: global + - image: image + - images: images diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/security_context_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/security_context_test.yaml new file mode 100644 index 0000000000..b2b710331f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/tests/security_context_test.yaml @@ -0,0 +1,77 @@ +suite: test deployment security context +templates: + - templates/deployment.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: pod securityContext set to defaults when no values provided + set: + cluster: my-cluster + licenseKey: us-whatever + asserts: + - equal: + path: spec.template.spec.securityContext + value: + runAsUser: 1000 + runAsNonRoot: true + - it: pod securityContext set common-library values + set: + cluster: test-cluster + licenseKey: us-whatever + podSecurityContext: + foobar: true + asserts: + - equal: + path: spec.template.spec.securityContext.foobar + value: true + - it: pod securityContext compatibility layer overrides values from common-library + set: + cluster: test-cluster + licenseKey: us-whatever + runAsUser: 1001 + podSecurityContext: + runAsUser: 1000 + runAsNonRoot: false + asserts: + - equal: + path: spec.template.spec.securityContext + value: + runAsUser: 1001 + runAsNonRoot: false + - it: pod securityContext compatibility layer overrides defaults + set: + cluster: test-cluster + licenseKey: us-whatever + runAsUser: 1001 + asserts: + - equal: + path: spec.template.spec.securityContext.runAsUser + value: 1001 + - it: set to defaults when no containerSecurityContext set + set: + cluster: my-cluster + licenseKey: us-whatever + asserts: + - equal: + path: spec.template.spec.containers[0].securityContext + value: + allowPrivilegeEscalation: false + privileged: false + readOnlyRootFilesystem: true + - equal: + path: spec.template.spec.containers[1].securityContext + value: + allowPrivilegeEscalation: false + privileged: false + readOnlyRootFilesystem: true + - it: set containerSecurityContext custom values + set: + cluster: test-cluster + licenseKey: us-whatever + containerSecurityContext: + foobar: true + asserts: + - equal: + path: spec.template.spec.containers[0].securityContext.foobar + value: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/values.yaml new file mode 100644 index 0000000000..fc473991df --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-kube-events/values.yaml @@ -0,0 +1,135 @@ +# -- Override the name of the chart +nameOverride: "" +# -- Override the full name of the release +fullnameOverride: "" + +# -- Name of the Kubernetes cluster monitored. Mandatory. Can be configured also with `global.cluster` +cluster: "" +# -- This set this license key to use. Can be configured also with `global.licenseKey` +licenseKey: "" +# -- In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` +customSecretName: "" +# -- In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` +customSecretLicenseKey: "" + +# -- Images used by the chart for the integration and agents +# @default -- See `values.yaml` +images: + # -- Image for the New Relic Kubernetes integration + # @default -- See `values.yaml` + integration: + registry: + repository: newrelic/nri-kube-events + tag: + pullPolicy: IfNotPresent + # -- Image for the New Relic Infrastructure Agent sidecar + # @default -- See `values.yaml` + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.57.2 + pullPolicy: IfNotPresent + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + +# -- Resources for the integration container +resources: {} + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + +# -- Resources for the forwarder sidecar container +forwarder: + resources: {} + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + +rbac: + # -- Specifies whether RBAC resources should be created + create: true + +# -- Settings controlling ServiceAccount creation +# @default -- See `values.yaml` +serviceAccount: + # serviceAccount.create -- (bool) Specifies whether a ServiceAccount should be created + # @default -- `true` + create: + # If not set and create is true, a name is generated using the fullname template + name: "" + # Specify any annotations to add to the ServiceAccount + annotations: + +# -- Annotations to add to the pod. +podAnnotations: {} +deployment: + # deployment.annotations -- Annotations to add to the Deployment. + annotations: {} +# -- Additional labels for chart pods +podLabels: {} +# -- Additional labels for chart objects +labels: {} + +# -- Amount of time to wait until timeout to send metrics to the metric forwarder +agentHTTPTimeout: "30s" + +# -- Configure where will the metrics be written. Mostly for debugging purposes. +# @default -- See `values.yaml` +sinks: + # -- Enable the stdout sink to also see all events in the logs. + stdout: false + # -- The newRelicInfra sink sends all events to New Relic. + newRelicInfra: true + +# -- Configure the various kinds of scrapers that should be run. +# @default -- See `values.yaml` +scrapers: + descriptions: + enabled: true + resyncPeriod: "24h" + events: + enabled: true + +# -- Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` +priorityClassName: "" +# -- (bool) Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` +# @default -- `false` +hostNetwork: +# -- Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` +dnsConfig: {} +# -- Sets security context (at pod level). Can be configured also with `global.podSecurityContext` +podSecurityContext: {} +# -- Sets security context (at container level). Can be configured also with `global.containerSecurityContext` +containerSecurityContext: {} + +# -- Sets pod/node affinities. Can be configured also with `global.affinity` +affinity: {} +# -- Sets pod's node selector. Can be configured also with `global.nodeSelector` +nodeSelector: {} +# -- Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` +tolerations: [] + +# -- Adds extra attributes to the cluster and all the metrics emitted to the backend. Can be configured also with `global.customAttributes` +customAttributes: {} + +# -- Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port`. Can be configured also with `global.proxy` +proxy: "" + +# -- (bool) Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` +# @default -- `false` +nrStaging: +fedramp: + # -- (bool) Enables FedRAMP. Can be configured also with `global.fedramp.enabled` + # @default -- `false` + enabled: + +# -- (bool) Sets the debug logs to this integration or all integrations if it is set globally. Can be configured also with `global.verboseLog` +# @default -- `false` +verboseLog: diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/.helmignore new file mode 100644 index 0000000000..f62b5519e5 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/.helmignore @@ -0,0 +1 @@ +templates/admission-webhooks/job-patch/README.md diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.lock new file mode 100644 index 0000000000..d442841bfa --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-08-30T00:58:04.140696675Z" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.yaml new file mode 100644 index 0000000000..5cc9d4a85b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/Chart.yaml @@ -0,0 +1,25 @@ +apiVersion: v2 +appVersion: 1.29.2 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy the New Relic metadata injection webhook. +home: https://hub.docker.com/r/newrelic/k8s-metadata-injection +icon: https://newrelic.com/assets/newrelic/source/NewRelic-logo-square.svg +keywords: +- infrastructure +- newrelic +- monitoring +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +name: nri-metadata-injection +sources: +- https://github.com/newrelic/k8s-metadata-injection +- https://github.com/newrelic/k8s-metadata-injection/tree/master/charts/nri-metadata-injection +version: 4.21.2 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md new file mode 100644 index 0000000000..dd922ef13a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md @@ -0,0 +1,68 @@ +# nri-metadata-injection + +A Helm chart to deploy the New Relic metadata injection webhook. + +**Homepage:** + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-metadata-injection https://newrelic.github.io/k8s-metadata-injection +helm upgrade --install nri-metadata-injection/nri-metadata-injection -f your-custom-values.yaml +``` + +## Source Code + +* +* + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Sets pod/node affinities. Can be configured also with `global.affinity` | +| certManager.enabled | bool | `false` | Use cert manager for webhook certs | +| certManager.rootCertificateDuration | string | `"43800h"` | Sets the root certificate duration. Defaults to 43800h (5 years). | +| certManager.webhookCertificateDuration | string | `"8760h"` | Sets certificate duration. Defaults to 8760h (1 year). | +| cluster | string | `""` | Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster` | +| containerSecurityContext | object | `{}` | Sets security context (at container level). Can be configured also with `global.containerSecurityContext` | +| customTLSCertificate | bool | `false` | Use custom tls certificates for the webhook, or let the chart handle it automatically. Ref: https://docs.newrelic.com/docs/integrations/kubernetes-integration/link-your-applications/link-your-applications-kubernetes#configure-injection | +| dnsConfig | object | `{}` | Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` | +| fullnameOverride | string | `""` | Override the full name of the release | +| hostNetwork | bool | false | Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` | +| image | object | See `values.yaml` | Image for the New Relic Metadata Injector | +| image.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| injectOnlyLabeledNamespaces | bool | `false` | Enable the metadata decoration only for pods living in namespaces labeled with 'newrelic-metadata-injection=enabled'. | +| jobImage | object | See `values.yaml` | Image for creating the needed certificates of this webhook to work | +| jobImage.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| jobImage.volumeMounts | list | `[]` | Volume mounts to add to the job, you might want to mount tmp if Pod Security Policies Enforce a read-only root. | +| jobImage.volumes | list | `[]` | Volumes to add to the job container | +| labels | object | `{}` | Additional labels for chart objects. Can be configured also with `global.labels` | +| nameOverride | string | `""` | Override the name of the chart | +| nodeSelector | object | `{}` | Sets pod's node selector. Can be configured also with `global.nodeSelector` | +| podAnnotations | object | `{}` | Annotations to be added to all pods created by the integration. | +| podLabels | object | `{}` | Additional labels for chart pods. Can be configured also with `global.podLabels` | +| podSecurityContext | object | `{}` | Sets security context (at pod level). Can be configured also with `global.podSecurityContext` | +| priorityClassName | string | `""` | Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` | +| rbac.pspEnabled | bool | `false` | Whether the chart should create Pod Security Policy objects. | +| replicas | int | `1` | | +| resources | object | 100m/30M -/80M | Image for creating the needed certificates of this webhook to work | +| timeoutSeconds | int | `28` | Webhook timeout Ref: https://kubernetes.io/docs/reference/access-authn-authz/extensible-admission-controllers/#timeouts | +| tolerations | list | `[]` | Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` | + +## Maintainers + +* [juanjjaramillo](https://github.com/juanjjaramillo) +* [csongnr](https://github.com/csongnr) +* [dbudziwojskiNR](https://github.com/dbudziwojskiNR) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md.gotmpl new file mode 100644 index 0000000000..752ba8aae8 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/README.md.gotmpl @@ -0,0 +1,41 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-metadata-injection https://newrelic.github.io/k8s-metadata-injection +helm upgrade --install nri-metadata-injection/nri-metadata-injection -f your-custom-values.yaml +``` + +{{ template "chart.sourcesSection" . }} + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/ci/test-values.yaml new file mode 100644 index 0000000000..6f79dea930 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/ci/test-values.yaml @@ -0,0 +1,5 @@ +cluster: test-cluster + +image: + repository: e2e/metadata-injection + tag: test # Defaults to AppVersion diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/NOTES.txt b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/NOTES.txt new file mode 100644 index 0000000000..544124d11e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/NOTES.txt @@ -0,0 +1,23 @@ +Your deployment of the New Relic metadata injection webhook is complete. You can check on the progress of this by running the following command: + + kubectl get deployments -o wide -w --namespace {{ .Release.Namespace }} {{ template "newrelic.common.naming.fullname" . }} + +{{- if .Values.customTLSCertificate }} +You have configure the chart to use a custom tls certificate, make sure to read the 'Manage custom certificates' section of the official docs to find the instructions on how to finish setting up the webhook. + +https://docs.newrelic.com/docs/integrations/kubernetes-integration/link-your-applications/link-your-applications-kubernetes#configure-injection +{{- end }} + +To validate the injection of metadata create a dummy pod containing Busybox by running: + + kubectl create -f https://git.io/vPieo + +Check if New Relic environment variables were injected: + + kubectl exec busybox0 -- env | grep NEW_RELIC_METADATA_KUBERNETES + + NEW_RELIC_METADATA_KUBERNETES_CLUSTER_NAME=fsi + NEW_RELIC_METADATA_KUBERNETES_NODE_NAME=nodea + NEW_RELIC_METADATA_KUBERNETES_NAMESPACE_NAME=default + NEW_RELIC_METADATA_KUBERNETES_POD_NAME=busybox0 + NEW_RELIC_METADATA_KUBERNETES_CONTAINER_NAME=busybox diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/_helpers.tpl new file mode 100644 index 0000000000..54a23e9811 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/_helpers.tpl @@ -0,0 +1,72 @@ +{{/* vim: set filetype=mustache: */}} + +{{- /* Allow to change pod defaults dynamically */ -}} +{{- define "nri-metadata-injection.securityContext.pod" -}} +{{- if include "newrelic.common.securityContext.pod" . -}} +{{- include "newrelic.common.securityContext.pod" . -}} +{{- else -}} +fsGroup: 1001 +runAsUser: 1001 +runAsGroup: 1001 +{{- end -}} +{{- end -}} + +{{- /* +Naming helpers +*/ -}} + +{{- define "nri-metadata-injection.name.admission" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "admission") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.admission" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.admission.serviceAccount" -}} +{{- if include "newrelic.common.serviceAccount.create" . -}} + {{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission") }} +{{- else -}} + {{ include "newrelic.common.serviceAccount.name" . }} +{{- end -}} +{{- end -}} + +{{- define "nri-metadata-injection.name.admission-create" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "admission-create") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.admission-create" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission-create") }} +{{- end -}} + +{{- define "nri-metadata-injection.name.admission-patch" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "admission-patch") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.admission-patch" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "admission-patch") }} +{{- end -}} + +{{- define "nri-metadata-injection.name.self-signed-issuer" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "self-signed-issuer") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.self-signed-issuer" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "self-signed-issuer") }} +{{- end -}} + +{{- define "nri-metadata-injection.name.root-issuer" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "root-issuer") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.root-issuer" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "root-issuer") }} +{{- end -}} + +{{- define "nri-metadata-injection.name.webhook-cert" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.name" .) "suffix" "webhook-cert") }} +{{- end -}} + +{{- define "nri-metadata-injection.fullname.webhook-cert" -}} +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" (dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "webhook-cert") }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrole.yaml new file mode 100644 index 0000000000..275b597c8d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrole.yaml @@ -0,0 +1,27 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "nri-metadata-injection.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "newrelic.common.naming.name" $ }}-admission + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - admissionregistration.k8s.io + resources: + - mutatingwebhookconfigurations + verbs: + - get + - update +{{- if .Values.rbac.pspEnabled }} + - apiGroups: ['policy'] + resources: ['podsecuritypolicies'] + verbs: ['use'] + resourceNames: + - {{ include "nri-metadata-injection.fullname.admission" . }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrolebinding.yaml new file mode 100644 index 0000000000..cf846745ed --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "nri-metadata-injection.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + app: {{ include "nri-metadata-injection.name.admission" . }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "nri-metadata-injection.fullname.admission" . }} +subjects: + - kind: ServiceAccount + name: {{ include "nri-metadata-injection.fullname.admission.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-createSecret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-createSecret.yaml new file mode 100644 index 0000000000..a04f279353 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-createSecret.yaml @@ -0,0 +1,61 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ include "nri-metadata-injection.fullname.admission-create" . }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "nri-metadata-injection.name.admission-create" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + template: + metadata: + name: {{ include "nri-metadata-injection.fullname.admission-create" . }} + {{- if .Values.podAnnotations }} + annotations: + {{- toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + labels: + app: {{ include "nri-metadata-injection.name.admission-create" . }} + {{- include "newrelic.common.labels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" ( list .Values.jobImage.pullSecrets ) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 -}} + {{- end }} + containers: + - name: create + image: {{ include "newrelic.common.images.image" ( dict "defaultRegistry" "registry.k8s.io" "imageRoot" .Values.jobImage "context" .) }} + imagePullPolicy: {{ .Values.jobImage.pullPolicy }} + args: + - create + - --host={{ include "newrelic.common.naming.fullname" . }},{{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }}.svc + - --namespace={{ .Release.Namespace }} + - --secret-name={{ include "nri-metadata-injection.fullname.admission" . }} + - --cert-name=tls.crt + - --key-name=tls.key + {{- if .Values.jobImage.volumeMounts }} + volumeMounts: + {{- .Values.jobImage.volumeMounts | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.jobImage.volumes }} + volumes: + {{- .Values.jobImage.volumes | toYaml | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + serviceAccountName: {{ include "nri-metadata-injection.fullname.admission.serviceAccount" . }} + securityContext: + runAsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- if .Values.tolerations }} + tolerations: + {{- toYaml .Values.tolerations | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-patchWebhook.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-patchWebhook.yaml new file mode 100644 index 0000000000..99374ef355 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/job-patchWebhook.yaml @@ -0,0 +1,61 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ include "nri-metadata-injection.fullname.admission-patch" . }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "nri-metadata-injection.name.admission-patch" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + template: + metadata: + name: {{ include "nri-metadata-injection.fullname.admission-patch" . }} + {{- if .Values.podAnnotations }} + annotations: + {{- toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + labels: + app: {{ include "nri-metadata-injection.name.admission-patch" . }} + {{- include "newrelic.common.labels" . | nindent 8 }} + spec: + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" ( list .Values.jobImage.pullSecrets ) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 -}} + {{- end }} + containers: + - name: patch + image: {{ include "newrelic.common.images.image" ( dict "defaultRegistry" "registry.k8s.io" "imageRoot" .Values.jobImage "context" .) }} + imagePullPolicy: {{ .Values.jobImage.pullPolicy }} + args: + - patch + - --webhook-name={{ include "newrelic.common.naming.fullname" . }} + - --namespace={{ .Release.Namespace }} + - --secret-name={{ include "nri-metadata-injection.fullname.admission" . }} + - --patch-failure-policy=Ignore + - --patch-validating=false + {{- if .Values.jobImage.volumeMounts }} + volumeMounts: + {{- .Values.jobImage.volumeMounts | toYaml | nindent 10 }} + {{- end }} + {{- if .Values.jobImage.volumes }} + volumes: + {{- .Values.jobImage.volumes | toYaml | nindent 8 }} + {{- end }} + restartPolicy: OnFailure + serviceAccountName: {{ include "nri-metadata-injection.fullname.admission.serviceAccount" . }} + securityContext: + runAsGroup: 2000 + runAsNonRoot: true + runAsUser: 2000 + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- if .Values.tolerations }} + tolerations: + {{- toYaml .Values.tolerations | nindent 8 }} + {{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/psp.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/psp.yaml new file mode 100644 index 0000000000..899ac95fe1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/psp.yaml @@ -0,0 +1,50 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled) (.Values.rbac.pspEnabled) (.Capabilities.APIVersions.Has "policy/v1beta1/PodSecurityPolicy")) }} +apiVersion: policy/v1beta1 +kind: PodSecurityPolicy +metadata: + name: {{ include "nri-metadata-injection.fullname.admission" . }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "nri-metadata-injection.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + privileged: false + # Required to prevent escalations to root. + # allowPrivilegeEscalation: false + # This is redundant with non-root + disallow privilege escalation, + # but we can provide it for defense in depth. + #requiredDropCapabilities: + # - ALL + # Allow core volume types. + volumes: + - 'configMap' + - 'emptyDir' + - 'projected' + - 'secret' + - 'downwardAPI' + - 'persistentVolumeClaim' + hostNetwork: false + hostIPC: false + hostPID: false + 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: + # Forbid adding the root group. + - min: 0 + max: 65535 + fsGroup: + rule: 'MustRunAs' + ranges: + # Forbid adding the root group. + - min: 0 + max: 65535 + readOnlyRootFilesystem: false +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/role.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/role.yaml new file mode 100644 index 0000000000..e426702579 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/role.yaml @@ -0,0 +1,21 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "nri-metadata-injection.fullname.admission" . }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "nri-metadata-injection.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: + - apiGroups: + - "" + resources: + - secrets + verbs: + - get + - create +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/rolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/rolebinding.yaml new file mode 100644 index 0000000000..e73bf472cd --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/rolebinding.yaml @@ -0,0 +1,21 @@ +{{- if (and (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "nri-metadata-injection.fullname.admission" . }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "nri-metadata-injection.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "nri-metadata-injection.fullname.admission" . }} +subjects: + - kind: ServiceAccount + name: {{ include "nri-metadata-injection.fullname.admission.serviceAccount" . }} + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/serviceaccount.yaml new file mode 100644 index 0000000000..027a59089a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/job-patch/serviceaccount.yaml @@ -0,0 +1,14 @@ +{{- $createServiceAccount := include "newrelic.common.serviceAccount.create" . -}} +{{- if (and $createServiceAccount (not .Values.customTLSCertificate) (not .Values.certManager.enabled)) -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "nri-metadata-injection.fullname.admission.serviceAccount" . }} + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-install,post-upgrade + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded + labels: + app: {{ include "nri-metadata-injection.name.admission" . }} + {{- include "newrelic.common.labels" . | nindent 4 }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/mutatingWebhookConfiguration.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/mutatingWebhookConfiguration.yaml new file mode 100644 index 0000000000..b196d4f593 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/admission-webhooks/mutatingWebhookConfiguration.yaml @@ -0,0 +1,36 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} +{{- if .Values.certManager.enabled }} + annotations: + certmanager.k8s.io/inject-ca-from: {{ printf "%s/%s-root-cert" .Release.Namespace (include "newrelic.common.naming.fullname" .) | quote }} + cert-manager.io/inject-ca-from: {{ printf "%s/%s-root-cert" .Release.Namespace (include "newrelic.common.naming.fullname" .) | quote }} +{{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +webhooks: +- name: metadata-injection.newrelic.com + clientConfig: + service: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + path: "/mutate" +{{- if not .Values.certManager.enabled }} + caBundle: "" +{{- end }} + rules: + - operations: ["CREATE"] + apiGroups: [""] + apiVersions: ["v1"] + resources: ["pods"] +{{- if .Values.injectOnlyLabeledNamespaces }} + scope: Namespaced + namespaceSelector: + matchLabels: + newrelic-metadata-injection: enabled +{{- end }} + failurePolicy: Ignore + timeoutSeconds: {{ .Values.timeoutSeconds }} + sideEffects: None + admissionReviewVersions: ["v1", "v1beta1"] diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/cert-manager.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/cert-manager.yaml new file mode 100644 index 0000000000..502fa44bb2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/cert-manager.yaml @@ -0,0 +1,53 @@ +{{ if .Values.certManager.enabled }} +--- +# Create a selfsigned Issuer, in order to create a root CA certificate for +# signing webhook serving certificates +apiVersion: cert-manager.io/v1 +kind: Issuer +metadata: + name: {{ include "nri-metadata-injection.fullname.self-signed-issuer" . }} + namespace: {{ .Release.Namespace }} +spec: + selfSigned: {} +--- +# Generate a CA Certificate used to sign certificates for the webhook +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + name: {{ include "newrelic.common.naming.fullname" . }}-root-cert + namespace: {{ .Release.Namespace }} +spec: + secretName: {{ include "newrelic.common.naming.fullname" . }}-root-cert + duration: {{ .Values.certManager.rootCertificateDuration}} + issuerRef: + name: {{ include "nri-metadata-injection.fullname.self-signed-issuer" . }} + commonName: "ca.webhook.nri" + isCA: true +--- +# Create an Issuer that uses the above generated CA certificate to issue certs +apiVersion: cert-manager.io/v1 +kind: Issuer +metadata: + name: {{ include "nri-metadata-injection.fullname.root-issuer" . }} + namespace: {{ .Release.Namespace }} +spec: + ca: + secretName: {{ include "newrelic.common.naming.fullname" . }}-root-cert +--- + +# Finally, generate a serving certificate for the webhook to use +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + name: {{ include "nri-metadata-injection.fullname.webhook-cert" . }} + namespace: {{ .Release.Namespace }} +spec: + secretName: {{ include "nri-metadata-injection.fullname.admission" . }} + duration: {{ .Values.certManager.webhookCertificateDuration }} + issuerRef: + name: {{ include "nri-metadata-injection.fullname.root-issuer" . }} + dnsNames: + - {{ include "newrelic.common.naming.fullname" . }} + - {{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }} + - {{ include "newrelic.common.naming.fullname" . }}.{{ .Release.Namespace }}.svc +{{ end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/deployment.yaml new file mode 100644 index 0000000000..4974dbbc1c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/deployment.yaml @@ -0,0 +1,85 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + replicas: {{ .Values.replicas }} + selector: + matchLabels: + {{- /* We cannot use the common library here because of a legacy issue */}} + {{- /* `selector` is immutable and the previous chart did not have all the idiomatic labels */}} + app.kubernetes.io/name: {{ include "newrelic.common.naming.name" . }} + template: + metadata: + {{- if .Values.podAnnotations }} + annotations: + {{- toYaml .Values.podAnnotations | nindent 8 }} + {{- end }} + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + {{- with include "nri-metadata-injection.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 -}} + {{- end }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} + {{- if include "newrelic.common.hostNetwork" . }} + dnsPolicy: ClusterFirstWithHostNet + {{- end }} + + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" ( list .Values.image.pullSecrets ) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 -}} + {{- end }} + containers: + - name: {{ include "newrelic.common.naming.name" . }} + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.image "context" .) }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 10 }} + {{- end }} + env: + - name: clusterName + value: {{ include "newrelic.common.cluster" . }} + ports: + - containerPort: 8443 + protocol: TCP + volumeMounts: + - name: tls-key-cert-pair + mountPath: /etc/tls-key-cert-pair + readinessProbe: + httpGet: + path: /health + port: 8080 + initialDelaySeconds: 1 + periodSeconds: 1 + {{- if .Values.resources }} + resources: + {{ toYaml .Values.resources | nindent 10 }} + {{- end }} + volumes: + - name: tls-key-cert-pair + secret: + secretName: {{ include "nri-metadata-injection.fullname.admission" . }} + nodeSelector: + kubernetes.io/os: linux + {{ include "newrelic.common.nodeSelector" . | nindent 8 }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 -}} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 8 -}} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/service.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/service.yaml new file mode 100644 index 0000000000..e4a57587c6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/templates/service.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + ports: + - port: 443 + targetPort: 8443 + selector: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 4 }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/cluster_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/cluster_test.yaml new file mode 100644 index 0000000000..a28487a06e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/cluster_test.yaml @@ -0,0 +1,39 @@ +suite: test cluster environment variable setup +templates: + - templates/deployment.yaml +release: + name: release + namespace: ns +tests: + - it: clusterName env is properly set + set: + cluster: my-cluster + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: clusterName + value: my-cluster + - it: fail when cluster is not defined + asserts: + - failedTemplate: + errorMessage: There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required. + - it: has a linux node selector by default + set: + cluster: my-cluster + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + - it: has a linux node selector and additional selectors + set: + cluster: my-cluster + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/job_serviceaccount_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/job_serviceaccount_test.yaml new file mode 100644 index 0000000000..63b6f0534b --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/job_serviceaccount_test.yaml @@ -0,0 +1,59 @@ +suite: test job' serviceAccount +templates: + - templates/admission-webhooks/job-patch/job-createSecret.yaml + - templates/admission-webhooks/job-patch/job-patchWebhook.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: RBAC points to the service account that is created by default + set: + cluster: test-cluster + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: my-release-nri-metadata-injection-admission + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: sa-test + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: spec.template.spec.serviceAccountName + value: default + + - it: has a linux node selector by default + set: + cluster: my-cluster + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + + - it: has a linux node selector and additional selectors + set: + cluster: my-cluster + nodeSelector: + aCoolTestLabel: aCoolTestValue + asserts: + - equal: + path: spec.template.spec.nodeSelector + value: + kubernetes.io/os: linux + aCoolTestLabel: aCoolTestValue diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/rbac_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/rbac_test.yaml new file mode 100644 index 0000000000..5a69191df9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/rbac_test.yaml @@ -0,0 +1,38 @@ +suite: test RBAC creation +templates: + - templates/admission-webhooks/job-patch/rolebinding.yaml + - templates/admission-webhooks/job-patch/clusterrolebinding.yaml +release: + name: my-release + namespace: my-namespace +tests: + - it: RBAC points to the service account that is created by default + set: + cluster: test-cluster + rbac.create: true + serviceAccount.create: true + asserts: + - equal: + path: subjects[0].name + value: my-release-nri-metadata-injection-admission + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + rbac.create: true + serviceAccount.create: false + serviceAccount.name: sa-test + asserts: + - equal: + path: subjects[0].name + value: sa-test + + - it: RBAC points to the service account the user supplies when serviceAccount is disabled + set: + cluster: test-cluster + rbac.create: true + serviceAccount.create: false + asserts: + - equal: + path: subjects[0].name + value: default diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/volume_mounts_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/volume_mounts_test.yaml new file mode 100644 index 0000000000..4a3c1327dd --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/tests/volume_mounts_test.yaml @@ -0,0 +1,30 @@ +suite: check volume mounts is properly set +templates: + - templates/admission-webhooks/job-patch/job-createSecret.yaml + - templates/admission-webhooks/job-patch/job-patchWebhook.yaml +release: + name: release + namespace: ns +tests: + - it: clusterName env is properly set + set: + cluster: my-cluster + jobImage: + volumeMounts: + - name: test-volume + volumePath: /test-volume + volumes: + - name: test-volume-container + emptyDir: {} + + asserts: + - contains: + path: spec.template.spec.containers[0].volumeMounts + content: + name: test-volume + volumePath: /test-volume + - contains: + path: spec.template.spec.volumes + content: + name: test-volume-container + emptyDir: {} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/values.yaml new file mode 100644 index 0000000000..849135c357 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-metadata-injection/values.yaml @@ -0,0 +1,102 @@ +# -- Override the name of the chart +nameOverride: "" +# -- Override the full name of the release +fullnameOverride: "" + +# -- Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster` +cluster: "" + +# -- Image for the New Relic Metadata Injector +# @default -- See `values.yaml` +image: + registry: + repository: newrelic/k8s-metadata-injection + tag: "" # Defaults to chart's appVersion + pullPolicy: IfNotPresent + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + +# -- Image for creating the needed certificates of this webhook to work +# @default -- See `values.yaml` +jobImage: + registry: # Defaults to registry.k8s.io + repository: ingress-nginx/kube-webhook-certgen + tag: v1.3.0 + pullPolicy: IfNotPresent + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + + # -- Volume mounts to add to the job, you might want to mount tmp if Pod Security Policies + # Enforce a read-only root. + volumeMounts: [] + # - name: tmp + # mountPath: /tmp + + # -- Volumes to add to the job container + volumes: [] + # - name: tmp + # emptyDir: {} + +rbac: + # rbac.pspEnabled -- Whether the chart should create Pod Security Policy objects. + pspEnabled: false + +replicas: 1 + +# -- Additional labels for chart objects. Can be configured also with `global.labels` +labels: {} +# -- Annotations to be added to all pods created by the integration. +podAnnotations: {} +# -- Additional labels for chart pods. Can be configured also with `global.podLabels` +podLabels: {} + +# -- Image for creating the needed certificates of this webhook to work +# @default -- 100m/30M -/80M +resources: + limits: + memory: 80M + requests: + cpu: 100m + memory: 30M + +# -- Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` +priorityClassName: "" +# -- (bool) Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` +# @default -- false +hostNetwork: +# -- Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` +dnsConfig: {} +# -- Sets security context (at pod level). Can be configured also with `global.podSecurityContext` +podSecurityContext: {} +# -- Sets security context (at container level). Can be configured also with `global.containerSecurityContext` +containerSecurityContext: {} + +certManager: + # certManager.enabled -- Use cert manager for webhook certs + enabled: false + # -- Sets the root certificate duration. Defaults to 43800h (5 years). + rootCertificateDuration: 43800h + # -- Sets certificate duration. Defaults to 8760h (1 year). + webhookCertificateDuration: 8760h + +# -- Sets pod/node affinities. Can be configured also with `global.affinity` +affinity: {} +# -- Sets pod's node selector. Can be configured also with `global.nodeSelector` +nodeSelector: {} +# -- Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` +tolerations: [] + +# -- Enable the metadata decoration only for pods living in namespaces labeled +# with 'newrelic-metadata-injection=enabled'. +injectOnlyLabeledNamespaces: false + +# -- Use custom tls certificates for the webhook, or let the chart handle it +# automatically. +# Ref: https://docs.newrelic.com/docs/integrations/kubernetes-integration/link-your-applications/link-your-applications-kubernetes#configure-injection +customTLSCertificate: false + +# -- Webhook timeout +# Ref: https://kubernetes.io/docs/reference/access-authn-authz/extensible-admission-controllers/#timeouts +timeoutSeconds: 28 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/.helmignore new file mode 100644 index 0000000000..50af031725 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/.helmignore @@ -0,0 +1,22 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.lock b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.lock new file mode 100644 index 0000000000..934f6dcbcc --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +digest: sha256:2e1da613fd8a52706bde45af077779c5d69e9e1641bdf5c982eaf6d1ac67a443 +generated: "2024-09-30T16:12:25.200183-07:00" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.yaml new file mode 100644 index 0000000000..d33f07729a --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/Chart.yaml @@ -0,0 +1,29 @@ +apiVersion: v2 +appVersion: 2.21.4 +dependencies: +- name: common-library + repository: https://helm-charts.newrelic.com + version: 1.3.0 +description: A Helm chart to deploy the New Relic Prometheus OpenMetrics integration +home: https://docs.newrelic.com/docs/infrastructure/prometheus-integrations/install-configure-openmetrics/configure-prometheus-openmetrics-integrations/ +icon: https://newrelic.com/themes/custom/curio/assets/mediakit/new_relic_logo_vertical.svg +keywords: +- prometheus +- newrelic +- monitoring +maintainers: +- name: alvarocabanas + url: https://github.com/alvarocabanas +- name: sigilioso + url: https://github.com/sigilioso +- name: gsanchezgavier + url: https://github.com/gsanchezgavier +- name: kang-makes + url: https://github.com/kang-makes +- name: paologallinaharbur + url: https://github.com/paologallinaharbur +name: nri-prometheus +sources: +- https://github.com/newrelic/nri-prometheus +- https://github.com/newrelic/nri-prometheus/tree/main/charts/nri-prometheus +version: 2.1.19 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md new file mode 100644 index 0000000000..0287b2b2a2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md @@ -0,0 +1,116 @@ +# nri-prometheus + +A Helm chart to deploy the New Relic Prometheus OpenMetrics integration + +**Homepage:** + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-prometheus https://newrelic.github.io/nri-prometheus +helm upgrade --install newrelic-prometheus nri-prometheus/nri-prometheus -f your-custom-values.yaml +``` + +## Source Code + +* +* + +## Scraping services and endpoints + +When a service is labeled or annotated with `scrape_enabled_label` (defaults to `prometheus.io/scrape`), +`nri-prometheus` will attempt to hit the service directly, rather than the endpoints behind it. + +This is the default behavior for compatibility reasons, but is known to cause issues if more than one endpoint +is behind the service, as metric queries will be load-balanced as well leading to inaccurate histograms. + +In order to change this behaviour set `scrape_endpoints` to `true` and `scrape_services` to `false`. +This will instruct `nri-prometheus` to scrape the underlying endpoints, as Prometheus server does. + +Existing users that are switching to this behavior should note that, depending on the number of endpoints +behind the services in the cluster the load and the metrics reported by those, data ingestion might see +an increase when flipping this option. Resource requirements might also be impacted, again depending on the number of new targets. + +While it is technically possible to set both `scrape_services` and `scrape_endpoints` to true, we do no recommend +doing so as it will lead to redundant metrics being processed, + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Chart particularities + +### Low data mode +See this snippet from the `values.yaml` file: +```yaml +global: + lowDataMode: false +lowDataMode: false +``` + +To reduce the amount ot metrics we send to New Relic, enabling the `lowDataMode` will add [these transformations](static/lowdatamodedefaults.yaml): +```yaml +transformations: + - description: "Low data mode defaults" + ignore_metrics: + # Ignore the following metrics. + # These metrics are already collected by the New Relic Kubernetes Integration. + - prefixes: + - kube_ + - container_ + - machine_ + - cadvisor_ +``` + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Sets pod/node affinities. Can be configured also with `global.affinity` | +| cluster | string | `""` | Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster` | +| config | object | See `values.yaml` | Provides your own `config.yaml` for this integration. Ref: https://docs.newrelic.com/docs/infrastructure/prometheus-integrations/install-configure-openmetrics/configure-prometheus-openmetrics-integrations/#example-configuration-file | +| containerSecurityContext | object | `{}` | Sets security context (at container level). Can be configured also with `global.containerSecurityContext` | +| customSecretLicenseKey | string | `""` | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` | +| customSecretName | string | `""` | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` | +| dnsConfig | object | `{}` | Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` | +| fedramp.enabled | bool | false | Enables FedRAMP. Can be configured also with `global.fedramp.enabled` | +| fullnameOverride | string | `""` | Override the full name of the release | +| hostNetwork | bool | `false` | Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` | +| image | object | See `values.yaml` | Image for the New Relic Kubernetes integration | +| image.pullSecrets | list | `[]` | The secrets that are needed to pull images from a custom registry. | +| labels | object | `{}` | Additional labels for chart objects. Can be configured also with `global.labels` | +| licenseKey | string | `""` | This set this license key to use. Can be configured also with `global.licenseKey` | +| lowDataMode | bool | false | Reduces number of metrics sent in order to reduce costs. Can be configured also with `global.lowDataMode` | +| nameOverride | string | `""` | Override the name of the chart | +| nodeSelector | object | `{}` | Sets pod's node selector. Can be configured also with `global.nodeSelector` | +| nrStaging | bool | false | Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` | +| podAnnotations | object | `{}` | Annotations to be added to all pods created by the integration. | +| podLabels | object | `{}` | Additional labels for chart pods. Can be configured also with `global.podLabels` | +| podSecurityContext | object | `{}` | Sets security context (at pod level). Can be configured also with `global.podSecurityContext` | +| priorityClassName | string | `""` | Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` | +| proxy | string | `""` | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port`. Can be configured also with `global.proxy` | +| rbac.create | bool | `true` | Specifies whether RBAC resources should be created | +| resources | object | `{}` | | +| serviceAccount.annotations | object | `{}` | Add these annotations to the service account we create. Can be configured also with `global.serviceAccount.annotations` | +| serviceAccount.create | bool | `true` | Configures if the service account should be created or not. Can be configured also with `global.serviceAccount.create` | +| serviceAccount.name | string | `nil` | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. Can be configured also with `global.serviceAccount.name` | +| tolerations | list | `[]` | Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` | +| verboseLog | bool | false | Sets the debug logs to this integration or all integrations if it is set globally. Can be configured also with `global.verboseLog` | + +## Maintainers + +* [alvarocabanas](https://github.com/alvarocabanas) +* [carlossscastro](https://github.com/carlossscastro) +* [sigilioso](https://github.com/sigilioso) +* [gsanchezgavier](https://github.com/gsanchezgavier) +* [kang-makes](https://github.com/kang-makes) +* [marcsanmi](https://github.com/marcsanmi) +* [paologallinaharbur](https://github.com/paologallinaharbur) +* [roobre](https://github.com/roobre) diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md.gotmpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md.gotmpl new file mode 100644 index 0000000000..5c1da45775 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/README.md.gotmpl @@ -0,0 +1,83 @@ +{{ template "chart.header" . }} +{{ template "chart.deprecationWarning" . }} + +{{ template "chart.description" . }} + +{{ template "chart.homepageLine" . }} + +# Helm installation + +You can install this chart using [`nri-bundle`](https://github.com/newrelic/helm-charts/tree/master/charts/nri-bundle) located in the +[helm-charts repository](https://github.com/newrelic/helm-charts) or directly from this repository by adding this Helm repository: + +```shell +helm repo add nri-prometheus https://newrelic.github.io/nri-prometheus +helm upgrade --install newrelic-prometheus nri-prometheus/nri-prometheus -f your-custom-values.yaml +``` + +{{ template "chart.sourcesSection" . }} + +## Scraping services and endpoints + +When a service is labeled or annotated with `scrape_enabled_label` (defaults to `prometheus.io/scrape`), +`nri-prometheus` will attempt to hit the service directly, rather than the endpoints behind it. + +This is the default behavior for compatibility reasons, but is known to cause issues if more than one endpoint +is behind the service, as metric queries will be load-balanced as well leading to inaccurate histograms. + +In order to change this behaviour set `scrape_endpoints` to `true` and `scrape_services` to `false`. +This will instruct `nri-prometheus` to scrape the underlying endpoints, as Prometheus server does. + +Existing users that are switching to this behavior should note that, depending on the number of endpoints +behind the services in the cluster the load and the metrics reported by those, data ingestion might see +an increase when flipping this option. Resource requirements might also be impacted, again depending on the number of new targets. + +While it is technically possible to set both `scrape_services` and `scrape_endpoints` to true, we do no recommend +doing so as it will lead to redundant metrics being processed, + +## Values managed globally + +This chart implements the [New Relic's common Helm library](https://github.com/newrelic/helm-charts/tree/master/library/common-library) which +means that it honors a wide range of defaults and globals common to most New Relic Helm charts. + +Options that can be defined globally include `affinity`, `nodeSelector`, `tolerations`, `proxy` and others. The full list can be found at +[user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md). + +## Chart particularities + +### Low data mode +See this snippet from the `values.yaml` file: +```yaml +global: + lowDataMode: false +lowDataMode: false +``` + +To reduce the amount ot metrics we send to New Relic, enabling the `lowDataMode` will add [these transformations](static/lowdatamodedefaults.yaml): +```yaml +transformations: + - description: "Low data mode defaults" + ignore_metrics: + # Ignore the following metrics. + # These metrics are already collected by the New Relic Kubernetes Integration. + - prefixes: + - kube_ + - container_ + - machine_ + - cadvisor_ +``` + +{{ template "chart.valuesSection" . }} + +{{ if .Maintainers }} +## Maintainers +{{ range .Maintainers }} +{{- if .Name }} +{{- if .Url }} +* [{{ .Name }}]({{ .Url }}) +{{- else }} +* {{ .Name }} +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/.helmignore b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/.helmignore new file mode 100644 index 0000000000..0e8a0eb36f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/.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/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/Chart.yaml new file mode 100644 index 0000000000..f2ee5497ee --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/Chart.yaml @@ -0,0 +1,17 @@ +apiVersion: v2 +description: Provides helpers to provide consistency on all the charts +keywords: +- newrelic +- chart-library +maintainers: +- name: juanjjaramillo + url: https://github.com/juanjjaramillo +- name: csongnr + url: https://github.com/csongnr +- name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR +- name: kang-makes + url: https://github.com/kang-makes +name: common-library +type: library +version: 1.3.0 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/DEVELOPERS.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/DEVELOPERS.md new file mode 100644 index 0000000000..7208c673ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/DEVELOPERS.md @@ -0,0 +1,747 @@ +# Functions/templates documented for chart writers +Here is some rough documentation separated by the file that contains the function, the function +name and how to use it. We are not covering functions that start with `_` (e.g. +`newrelic.common.license._licenseKey`) because they are used internally by this library for +other helpers. Helm does not have the concept of "public" or "private" functions/templates so +this is a convention of ours. + +## _naming.tpl +These functions are used to name objects. + +### `newrelic.common.naming.name` +This is the same as the idiomatic `CHART-NAME.name` that is created when you use `helm create`. + +It honors `.Values.nameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.name" . }} +``` + +### `newrelic.common.naming.fullname` +This is the same as the idiomatic `CHART-NAME.fullname` that is created when you use `helm create` + +It honors `.Values.fullnameOverride`. + +Usage: +```mustache +{{ include "newrelic.common.naming.fullname" . }} +``` + +### `newrelic.common.naming.chart` +This is the same as the idiomatic `CHART-NAME.chart` that is created when you use `helm create`. + +It is mostly useless for chart writers. It is used internally for templating the labels but there +is no reason to keep it "private". + +Usage: +```mustache +{{ include "newrelic.common.naming.chart" . }} +``` + +### `newrelic.common.naming.truncateToDNS` +This is a useful template that could be used to trim a string to 63 chars and does not end with a dash (`-`). +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $truncatedName := include "newrelic.common.naming.truncateToDNS" $nameToTruncate }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-string-that-should-be */ -}} +``` + +### `newrelic.common.naming.truncateToDNSWithSuffix` +This template function is the same as the above but instead of receiving a string you should give a `dict` +with a `name` and a `suffix`. This function will join them with a dash (`-`) and trim the `name` so the +result of `name-suffix` is no more than 63 chars + +Usage: +```mustache +{{ $nameToTruncate := "a-really-really-really-really-REALLY-long-string-that-should-be-truncated-because-it-is-enought-long-to-brak-something" +{{- $suffix := "A-NOT-SO-LONG-SUFFIX" }} +{{- $truncatedName := include "truncateToDNSWithSuffix" (dict "name" $nameToTruncate "suffix" $suffix) }} +{{- $truncatedName }} +{{- /* This should print: a-really-really-really-really-REALLY-long-A-NOT-SO-LONG-SUFFIX */ -}} +``` + + + +## _labels.tpl +### `newrelic.common.labels`, `newrelic.common.labels.selectorLabels` and `newrelic.common.labels.podLabels` +These are functions that are used to label objects. They are configured by this `values.yaml` +```yaml +global: + podLabels: {} # included in all the pods of all the charts that implement this library + labels: {} # included in all the objects of all the charts that implement this library +podLabels: {} # included in all the pods of this chart +labels: {} # included in all the objects of this chart +``` + +label maps are merged from global to local values. + +And chart writer should use them like this: +```mustache +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + selector: + matchLabels: + {{- include "newrelic.common.labels.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} +``` + +`newrelic.common.labels.podLabels` includes `newrelic.common.labels.selectorLabels` automatically. + + + +## _priority-class-name.tpl +### `newrelic.common.priorityClassName` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + priorityClassName: "" +priorityClassName: "" +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `priorityClassName` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} +``` + + + +## _hostnetwork.tpl +### `newrelic.common.hostNetwork` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + hostNetwork: # Note that this is empty (nil) +hostNetwork: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `hostNetwork` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.hostNetwork" . }} + hostNetwork: {{ . }} + {{- end }} +``` + +### `newrelic.common.hostNetwork.value` +This function is an abstraction of the function above but this returns directly "true" or "false". + +Be careful with using this with an `if` as Helm does evaluate "false" (string) as `true`. + +Usage (example in a pod spec): +```mustache +spec: + hostNetwork: {{ include "newrelic.common.hostNetwork.value" . }} +``` + + + +## _dnsconfig.tpl +### `newrelic.common.dnsConfig` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + dnsConfig: {} +dnsConfig: {} +``` + +Be careful: chart writers should put an empty string (or any kind of Helm falsiness) for this +library to work properly. If in your values a non-falsy `dnsConfig` is found, the global +one is going to be always ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _images.tpl +These functions help us to deal with how images are templated. This allows setting `registries` +where to fetch images globally while being flexible enough to fit in different maps of images +and deployments with one or more images. This is the example of a complex `values.yaml` that +we are going to use during the documentation of these functions: + +```yaml +global: + images: + registry: nexus-3-instance.internal.clients-domain.tld +jobImage: + registry: # defaults to "example.tld" when empty in these examples + repository: ingress-nginx/kube-webhook-certgen + tag: v1.1.1 + pullPolicy: IfNotPresent + pullSecrets: [] +images: + integration: + registry: + repository: newrelic/nri-kube-events + tag: 1.8.0 + pullPolicy: IfNotPresent + agent: + registry: + repository: newrelic/k8s-events-forwarder + tag: 1.22.0 + pullPolicy: IfNotPresent + pullSecrets: [] +``` + +### `newrelic.common.images.image` +This will return a string with the image ready to be downloaded that includes the registry, the image and the tag. +`defaultRegistry` is used to keep `registry` field empty in `values.yaml` so you can override the image using +`global.images.registry`, your local `jobImage.registry` and be able to fallback to a registry that is not `docker.io` +(Or the default repository that the client could have set in the CRI). + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.image" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.registry` +It returns the registry from the global or local values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For the integration */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.images.agent "context" .) }} +{{- /* For jobImage */}} +{{ include "newrelic.common.images.registry" ( dict "defaultRegistry" "example.tld" "imageRoot" .Values.jobImage "context" .) }} +``` + +### `newrelic.common.images.repository` +It returns the image from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.repository" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.tag` +It returns the image's tag from the values. You should avoid using this helper to create your image +URL and use `newrelic.common.images.image` instead, but it is there to be used in case it is needed. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.jobImage "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.integration "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.images.agent "context" .) }} +``` + +### `newrelic.common.images.renderPullSecrets` +If returns a merged map that contains the pull secrets from the global configuration and the local one. + +Usage: +```mustache +{{- /* For jobImage */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.jobImage.pullSecrets "context" .) }} +{{- /* For the integration */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +{{- /* For the agent */}} +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" .Values.images.pullSecrets "context" .) }} +``` + + + +## _serviceaccount.tpl +These functions are used to evaluate if the service account should be created, with which name and add annotations to it. + +The functions that the common library has implemented for service accounts are: +* `newrelic.common.serviceAccount.create` +* `newrelic.common.serviceAccount.name` +* `newrelic.common.serviceAccount.annotations` + +Usage: +```mustache +{{- if include "newrelic.common.serviceAccount.create" . -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end }} +``` + + + +## _affinity.tpl, _nodeselector.tpl and _tolerations.tpl +These three files are almost the same and they follow the idiomatic way of `helm create`. + +Each function also looks if there is a global value like the other helpers. +```yaml +global: + affinity: {} + nodeSelector: {} + tolerations: [] +affinity: {} +nodeSelector: {} +tolerations: [] +``` + +The values here are replaced instead of be merged. If a value at root level is found, the global one is ignored. + +Usage (example in a pod spec): +```mustache +spec: + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 4 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 4 }} + {{- end }} +``` + + + +## _agent-config.tpl +### `newrelic.common.agentConfig.defaults` +This returns a YAML that the agent can use directly as a config that includes other options from the values file like verbose mode, +custom attributes, FedRAMP and such. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + name: {{ include newrelic.common.naming.truncateToDNSWithSuffix (dict "name" (include "newrelic.common.naming.fullname" .) suffix "agent-config") }} + namespace: {{ .Release.Namespace }} +data: + newrelic-infra.yml: |- + # This is the configuration file for the infrastructure agent. See: + # https://docs.newrelic.com/docs/infrastructure/install-infrastructure-agent/configuration/infrastructure-agent-configuration-settings/ + {{- include "newrelic.common.agentConfig.defaults" . | nindent 4 }} +``` + + + +## _cluster.tpl +### `newrelic.common.cluster` +Returns the cluster name + +Usage: +```mustache +{{ include "newrelic.common.cluster" . }} +``` + + + +## _custom-attributes.tpl +### `newrelic.common.customAttributes` +Return custom attributes in YAML format. + +Usage: +```mustache +apiVersion: v1 +kind: ConfigMap +metadata: + name: example +data: + custom-attributes.yaml: | + {{- include "newrelic.common.customAttributes" . | nindent 4 }} + custom-attributes.json: | + {{- include "newrelic.common.customAttributes" . | fromYaml | toJson | nindent 4 }} +``` + + + +## _fedramp.tpl +### `newrelic.common.fedramp.enabled` +Returns true if FedRAMP is enabled or an empty string if not. It can be safely used in conditionals as an empty string is a Helm falsiness. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled" . }} +``` + +### `newrelic.common.fedramp.enabled.value` +Returns true if FedRAMP is enabled or false if not. This is to have the value of FedRAMP ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.fedramp.enabled.value" . }} +``` + + + +## _license.tpl +### `newrelic.common.license.secretName` and ### `newrelic.common.license.secretKeyName` +Returns the secret and key inside the secret where to read the license key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the license key. + +To create the secret use `newrelic.common.license.secret`. + +Usage: +```mustache +{{- if and (.Values.controlPlane.enabled) (not (include "newrelic.fargate" .)) }} +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + containers: + - name: agent + env: + - name: "NRIA_LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} +``` + + + +## _license_secret.tpl +### `newrelic.common.license.secret` +This function templates the secret that is used by agents and integrations with the license Key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any license key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.license.secret" . -}} +``` + + + +## _insights.tpl +### `newrelic.common.insightsKey.secretName` and ### `newrelic.common.insightsKey.secretKeyName` +Returns the secret and key inside the secret where to read the insights key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.insightsKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "INSIGHTS_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + key: {{ include "newrelic.common.insightsKey.secretKeyName" . }} +``` + + + +## _insights_secret.tpl +### `newrelic.common.insightsKey.secret` +This function templates the secret that is used by agents and integrations with the insights key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any insights key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.insightsKey.secret" . -}} +``` + + + +## _userkey.tpl +### `newrelic.common.userKey.secretName` and ### `newrelic.common.userKey.secretKeyName` +Returns the secret and key inside the secret where to read a user key. + +The common library will take care of using a user-provided custom secret or creating a secret that contains the insights key. + +To create the secret use `newrelic.common.userKey.secret`. + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: statsd +spec: + containers: + - name: statsd + env: + - name: "API_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.userKey.secretName" . }} + key: {{ include "newrelic.common.userKey.secretKeyName" . }} +``` + + + +## _userkey_secret.tpl +### `newrelic.common.userKey.secret` +This function templates the secret that is used by agents and integrations with a user key provided by the user. It will +template nothing (empty string) if the user provides a custom pair of secret name and key. + +This template also fails in case the user has not provided any API key or custom secret so no safety checks have to be done +by chart writers. + +You just must have a template with these two lines: +```mustache +{{- /* Common library will take care of creating the secret or not. */ -}} +{{- include "newrelic.common.userKey.secret" . -}} +``` + + + +## _region.tpl +### `newrelic.common.region.validate` +Given a string, return a normalized name for the region if valid. + +This function does not need the context of the chart, only the value to be validated. The region returned +honors the region [definition of the newrelic-client-go implementation](https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21) +so (as of 2024/09/14) it returns the region as "US", "EU", "Staging", or "Local". + +In case the region provided does not match these 4, the helper calls `fail` and abort the templating. + +Usage: +```mustache +{{ include "newrelic.common.region.validate" "us" }} +``` + +### `newrelic.common.region` +It reads global and local variables for `region`: +```yaml +global: + region: # Note that this can be empty (nil) or "" (empty string) +region: # Note that this can be empty (nil) or "" (empty string) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in your +values a `region` is defined, the global one is going to be always ignored. + +This function gives protection so it enforces users to give the license key as a value in their +`values.yaml` or specify a global or local `region` value. To understand how the `region` value +works, read the documentation of `newrelic.common.region.validate`. + +The function will change the region from US, EU or Staging based of the license key and the +`nrStaging` toggle. Whichever region is computed from the license/toggle can be overridden by +the `region` value. + +Usage: +```mustache +{{ include "newrelic.common.region" . }} +``` + + + +## _low-data-mode.tpl +### `newrelic.common.lowDataMode` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + lowDataMode: # Note that this is empty (nil) +lowDataMode: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `lowdataMode` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.lowDataMode" . }} +``` + + + +## _privileged.tpl +### `newrelic.common.privileged` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + privileged: # Note that this is empty (nil) +privileged: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `privileged` is defined, the global one is going to be always ignored. + +Chart writers could override this and put directly a `true` in the `values.yaml` to override the +default of the common library. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.privileged" . }} +``` + +### `newrelic.common.privileged.value` +Returns true if privileged mode is enabled or false if not. This is to have the value of privileged ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.privileged.value" . }} +``` + + + +## _proxy.tpl +### `newrelic.common.proxy` +Returns the proxy URL configured by the user. + +Usage: +```mustache +{{ include "newrelic.common.proxy" . }} +``` + + + +## _security-context.tpl +Use these functions to share the security context among all charts. Useful in clusters that have security enforcing not to +use the root user (like OpenShift) or users that have an admission webhooks. + +The functions are: +* `newrelic.common.securityContext.container` +* `newrelic.common.securityContext.pod` + +Usage: +```mustache +apiVersion: v1 +kind: Pod +metadata: + name: example +spec: + spec: + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + + containers: + - name: example + {{- with include "nriKubernetes.securityContext.container" . }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} +``` + + + +## _staging.tpl +### `newrelic.common.nrStaging` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + nrStaging: # Note that this is empty (nil) +nrStaging: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `nrStaging` is defined, the global one is going to be always ignored. + +This function returns "true" of "" (empty string) so it can be used for evaluating conditionals. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging" . }} +``` + +### `newrelic.common.nrStaging.value` +Returns true if staging is enabled or false if not. This is to have the staging value ready to be templated. + +Usage: +```mustache +{{ include "newrelic.common.nrStaging.value" . }} +``` + + + +## _verbose-log.tpl +### `newrelic.common.verboseLog` +Like almost everything in this library, it reads global and local variables: +```yaml +global: + verboseLog: # Note that this is empty (nil) +verboseLog: # Note that this is empty (nil) +``` + +Be careful: chart writers should NOT PUT ANY VALUE for this library to work properly. If in you +values a `verboseLog` is defined, the global one is going to be always ignored. + +Usage: +```mustache +{{ include "newrelic.common.verboseLog" . }} +``` + +### `newrelic.common.verboseLog.valueAsBoolean` +Returns true if verbose is enabled or false if not. This is to have the verbose value ready to be templated as a boolean + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsBoolean" . }} +``` + +### `newrelic.common.verboseLog.valueAsInt` +Returns 1 if verbose is enabled or 0 if not. This is to have the verbose value ready to be templated as an integer + +Usage: +```mustache +{{ include "newrelic.common.verboseLog.valueAsInt" . }} +``` diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/README.md b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/README.md new file mode 100644 index 0000000000..10f08ca677 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/README.md @@ -0,0 +1,106 @@ +# Helm Common library + +The common library is a way to unify the UX through all the Helm charts that implement it. + +The tooling suite that New Relic is huge and growing and this allows to set things globally +and locally for a single chart. + +## Documentation for chart writers + +If you are writing a chart that is going to use this library you can check the [developers guide](/library/common-library/DEVELOPERS.md) to see all +the functions/templates that we have implemented, what they do and how to use them. + +## Values managed globally + +We want to have a seamless experience through all the charts so we created this library that tries to standardize the behaviour +of all the charts. Sadly, because of the complexity of all these integrations, not all the charts behave exactly as expected. + +An example is `newrelic-infrastructure` that ignores `hostNetwork` in the control plane scraper because most of the users has the +control plane listening in the node to `localhost`. + +For each chart that has a special behavior (or further information of the behavior) there is a "chart particularities" section +in its README.md that explains which is the expected behavior. + +At the time of writing this, all the charts from `nri-bundle` except `newrelic-logging` and `synthetics-minion` implements this +library and honors global options as described in this document. + +Here is a list of global options: + +| Global keys | Local keys | Default | Merged[1](#values-managed-globally-1) | Description | +|-------------|------------|---------|--------------------------------------------------|-------------| +| global.cluster | cluster | `""` | | Name of the Kubernetes cluster monitored | +| global.licenseKey | licenseKey | `""` | | This set this license key to use | +| global.customSecretName | customSecretName | `""` | | In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there | +| global.customSecretLicenseKey | customSecretLicenseKey | `""` | | In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located | +| global.podLabels | podLabels | `{}` | yes | Additional labels for chart pods | +| global.labels | labels | `{}` | yes | Additional labels for chart objects | +| global.priorityClassName | priorityClassName | `""` | | Sets pod's priorityClassName | +| global.hostNetwork | hostNetwork | `false` | | Sets pod's hostNetwork | +| global.dnsConfig | dnsConfig | `{}` | | Sets pod's dnsConfig | +| global.images.registry | See [Further information](#values-managed-globally-2) | `""` | | Changes the registry where to get the images. Useful when there is an internal image cache/proxy | +| global.images.pullSecrets | See [Further information](#values-managed-globally-2) | `[]` | yes | Set secrets to be able to fetch images | +| global.podSecurityContext | podSecurityContext | `{}` | | Sets security context (at pod level) | +| global.containerSecurityContext | containerSecurityContext | `{}` | | Sets security context (at container level) | +| global.affinity | affinity | `{}` | | Sets pod/node affinities | +| global.nodeSelector | nodeSelector | `{}` | | Sets pod's node selector | +| global.tolerations | tolerations | `[]` | | Sets pod's tolerations to node taints | +| global.serviceAccount.create | serviceAccount.create | `true` | | Configures if the service account should be created or not | +| global.serviceAccount.name | serviceAccount.name | name of the release | | Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. | +| global.serviceAccount.annotations | serviceAccount.annotations | `{}` | yes | Add these annotations to the service account we create | +| global.customAttributes | customAttributes | `{}` | | Adds extra attributes to the cluster and all the metrics emitted to the backend | +| global.fedramp | fedramp | `false` | | Enables FedRAMP | +| global.lowDataMode | lowDataMode | `false` | | Reduces number of metrics sent in order to reduce costs | +| global.privileged | privileged | Depends on the chart | | In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | +| global.proxy | proxy | `""` | | Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` | +| global.nrStaging | nrStaging | `false` | | Send the metrics to the staging backend. Requires a valid staging license key | +| global.verboseLog | verboseLog | `false` | | Sets the debug/trace logs to this integration or all integrations if it is set globally | + +### Further information + +#### 1. Merged + +Merged means that the values from global are not replaced by the local ones. Think in this example: +```yaml +global: + labels: + global: global + hostNetwork: true + nodeSelector: + global: global + +labels: + local: local +nodeSelector: + local: local +hostNetwork: false +``` + +This values will template `hostNetwork` to `false`, a map of labels `{ "global": "global", "local": "local" }` and a `nodeSelector` with +`{ "local": "local" }`. + +As Helm by default merges all the maps it could be confusing that we have two behaviors (merging `labels` and replacing `nodeSelector`) +the `values` from global to local. This is the rationale behind this: +* `hostNetwork` is templated to `false` because is overriding the value defined globally. +* `labels` are merged because the user may want to label all the New Relic pods at once and label other solution pods differently for + clarity' sake. +* `nodeSelector` does not merge as `labels` because could make it harder to overwrite/delete a selector that comes from global because + of the logic that Helm follows merging maps. + + +#### 2. Fine grain registries + +Some charts only have 1 image while others that can have 2 or more images. The local path for the registry can change depending +on the chart itself. + +As this is mostly unique per helm chart, you should take a look to the chart's values table (or directly to the `values.yaml` file to see all the +images that you can change. + +This should only be needed if you have an advanced setup that forces you to have granularity enough to force a proxy/cache registry per integration. + + + +#### 3. Privileged mode + +By default, from the common library, the privileged mode is set to false. But most of the helm charts require this to be true to fetch more +metrics so could see a true in some charts. The consequences of the privileged mode differ from one chart to another so for each chart that +honors the privileged mode toggle should be a section in the README explaining which is the behavior with it enabled or disabled. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_affinity.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_affinity.tpl new file mode 100644 index 0000000000..1b2636754e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_affinity.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod affinity */ -}} +{{- define "newrelic.common.affinity" -}} + {{- if .Values.affinity -}} + {{- toYaml .Values.affinity -}} + {{- else if .Values.global -}} + {{- if .Values.global.affinity -}} + {{- toYaml .Values.global.affinity -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_agent-config.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_agent-config.tpl new file mode 100644 index 0000000000..9c32861a02 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_agent-config.tpl @@ -0,0 +1,26 @@ +{{/* +This helper should return the defaults that all agents should have +*/}} +{{- define "newrelic.common.agentConfig.defaults" -}} +{{- if include "newrelic.common.verboseLog" . }} +log: + level: trace +{{- end }} + +{{- if (include "newrelic.common.nrStaging" . ) }} +staging: true +{{- end }} + +{{- with include "newrelic.common.proxy" . }} +proxy: {{ . | quote }} +{{- end }} + +{{- with include "newrelic.common.fedramp.enabled" . }} +fedramp: {{ . }} +{{- end }} + +{{- with fromYaml ( include "newrelic.common.customAttributes" . ) }} +custom_attributes: + {{- toYaml . | nindent 2 }} +{{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_cluster.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_cluster.tpl new file mode 100644 index 0000000000..0197dd35a3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_cluster.tpl @@ -0,0 +1,15 @@ +{{/* +Return the cluster +*/}} +{{- define "newrelic.common.cluster" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.cluster -}} + {{- .Values.cluster -}} +{{- else if $global.cluster -}} + {{- $global.cluster -}} +{{- else -}} + {{ fail "There is not cluster name definition set neither in `.global.cluster' nor `.cluster' in your values.yaml. Cluster name is required." }} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_custom-attributes.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_custom-attributes.tpl new file mode 100644 index 0000000000..92020719c3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_custom-attributes.tpl @@ -0,0 +1,17 @@ +{{/* +This will render custom attributes as a YAML ready to be templated or be used with `fromYaml`. +*/}} +{{- define "newrelic.common.customAttributes" -}} +{{- $customAttributes := dict -}} + +{{- $global := index .Values "global" | default dict -}} +{{- if $global.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes $global.customAttributes -}} +{{- end -}} + +{{- if .Values.customAttributes -}} +{{- $customAttributes = mergeOverwrite $customAttributes .Values.customAttributes -}} +{{- end -}} + +{{- toYaml $customAttributes -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_dnsconfig.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_dnsconfig.tpl new file mode 100644 index 0000000000..d4e40aa8af --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_dnsconfig.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod dnsConfig */ -}} +{{- define "newrelic.common.dnsConfig" -}} + {{- if .Values.dnsConfig -}} + {{- toYaml .Values.dnsConfig -}} + {{- else if .Values.global -}} + {{- if .Values.global.dnsConfig -}} + {{- toYaml .Values.global.dnsConfig -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_fedramp.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_fedramp.tpl new file mode 100644 index 0000000000..9df8d6b5e9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_fedramp.tpl @@ -0,0 +1,25 @@ +{{- /* Defines the fedRAMP flag */ -}} +{{- define "newrelic.common.fedramp.enabled" -}} + {{- if .Values.fedramp -}} + {{- if .Values.fedramp.enabled -}} + {{- .Values.fedramp.enabled -}} + {{- end -}} + {{- else if .Values.global -}} + {{- if .Values.global.fedramp -}} + {{- if .Values.global.fedramp.enabled -}} + {{- .Values.global.fedramp.enabled -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + + + +{{- /* Return FedRAMP value directly ready to be templated */ -}} +{{- define "newrelic.common.fedramp.enabled.value" -}} +{{- if include "newrelic.common.fedramp.enabled" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_hostnetwork.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_hostnetwork.tpl new file mode 100644 index 0000000000..4cf017ef7e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_hostnetwork.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the hostNetwork toggle. +This helper allows to override the global `.global.hostNetwork` with the value of `.hostNetwork`. +Returns "true" if `hostNetwork` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.hostNetwork" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} + +{{- /* +`get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs + +We also want only to return when this is true, returning `false` here will template "false" (string) when doing +an `(include "newrelic.common.hostNetwork" .)`, which is not an "empty string" so it is `true` if it is used +as an evaluation somewhere else. +*/ -}} +{{- if get .Values "hostNetwork" | kindIs "bool" -}} + {{- if .Values.hostNetwork -}} + {{- .Values.hostNetwork -}} + {{- end -}} +{{- else if get $global "hostNetwork" | kindIs "bool" -}} + {{- if $global.hostNetwork -}} + {{- $global.hostNetwork -}} + {{- end -}} +{{- end -}} +{{- end -}} + + +{{- /* +Abstraction of the hostNetwork toggle. +This helper abstracts the function "newrelic.common.hostNetwork" to return true or false directly. +*/ -}} +{{- define "newrelic.common.hostNetwork.value" -}} +{{- if include "newrelic.common.hostNetwork" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_images.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_images.tpl new file mode 100644 index 0000000000..d4fb432905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_images.tpl @@ -0,0 +1,94 @@ +{{- /* +Return the proper image name +{{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.image" -}} + {{- $registryName := include "newrelic.common.images.registry" ( dict "imageRoot" .imageRoot "defaultRegistry" .defaultRegistry "context" .context ) -}} + {{- $repositoryName := include "newrelic.common.images.repository" .imageRoot -}} + {{- $tag := include "newrelic.common.images.tag" ( dict "imageRoot" .imageRoot "context" .context) -}} + + {{- if $registryName -}} + {{- printf "%s/%s:%s" $registryName $repositoryName $tag | quote -}} + {{- else -}} + {{- printf "%s:%s" $repositoryName $tag | quote -}} + {{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image registry +{{ include "newrelic.common.images.registry" ( dict "imageRoot" .Values.path.to.the.image "defaultRegistry" "your.private.registry.tld" "context" .) }} +*/ -}} +{{- define "newrelic.common.images.registry" -}} +{{- $globalRegistry := "" -}} +{{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- with .context.Values.global.images.registry -}} + {{- $globalRegistry = . -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- $localRegistry := "" -}} +{{- if .imageRoot.registry -}} + {{- $localRegistry = .imageRoot.registry -}} +{{- end -}} + +{{- $registry := $localRegistry | default $globalRegistry | default .defaultRegistry -}} +{{- if $registry -}} + {{- $registry -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Return the proper image repository +{{ include "newrelic.common.images.repository" .Values.path.to.the.image }} +*/ -}} +{{- define "newrelic.common.images.repository" -}} + {{- .repository -}} +{{- end -}} + + + +{{- /* +Return the proper image tag +{{ include "newrelic.common.images.tag" ( dict "imageRoot" .Values.path.to.the.image "context" .) }} +*/ -}} +{{- define "newrelic.common.images.tag" -}} + {{- .imageRoot.tag | default .context.Chart.AppVersion | toString -}} +{{- end -}} + + + +{{- /* +Return the proper Image Pull Registry Secret Names evaluating values as templates +{{ include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.path.to.the.images.pullSecrets1, .Values.path.to.the.images.pullSecrets2) "context" .) }} +*/ -}} +{{- define "newrelic.common.images.renderPullSecrets" -}} + {{- $flatlist := list }} + + {{- if .context.Values.global -}} + {{- if .context.Values.global.images -}} + {{- if .context.Values.global.images.pullSecrets -}} + {{- range .context.Values.global.images.pullSecrets -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- range .pullSecrets -}} + {{- if not (empty .) -}} + {{- range . -}} + {{- $flatlist = append $flatlist . -}} + {{- end -}} + {{- end -}} + {{- end -}} + + {{- if $flatlist -}} + {{- toYaml $flatlist -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights.tpl new file mode 100644 index 0000000000..895c377326 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the Insights Key. +*/}} +{{- define "newrelic.common.insightsKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "insightskey" ) -}} +{{- include "newrelic.common.insightsKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +*/}} +{{- define "newrelic.common.insightsKey.secretKeyName" -}} +{{- include "newrelic.common.insightsKey._customSecretKey" . | default "insightsKey" -}} +{{- end -}} + +{{/* +Return local insightsKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._licenseKey" -}} +{{- if .Values.insightsKey -}} + {{- .Values.insightsKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.insightsKey -}} + {{- .Values.global.insightsKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the Insights Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretName" -}} +{{- if .Values.customInsightsKeySecretName -}} + {{- .Values.customInsightsKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretName -}} + {{- .Values.global.customInsightsKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the Insights Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.insightsKey._customSecretKey" -}} +{{- if .Values.customInsightsKeySecretKey -}} + {{- .Values.customInsightsKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customInsightsKeySecretKey }} + {{- .Values.global.customInsightsKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights_secret.yaml.tpl new file mode 100644 index 0000000000..556caa6ca6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_insights_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the insights key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.insightsKey.secret" }} +{{- if not (include "newrelic.common.insightsKey._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.insightsKey._licenseKey" .) }} + {{- fail "You must specify a insightsKey or a customInsightsSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.insightsKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.insightsKey.secretKeyName" . }}: {{ include "newrelic.common.insightsKey._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_labels.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_labels.tpl new file mode 100644 index 0000000000..b025948285 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_labels.tpl @@ -0,0 +1,54 @@ +{{/* +This will render the labels that should be used in all the manifests used by the helm chart. +*/}} +{{- define "newrelic.common.labels" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- $chart := dict "helm.sh/chart" (include "newrelic.common.naming.chart" . ) -}} +{{- $managedBy := dict "app.kubernetes.io/managed-by" .Release.Service -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $labels := mustMergeOverwrite $chart $managedBy $selectorLabels -}} +{{- if .Chart.AppVersion -}} +{{- $labels = mustMergeOverwrite $labels (dict "app.kubernetes.io/version" .Chart.AppVersion) -}} +{{- end -}} + +{{- $globalUserLabels := $global.labels | default dict -}} +{{- $localUserLabels := .Values.labels | default dict -}} + +{{- $labels = mustMergeOverwrite $labels $globalUserLabels $localUserLabels -}} + +{{- toYaml $labels -}} +{{- end -}} + + + +{{/* +This will render the labels that should be used in deployments/daemonsets template pods as a selector. +*/}} +{{- define "newrelic.common.labels.selectorLabels" -}} +{{- $name := dict "app.kubernetes.io/name" ( include "newrelic.common.naming.name" . ) -}} +{{- $instance := dict "app.kubernetes.io/instance" .Release.Name -}} + +{{- $selectorLabels := mustMergeOverwrite $name $instance -}} + +{{- toYaml $selectorLabels -}} +{{- end }} + + + +{{/* +Pod labels +*/}} +{{- define "newrelic.common.labels.podLabels" -}} +{{- $selectorLabels := fromYaml (include "newrelic.common.labels.selectorLabels" . ) -}} + +{{- $global := index .Values "global" | default dict -}} +{{- $globalPodLabels := $global.podLabels | default dict }} + +{{- $localPodLabels := .Values.podLabels | default dict }} + +{{- $podLabels := mustMergeOverwrite $selectorLabels $globalPodLabels $localPodLabels -}} + +{{- toYaml $podLabels -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license.tpl new file mode 100644 index 0000000000..cb349f6bb6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license.tpl @@ -0,0 +1,68 @@ +{{/* +Return the name of the secret holding the License Key. +*/}} +{{- define "newrelic.common.license.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "license" ) -}} +{{- include "newrelic.common.license._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +*/}} +{{- define "newrelic.common.license.secretKeyName" -}} +{{- include "newrelic.common.license._customSecretKey" . | default "licenseKey" -}} +{{- end -}} + +{{/* +Return local licenseKey if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._licenseKey" -}} +{{- if .Values.licenseKey -}} + {{- .Values.licenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.licenseKey -}} + {{- .Values.global.licenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the License Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretName" -}} +{{- if .Values.customSecretName -}} + {{- .Values.customSecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretName -}} + {{- .Values.global.customSecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the License Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._customSecretKey" -}} +{{- if .Values.customSecretLicenseKey -}} + {{- .Values.customSecretLicenseKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customSecretLicenseKey }} + {{- .Values.global.customSecretLicenseKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the user set a custom secret for the license. +This helper is for internal use. +*/}} +{{- define "newrelic.common.license._usesCustomSecret" -}} +{{- if or (include "newrelic.common.license._customSecretName" .) (include "newrelic.common.license._customSecretKey" .) -}} +true +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license_secret.yaml.tpl new file mode 100644 index 0000000000..610a0a3370 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_license_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the license key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.license.secret" }} +{{- if not (include "newrelic.common.license._customSecretName" .) }} +{{- /* Fail if licenseKey is empty and required: */ -}} +{{- if not (include "newrelic.common.license._licenseKey" .) }} + {{- fail "You must specify a licenseKey or a customSecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.license.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.license.secretKeyName" . }}: {{ include "newrelic.common.license._licenseKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_low-data-mode.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_low-data-mode.tpl new file mode 100644 index 0000000000..3dd55ef2ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_low-data-mode.tpl @@ -0,0 +1,26 @@ +{{- /* +Abstraction of the lowDataMode toggle. +This helper allows to override the global `.global.lowDataMode` with the value of `.lowDataMode`. +Returns "true" if `lowDataMode` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.lowDataMode" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "lowDataMode" | kindIs "bool") -}} + {{- if .Values.lowDataMode -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.lowDataMode" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.lowDataMode -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "lowDataMode" | kindIs "bool" -}} + {{- if $global.lowDataMode -}} + {{- $global.lowDataMode -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_naming.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_naming.tpl new file mode 100644 index 0000000000..19fa92648c --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_naming.tpl @@ -0,0 +1,73 @@ +{{/* +This is an function to be called directly with a string just to truncate strings to +63 chars because some Kubernetes name fields are limited to that. +*/}} +{{- define "newrelic.common.naming.truncateToDNS" -}} +{{- . | trunc 63 | trimSuffix "-" }} +{{- end }} + + + +{{- /* +Given a name and a suffix returns a 'DNS Valid' which always include the suffix, truncating the name if needed. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If suffix is too long it gets truncated but it always takes precedence over name, so a 63 chars suffix would suppress the name. +Usage: +{{ include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" "" "suffix" "my-suffix" ) }} +*/ -}} +{{- define "newrelic.common.naming.truncateToDNSWithSuffix" -}} +{{- $suffix := (include "newrelic.common.naming.truncateToDNS" .suffix) -}} +{{- $maxLen := (max (sub 63 (add1 (len $suffix))) 0) -}} {{- /* We prepend "-" to the suffix so an additional character is needed */ -}} + +{{- $newName := .name | trunc ($maxLen | int) | trimSuffix "-" -}} +{{- if $newName -}} +{{- printf "%s-%s" $newName $suffix -}} +{{- else -}} +{{ $suffix }} +{{- end -}} + +{{- end -}} + + + +{{/* +Expand the name of the chart. +Uses the Chart name by default if nameOverride is not set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.name" -}} +{{- $name := .Values.nameOverride | default .Chart.Name -}} +{{- include "newrelic.common.naming.truncateToDNS" $name -}} +{{- end }} + + + +{{/* +Create a default fully qualified app name. +By default the full name will be "" just in if it has the chart name included in that, if not +it will be concatenated like "-". This could change if fullnameOverride or +nameOverride are set. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "newrelic.common.naming.fullname" -}} +{{- $name := include "newrelic.common.naming.name" . -}} + +{{- if .Values.fullnameOverride -}} + {{- $name = .Values.fullnameOverride -}} +{{- else if not (contains $name .Release.Name) -}} + {{- $name = printf "%s-%s" .Release.Name $name -}} +{{- end -}} + +{{- include "newrelic.common.naming.truncateToDNS" $name -}} + +{{- end -}} + + + +{{/* +Create chart name and version as used by the chart label. +This function should not be used for naming objects. Use "common.naming.{name,fullname}" instead. +*/}} +{{- define "newrelic.common.naming.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_nodeselector.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_nodeselector.tpl new file mode 100644 index 0000000000..d488873412 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_nodeselector.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod nodeSelector */ -}} +{{- define "newrelic.common.nodeSelector" -}} + {{- if .Values.nodeSelector -}} + {{- toYaml .Values.nodeSelector -}} + {{- else if .Values.global -}} + {{- if .Values.global.nodeSelector -}} + {{- toYaml .Values.global.nodeSelector -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_priority-class-name.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_priority-class-name.tpl new file mode 100644 index 0000000000..50182b7343 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_priority-class-name.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the pod priorityClassName */ -}} +{{- define "newrelic.common.priorityClassName" -}} + {{- if .Values.priorityClassName -}} + {{- .Values.priorityClassName -}} + {{- else if .Values.global -}} + {{- if .Values.global.priorityClassName -}} + {{- .Values.global.priorityClassName -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_privileged.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_privileged.tpl new file mode 100644 index 0000000000..f3ae814dd9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_privileged.tpl @@ -0,0 +1,28 @@ +{{- /* +This is a helper that returns whether the chart should assume the user is fine deploying privileged pods. +*/ -}} +{{- define "newrelic.common.privileged" -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists. */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if get .Values "privileged" | kindIs "bool" -}} + {{- if .Values.privileged -}} + {{- .Values.privileged -}} + {{- end -}} +{{- else if get $global "privileged" | kindIs "bool" -}} + {{- if $global.privileged -}} + {{- $global.privileged -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* Return directly "true" or "false" based in the exist of "newrelic.common.privileged" */ -}} +{{- define "newrelic.common.privileged.value" -}} +{{- if include "newrelic.common.privileged" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_proxy.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_proxy.tpl new file mode 100644 index 0000000000..60f34c7ec1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_proxy.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the proxy */ -}} +{{- define "newrelic.common.proxy" -}} + {{- if .Values.proxy -}} + {{- .Values.proxy -}} + {{- else if .Values.global -}} + {{- if .Values.global.proxy -}} + {{- .Values.global.proxy -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_region.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_region.tpl new file mode 100644 index 0000000000..bdcacf3235 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_region.tpl @@ -0,0 +1,74 @@ +{{/* +Return the region that is being used by the user +*/}} +{{- define "newrelic.common.region" -}} +{{- if and (include "newrelic.common.license._usesCustomSecret" .) (not (include "newrelic.common.region._fromValues" .)) -}} + {{- fail "This Helm Chart is not able to compute the region. You must specify a .global.region or .region if the license is set using a custom secret." -}} +{{- end -}} + +{{- /* Defaults */ -}} +{{- $region := "us" -}} +{{- if include "newrelic.common.nrStaging" . -}} + {{- $region = "staging" -}} +{{- else if include "newrelic.common.region._isEULicenseKey" . -}} + {{- $region = "eu" -}} +{{- end -}} + +{{- include "newrelic.common.region.validate" (include "newrelic.common.region._fromValues" . | default $region ) -}} +{{- end -}} + + + +{{/* +Returns the region from the values if valid. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. + +Usage: `include "newrelic.common.region.validate" "us"` +*/}} +{{- define "newrelic.common.region.validate" -}} +{{- /* Ref: https://github.com/newrelic/newrelic-client-go/blob/cbe3e4cf2b95fd37095bf2ffdc5d61cffaec17e2/pkg/region/region_constants.go#L8-L21 */ -}} +{{- $region := . | lower -}} +{{- if eq $region "us" -}} + US +{{- else if eq $region "eu" -}} + EU +{{- else if eq $region "staging" -}} + Staging +{{- else if eq $region "local" -}} + Local +{{- else -}} + {{- fail (printf "the region provided is not valid: %s not in \"US\" \"EU\" \"Staging\" \"Local\"" .) -}} +{{- end -}} +{{- end -}} + + + +{{/* +Returns the region from the values. This only return the value from the `values.yaml`. +More intelligence should be used to compute the region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._fromValues" -}} +{{- if .Values.region -}} + {{- .Values.region -}} +{{- else if .Values.global -}} + {{- if .Values.global.region -}} + {{- .Values.global.region -}} + {{- end -}} +{{- end -}} +{{- end -}} + + + +{{/* +Return empty string (falsehood) or "true" if the license is for EU region. +This helper is for internal use. +*/}} +{{- define "newrelic.common.region._isEULicenseKey" -}} +{{- if not (include "newrelic.common.license._usesCustomSecret" .) -}} + {{- $license := include "newrelic.common.license._licenseKey" . -}} + {{- if hasPrefix "eu" $license -}} + true + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_security-context.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_security-context.tpl new file mode 100644 index 0000000000..9edfcabfd0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_security-context.tpl @@ -0,0 +1,23 @@ +{{- /* Defines the container securityContext context */ -}} +{{- define "newrelic.common.securityContext.container" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.containerSecurityContext -}} + {{- toYaml .Values.containerSecurityContext -}} +{{- else if $global.containerSecurityContext -}} + {{- toYaml $global.containerSecurityContext -}} +{{- end -}} +{{- end -}} + + + +{{- /* Defines the pod securityContext context */ -}} +{{- define "newrelic.common.securityContext.pod" -}} +{{- $global := index .Values "global" | default dict -}} + +{{- if .Values.podSecurityContext -}} + {{- toYaml .Values.podSecurityContext -}} +{{- else if $global.podSecurityContext -}} + {{- toYaml $global.podSecurityContext -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_serviceaccount.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_serviceaccount.tpl new file mode 100644 index 0000000000..2d352f6ea9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_serviceaccount.tpl @@ -0,0 +1,90 @@ +{{- /* Defines if the service account has to be created or not */ -}} +{{- define "newrelic.common.serviceAccount.create" -}} +{{- $valueFound := false -}} + +{{- /* Look for a global creation of a service account */ -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "create" | kindIs "bool") -}} + {{- $valueFound = true -}} + {{- if .Values.serviceAccount.create -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.serviceAccount.name" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.serviceAccount.create -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* Look for a local creation of a service account */ -}} +{{- if not $valueFound -}} + {{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} + {{- $global := index .Values "global" | default dict -}} + {{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "create" | kindIs "bool" -}} + {{- $valueFound = true -}} + {{- if $global.serviceAccount.create -}} + {{- $global.serviceAccount.create -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- /* In case no serviceAccount value has been found, default to "true" */ -}} +{{- if not $valueFound -}} +true +{{- end -}} +{{- end -}} + + + +{{- /* Defines the name of the service account */ -}} +{{- define "newrelic.common.serviceAccount.name" -}} +{{- $localServiceAccount := "" -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if (get .Values.serviceAccount "name" | kindIs "string") -}} + {{- $localServiceAccount = .Values.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := "" -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "name" | kindIs "string" -}} + {{- $globalServiceAccount = $global.serviceAccount.name -}} + {{- end -}} +{{- end -}} + +{{- if (include "newrelic.common.serviceAccount.create" .) -}} + {{- $localServiceAccount | default $globalServiceAccount | default (include "newrelic.common.naming.fullname" .) -}} +{{- else -}} + {{- $localServiceAccount | default $globalServiceAccount | default "default" -}} +{{- end -}} +{{- end -}} + + + +{{- /* Merge the global and local annotations for the service account */ -}} +{{- define "newrelic.common.serviceAccount.annotations" -}} +{{- $localServiceAccount := dict -}} +{{- if get .Values "serviceAccount" | kindIs "map" -}} + {{- if get .Values.serviceAccount "annotations" -}} + {{- $localServiceAccount = .Values.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $globalServiceAccount := dict -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "serviceAccount" | kindIs "map" -}} + {{- if get $global.serviceAccount "annotations" -}} + {{- $globalServiceAccount = $global.serviceAccount.annotations -}} + {{- end -}} +{{- end -}} + +{{- $merged := mustMergeOverwrite $globalServiceAccount $localServiceAccount -}} + +{{- if $merged -}} + {{- toYaml $merged -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_staging.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_staging.tpl new file mode 100644 index 0000000000..bd9ad09bb9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_staging.tpl @@ -0,0 +1,39 @@ +{{- /* +Abstraction of the nrStaging toggle. +This helper allows to override the global `.global.nrStaging` with the value of `.nrStaging`. +Returns "true" if `nrStaging` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.nrStaging" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "nrStaging" | kindIs "bool") -}} + {{- if .Values.nrStaging -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.nrStaging" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.nrStaging -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "nrStaging" | kindIs "bool" -}} + {{- if $global.nrStaging -}} + {{- $global.nrStaging -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Returns "true" of "false" directly instead of empty string (Helm falsiness) based on the exit of "newrelic.common.nrStaging" +*/ -}} +{{- define "newrelic.common.nrStaging.value" -}} +{{- if include "newrelic.common.nrStaging" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_tolerations.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_tolerations.tpl new file mode 100644 index 0000000000..e016b38e27 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_tolerations.tpl @@ -0,0 +1,10 @@ +{{- /* Defines the Pod tolerations */ -}} +{{- define "newrelic.common.tolerations" -}} + {{- if .Values.tolerations -}} + {{- toYaml .Values.tolerations -}} + {{- else if .Values.global -}} + {{- if .Values.global.tolerations -}} + {{- toYaml .Values.global.tolerations -}} + {{- end -}} + {{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey.tpl new file mode 100644 index 0000000000..982ea8e09d --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey.tpl @@ -0,0 +1,56 @@ +{{/* +Return the name of the secret holding the API Key. +*/}} +{{- define "newrelic.common.userKey.secretName" -}} +{{- $default := include "newrelic.common.naming.truncateToDNSWithSuffix" ( dict "name" (include "newrelic.common.naming.fullname" .) "suffix" "userkey" ) -}} +{{- include "newrelic.common.userKey._customSecretName" . | default $default -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +*/}} +{{- define "newrelic.common.userKey.secretKeyName" -}} +{{- include "newrelic.common.userKey._customSecretKey" . | default "userKey" -}} +{{- end -}} + +{{/* +Return local API Key if set, global otherwise. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._userKey" -}} +{{- if .Values.userKey -}} + {{- .Values.userKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.userKey -}} + {{- .Values.global.userKey -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name of the secret holding the API Key. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretName" -}} +{{- if .Values.customUserKeySecretName -}} + {{- .Values.customUserKeySecretName -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretName -}} + {{- .Values.global.customUserKeySecretName -}} + {{- end -}} +{{- end -}} +{{- end -}} + +{{/* +Return the name key for the API Key inside the secret. +This helper is for internal use. +*/}} +{{- define "newrelic.common.userKey._customSecretKey" -}} +{{- if .Values.customUserKeySecretKey -}} + {{- .Values.customUserKeySecretKey -}} +{{- else if .Values.global -}} + {{- if .Values.global.customUserKeySecretKey }} + {{- .Values.global.customUserKeySecretKey -}} + {{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey_secret.yaml.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey_secret.yaml.tpl new file mode 100644 index 0000000000..b979856548 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_userkey_secret.yaml.tpl @@ -0,0 +1,21 @@ +{{/* +Renders the user key secret if user has not specified a custom secret. +*/}} +{{- define "newrelic.common.userKey.secret" }} +{{- if not (include "newrelic.common.userKey._customSecretName" .) }} +{{- /* Fail if user key is empty and required: */ -}} +{{- if not (include "newrelic.common.userKey._userKey" .) }} + {{- fail "You must specify a userKey or a customUserKeySecretName containing it" }} +{{- end }} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "newrelic.common.userKey.secretName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +data: + {{ include "newrelic.common.userKey.secretKeyName" . }}: {{ include "newrelic.common.userKey._userKey" . | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_verbose-log.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_verbose-log.tpl new file mode 100644 index 0000000000..2286d46815 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/templates/_verbose-log.tpl @@ -0,0 +1,54 @@ +{{- /* +Abstraction of the verbose toggle. +This helper allows to override the global `.global.verboseLog` with the value of `.verboseLog`. +Returns "true" if `verbose` is enabled, otherwise "" (empty string) +*/ -}} +{{- define "newrelic.common.verboseLog" -}} +{{- /* `get` will return "" (empty string) if value is not found, and the value otherwise, so we can type-assert with kindIs */ -}} +{{- if (get .Values "verboseLog" | kindIs "bool") -}} + {{- if .Values.verboseLog -}} + {{- /* + We want only to return when this is true, returning `false` here will template "false" (string) when doing + an `(include "newrelic.common.verboseLog" .)`, which is not an "empty string" so it is `true` if it is used + as an evaluation somewhere else. + */ -}} + {{- .Values.verboseLog -}} + {{- end -}} +{{- else -}} +{{- /* This allows us to use `$global` as an empty dict directly in case `Values.global` does not exists */ -}} +{{- $global := index .Values "global" | default dict -}} +{{- if get $global "verboseLog" | kindIs "bool" -}} + {{- if $global.verboseLog -}} + {{- $global.verboseLog -}} + {{- end -}} +{{- end -}} +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return true or false directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsBoolean" -}} +{{- if include "newrelic.common.verboseLog" . -}} +true +{{- else -}} +false +{{- end -}} +{{- end -}} + + + +{{- /* +Abstraction of the verbose toggle. +This helper abstracts the function "newrelic.common.verboseLog" to return 1 or 0 directly. +*/ -}} +{{- define "newrelic.common.verboseLog.valueAsInt" -}} +{{- if include "newrelic.common.verboseLog" . -}} +1 +{{- else -}} +0 +{{- end -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/values.yaml new file mode 100644 index 0000000000..75e2d112ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/charts/common-library/values.yaml @@ -0,0 +1 @@ +# values are not needed for the library chart, however this file is still needed for helm lint to work. diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-lowdatamode-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-lowdatamode-values.yaml new file mode 100644 index 0000000000..57b307a2d0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-lowdatamode-values.yaml @@ -0,0 +1,9 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +lowDataMode: true + +image: + repository: e2e/nri-prometheus + tag: "test" # Defaults to chart's appVersion diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-override-global-lowdatamode.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-override-global-lowdatamode.yaml new file mode 100644 index 0000000000..7ff1a730fa --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-override-global-lowdatamode.yaml @@ -0,0 +1,10 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + lowDataMode: true + +lowDataMode: false + +image: + repository: e2e/nri-prometheus + tag: "test" # Defaults to chart's appVersion diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-values.yaml new file mode 100644 index 0000000000..fcd07b2d3e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/ci/test-values.yaml @@ -0,0 +1,104 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +lowDataMode: true + +nameOverride: my-custom-name + +image: + registry: + repository: e2e/nri-prometheus + tag: "test" + imagePullPolicy: IfNotPresent + +resources: + limits: + cpu: 200m + memory: 512Mi + requests: + cpu: 100m + memory: 256Mi + +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 generated using the name template + name: "" + # Specify any annotations to add to the ServiceAccount + annotations: + foo: bar + +# If you wish to provide your own config.yaml file include it under config: +# the sample config file is included here as an example. +config: + scrape_duration: "60s" + scrape_timeout: "15s" + + scrape_services: false + scrape_endpoints: true + + audit: false + + insecure_skip_verify: false + + scrape_enabled_label: "prometheus.io/scrape" + + require_scrape_enabled_label_for_nodes: true + + transformations: + - description: "Custom transformation Example" + rename_attributes: + - metric_prefix: "foo_" + attributes: + old_label: "newLabel" + ignore_metrics: + - prefixes: + - bar_ + copy_attributes: + - from_metric: "foo_info" + to_metrics: "foo_" + match_by: + - namespace + +podAnnotations: + custom-pod-annotation: test + +podSecurityContext: + runAsUser: 1000 + runAsGroup: 3000 + fsGroup: 2000 + +containerSecurityContext: + runAsUser: 2000 + +tolerations: + - key: "key1" + operator: "Exists" + effect: "NoSchedule" + +nodeSelector: + kubernetes.io/os: linux + +affinity: + nodeAffinity: + requiredDuringSchedulingIgnoredDuringExecution: + nodeSelectorTerms: + - matchExpressions: + - key: kubernetes.io/os + operator: In + values: + - linux + +nrStaging: false + +fedramp: + enabled: true + +proxy: + +verboseLog: true diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/static/lowdatamodedefaults.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/static/lowdatamodedefaults.yaml new file mode 100644 index 0000000000..f749e28da4 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/static/lowdatamodedefaults.yaml @@ -0,0 +1,10 @@ +transformations: + - description: "Low data mode defaults" + ignore_metrics: + # Ignore the following metrics. + # These metrics are already collected by the New Relic Kubernetes Integration. + - prefixes: + - kube_ + - container_ + - machine_ + - cadvisor_ diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/_helpers.tpl b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/_helpers.tpl new file mode 100644 index 0000000000..23c072bd72 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/_helpers.tpl @@ -0,0 +1,15 @@ +{{/* vim: set filetype=mustache: */}} + +{{/* +Returns mergeTransformations +Helm can't merge maps of different types. Need to manually create a `transformations` section. +*/}} +{{- define "nri-prometheus.mergeTransformations" -}} + {{/* Remove current `transformations` from config. */}} + {{- omit .Values.config "transformations" | toYaml | nindent 4 -}} + {{/* Create new `transformations` yaml section with merged configs from .Values.config.transformations and lowDataMode. */}} + transformations: + {{- .Values.config.transformations | toYaml | nindent 4 -}} + {{ $lowDataDefault := .Files.Get "static/lowdatamodedefaults.yaml" | fromYaml }} + {{- $lowDataDefault.transformations | toYaml | nindent 4 -}} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrole.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrole.yaml new file mode 100644 index 0000000000..ac4734d31e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrole.yaml @@ -0,0 +1,23 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +rules: +- apiGroups: [""] + resources: + - "nodes" + - "nodes/metrics" + - "nodes/stats" + - "nodes/proxy" + - "pods" + - "services" + - "endpoints" + verbs: ["get", "list", "watch"] +- nonResourceURLs: + - /metrics + verbs: + - get +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrolebinding.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrolebinding.yaml new file mode 100644 index 0000000000..44244653ff --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/clusterrolebinding.yaml @@ -0,0 +1,16 @@ +{{- if .Values.rbac.create }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "newrelic.common.naming.fullname" . }} +subjects: +- kind: ServiceAccount + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/configmap.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/configmap.yaml new file mode 100644 index 0000000000..5daeed64ad --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/configmap.yaml @@ -0,0 +1,21 @@ +kind: ConfigMap +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +apiVersion: v1 +data: + config.yaml: | + cluster_name: {{ include "newrelic.common.cluster" . }} +{{- if .Values.config -}} + {{- if and (.Values.config.transformations) (include "newrelic.common.lowDataMode" .) -}} + {{- include "nri-prometheus.mergeTransformations" . -}} + {{- else if (include "newrelic.common.lowDataMode" .) -}} + {{ $lowDataDefault := .Files.Get "static/lowdatamodedefaults.yaml" | fromYaml }} + {{- mergeOverwrite (deepCopy .Values.config) $lowDataDefault | toYaml | nindent 4 -}} + {{- else }} + {{- .Values.config | toYaml | nindent 4 -}} + {{- end -}} +{{- end -}} + diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/deployment.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/deployment.yaml new file mode 100644 index 0000000000..8529b71f40 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/deployment.yaml @@ -0,0 +1,98 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "newrelic.common.naming.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} +spec: + replicas: 1 + selector: + matchLabels: + {{- /* We cannot use the common library here because of a legacy issue */}} + {{- /* `selector` is inmutable and the previous chart did not have all the idiomatic labels */}} + app.kubernetes.io/name: {{ include "newrelic.common.naming.name" . }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + {{- with .Values.podAnnotations }} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "newrelic.common.labels.podLabels" . | nindent 8 }} + spec: + serviceAccountName: {{ include "newrelic.common.serviceAccount.name" . }} + {{- with include "newrelic.common.securityContext.pod" . }} + securityContext: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.images.renderPullSecrets" ( dict "pullSecrets" (list .Values.image.pullSecrets) "context" .) }} + imagePullSecrets: + {{- . | nindent 8 }} + {{- end }} + containers: + - name: nri-prometheus + {{- with include "newrelic.common.securityContext.container" . }} + securityContext: + {{- . | nindent 10 }} + {{- end }} + image: {{ include "newrelic.common.images.image" ( dict "imageRoot" .Values.image "context" .) }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + args: + - "--configfile=/etc/nri-prometheus/config.yaml" + ports: + - containerPort: 8080 + volumeMounts: + - name: config-volume + mountPath: /etc/nri-prometheus/ + env: + - name: "LICENSE_KEY" + valueFrom: + secretKeyRef: + name: {{ include "newrelic.common.license.secretName" . }} + key: {{ include "newrelic.common.license.secretKeyName" . }} + {{- if (include "newrelic.common.nrStaging" .) }} + - name: "METRIC_API_URL" + value: "https://staging-metric-api.newrelic.com/metric/v1/infra" + {{- else if (include "newrelic.common.fedramp.enabled" .) }} + - name: "METRIC_API_URL" + value: "https://gov-metric-api.newrelic.com/metric/v1" + {{- end }} + {{- with include "newrelic.common.proxy" . }} + - name: EMITTER_PROXY + value: {{ . | quote }} + {{- end }} + {{- with include "newrelic.common.verboseLog" . }} + - name: "VERBOSE" + value: {{ . | quote }} + {{- end }} + - name: "BEARER_TOKEN_FILE" + value: "/var/run/secrets/kubernetes.io/serviceaccount/token" + - name: "CA_FILE" + value: "/var/run/secrets/kubernetes.io/serviceaccount/ca.crt" + resources: + {{- toYaml .Values.resources | nindent 10 }} + volumes: + - name: config-volume + configMap: + name: {{ include "newrelic.common.naming.fullname" . }} + {{- with include "newrelic.common.priorityClassName" . }} + priorityClassName: {{ . }} + {{- end }} + {{- with include "newrelic.common.dnsConfig" . }} + dnsConfig: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.nodeSelector" . }} + nodeSelector: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.affinity" . }} + affinity: + {{- . | nindent 8 }} + {{- end }} + {{- with include "newrelic.common.tolerations" . }} + tolerations: + {{- . | nindent 8 }} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/secret.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/secret.yaml new file mode 100644 index 0000000000..f558ee86c9 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/secret.yaml @@ -0,0 +1,2 @@ +{{- /* Common library will take care of creating the secret or not. */}} +{{- include "newrelic.common.license.secret" . }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/serviceaccount.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/serviceaccount.yaml new file mode 100644 index 0000000000..df451ec905 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if (include "newrelic.common.serviceAccount.create" .) }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "newrelic.common.serviceAccount.name" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "newrelic.common.labels" . | nindent 4 }} + {{- with (include "newrelic.common.serviceAccount.annotations" .) }} + annotations: + {{- . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/configmap_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/configmap_test.yaml new file mode 100644 index 0000000000..ae7d921fe6 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/configmap_test.yaml @@ -0,0 +1,86 @@ +suite: test nri-prometheus configmap +templates: + - templates/configmap.yaml + - templates/deployment.yaml +tests: + - it: creates the config map with default config in values.yaml and cluster_name. + set: + licenseKey: fakeLicense + cluster: my-cluster-name + asserts: + - equal: + path: data["config.yaml"] + value: |- + cluster_name: my-cluster-name + audit: false + insecure_skip_verify: false + require_scrape_enabled_label_for_nodes: true + scrape_enabled_label: prometheus.io/scrape + scrape_endpoints: false + scrape_services: true + transformations: [] + template: templates/configmap.yaml + + - it: creates the config map with lowDataMode. + set: + licenseKey: fakeLicense + cluster: my-cluster-name + lowDataMode: true + asserts: + - equal: + path: data["config.yaml"] + value: |- + cluster_name: my-cluster-name + audit: false + insecure_skip_verify: false + require_scrape_enabled_label_for_nodes: true + scrape_enabled_label: prometheus.io/scrape + scrape_endpoints: false + scrape_services: true + transformations: + - description: Low data mode defaults + ignore_metrics: + - prefixes: + - kube_ + - container_ + - machine_ + - cadvisor_ + template: templates/configmap.yaml + + - it: merges existing transformation with lowDataMode. + set: + licenseKey: fakeLicense + cluster: my-cluster-name + lowDataMode: true + config: + transformations: + - description: Custom transformation Example + rename_attributes: + - metric_prefix: test_ + attributes: + container_name: containerName + asserts: + - equal: + path: data["config.yaml"] + value: |- + cluster_name: my-cluster-name + audit: false + insecure_skip_verify: false + require_scrape_enabled_label_for_nodes: true + scrape_enabled_label: prometheus.io/scrape + scrape_endpoints: false + scrape_services: true + transformations: + - description: Custom transformation Example + rename_attributes: + - attributes: + container_name: containerName + metric_prefix: test_ + - description: Low data mode defaults + ignore_metrics: + - prefixes: + - kube_ + - container_ + - machine_ + - cadvisor_ + template: templates/configmap.yaml diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/deployment_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/deployment_test.yaml new file mode 100644 index 0000000000..cb6f903405 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/deployment_test.yaml @@ -0,0 +1,82 @@ +suite: test deployment +templates: + - templates/deployment.yaml + - templates/configmap.yaml + +release: + name: release + +tests: + - it: adds defaults. + set: + licenseKey: fakeLicense + cluster: test + asserts: + - equal: + path: spec.template.metadata.labels["app.kubernetes.io/instance"] + value: release + template: templates/deployment.yaml + - equal: + path: spec.template.metadata.labels["app.kubernetes.io/name"] + value: nri-prometheus + template: templates/deployment.yaml + - equal: + path: spec.selector.matchLabels + value: + app.kubernetes.io/name: nri-prometheus + template: templates/deployment.yaml + - isNotEmpty: + path: spec.template.metadata.annotations["checksum/config"] + template: templates/deployment.yaml + + - it: adds METRIC_API_URL when nrStaging is true. + set: + licenseKey: fakeLicense + cluster: test + nrStaging: true + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: "METRIC_API_URL" + value: "https://staging-metric-api.newrelic.com/metric/v1/infra" + template: templates/deployment.yaml + + - it: adds FedRamp endpoint when FedRamp is enabled. + set: + licenseKey: fakeLicense + cluster: test + fedramp: + enabled: true + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: "METRIC_API_URL" + value: "https://gov-metric-api.newrelic.com/metric/v1" + template: templates/deployment.yaml + + - it: adds proxy when enabled. + set: + licenseKey: fakeLicense + cluster: test + proxy: "https://my-proxy:9999" + asserts: + - contains: + path: spec.template.spec.containers[0].env + content: + name: "EMITTER_PROXY" + value: "https://my-proxy:9999" + template: templates/deployment.yaml + + - it: set priorityClassName. + set: + licenseKey: fakeLicense + cluster: test + priorityClassName: foo + asserts: + - equal: + path: spec.template.spec.priorityClassName + value: foo + template: templates/deployment.yaml + diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/labels_test.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/labels_test.yaml new file mode 100644 index 0000000000..2b6cb53bb1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/tests/labels_test.yaml @@ -0,0 +1,32 @@ +suite: test object names +templates: + - templates/clusterrole.yaml + - templates/clusterrolebinding.yaml + - templates/configmap.yaml + - templates/deployment.yaml + - templates/secret.yaml + - templates/serviceaccount.yaml + +release: + name: release + revision: + +tests: + - it: adds default labels. + set: + licenseKey: fakeLicense + cluster: test + asserts: + - equal: + path: metadata.labels["app.kubernetes.io/instance"] + value: release + - equal: + path: metadata.labels["app.kubernetes.io/managed-by"] + value: Helm + - equal: + path: metadata.labels["app.kubernetes.io/name"] + value: nri-prometheus + - isNotEmpty: + path: metadata.labels["app.kubernetes.io/version"] + - isNotEmpty: + path: metadata.labels["helm.sh/chart"] diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/values.yaml new file mode 100644 index 0000000000..4c562cc660 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/nri-prometheus/values.yaml @@ -0,0 +1,251 @@ +# -- Override the name of the chart +nameOverride: "" +# -- Override the full name of the release +fullnameOverride: "" + +# -- Name of the Kubernetes cluster monitored. Can be configured also with `global.cluster` +cluster: "" +# -- This set this license key to use. Can be configured also with `global.licenseKey` +licenseKey: "" +# -- In case you don't want to have the license key in you values, this allows you to point to a user created secret to get the key from there. Can be configured also with `global.customSecretName` +customSecretName: "" +# -- In case you don't want to have the license key in you values, this allows you to point to which secret key is the license key located. Can be configured also with `global.customSecretLicenseKey` +customSecretLicenseKey: "" + +# -- Image for the New Relic Kubernetes integration +# @default -- See `values.yaml` +image: + registry: + repository: newrelic/nri-prometheus + tag: "" # Defaults to chart's appVersion + imagePullPolicy: IfNotPresent + # -- The secrets that are needed to pull images from a custom registry. + pullSecrets: [] + # - name: regsecret + +resources: {} + # limits: + # cpu: 200m + # memory: 512Mi + # requests: + # cpu: 100m + # memory: 256Mi + +rbac: + # -- Specifies whether RBAC resources should be created + create: true + +serviceAccount: + # -- Add these annotations to the service account we create. Can be configured also with `global.serviceAccount.annotations` + annotations: {} + # -- Configures if the service account should be created or not. Can be configured also with `global.serviceAccount.create` + create: true + # -- Change the name of the service account. This is honored if you disable on this cahrt the creation of the service account so you can use your own. Can be configured also with `global.serviceAccount.name` + name: + +# -- Annotations to be added to all pods created by the integration. +podAnnotations: {} +# -- Additional labels for chart pods. Can be configured also with `global.podLabels` +podLabels: {} +# -- Additional labels for chart objects. Can be configured also with `global.labels` +labels: {} + +# -- Sets pod's priorityClassName. Can be configured also with `global.priorityClassName` +priorityClassName: "" +# -- (bool) Sets pod's hostNetwork. Can be configured also with `global.hostNetwork` +# @default -- `false` +hostNetwork: +# -- Sets pod's dnsConfig. Can be configured also with `global.dnsConfig` +dnsConfig: {} + +# -- Sets security context (at pod level). Can be configured also with `global.podSecurityContext` +podSecurityContext: {} +# -- Sets security context (at container level). Can be configured also with `global.containerSecurityContext` +containerSecurityContext: {} + +# -- Sets pod/node affinities. Can be configured also with `global.affinity` +affinity: {} +# -- Sets pod's node selector. Can be configured also with `global.nodeSelector` +nodeSelector: {} +# -- Sets pod's tolerations to node taints. Can be configured also with `global.tolerations` +tolerations: [] + + +# -- Provides your own `config.yaml` for this integration. +# Ref: https://docs.newrelic.com/docs/infrastructure/prometheus-integrations/install-configure-openmetrics/configure-prometheus-openmetrics-integrations/#example-configuration-file +# @default -- See `values.yaml` +config: + # How often the integration should run. + # Default: "30s" + # scrape_duration: "30s" + + # The HTTP client timeout when fetching data from targets. + # Default: "5s" + # scrape_timeout: "5s" + + # scrape_services Allows to enable scraping the service and not the endpoints behind. + # When endpoints are scraped this is no longer needed + scrape_services: true + + # scrape_endpoints Allows to enable scraping directly endpoints instead of services as prometheus service natively does. + # Please notice that depending on the number of endpoints behind a service the load can increase considerably + scrape_endpoints: false + + # How old must the entries used for calculating the counters delta be + # before the telemetry emitter expires them. + # Default: "5m" + # telemetry_emitter_delta_expiration_age: "5m" + + # How often must the telemetry emitter check for expired delta entries. + # Default: "5m" + # telemetry_emitter_delta_expiration_check_interval: "5m" + + # Whether the integration should run in audit mode or not. Defaults to false. + # Audit mode logs the uncompressed data sent to New Relic. Use this to log all data sent. + # It does not include verbose mode. This can lead to a high log volume, use with care. + # Default: false + audit: false + + # Whether the integration should skip TLS verification or not. + # Default: false + insecure_skip_verify: false + + # The label used to identify scrapeable targets. + # Targets can be identified using a label or annotation. + # Default: "prometheus.io/scrape" + scrape_enabled_label: "prometheus.io/scrape" + + # Whether k8s nodes need to be labelled to be scraped or not. + # Default: true + require_scrape_enabled_label_for_nodes: true + + # Number of worker threads used for scraping targets. + # For large clusters with many (>400) targets, slowly increase until scrape + # time falls between the desired `scrape_duration`. + # Increasing this value too much will result in huge memory consumption if too + # many metrics are being scraped. + # Default: 4 + # worker_threads: 4 + + # Maximum number of metrics to keep in memory until a report is triggered. + # Changing this value is not recommended unless instructed by the New Relic support team. + # max_stored_metrics: 10000 + + # Minimum amount of time to wait between reports. Cannot be lowered than the default, 200ms. + # Changing this value is not recommended unless instructed by the New Relic support team. + # min_emitter_harvest_period: 200ms + + # targets: + # - description: Secure etcd example + # urls: ["https://192.168.3.1:2379", "https://192.168.3.2:2379", "https://192.168.3.3:2379"] + # If true the Kubernetes Service Account token will be included as a Bearer token in the HTTP request. + # use_bearer: false + # tls_config: + # ca_file_path: "/etc/etcd/etcd-client-ca.crt" + # cert_file_path: "/etc/etcd/etcd-client.crt" + # key_file_path: "/etc/etcd/etcd-client.key" + + # Certificate to add to the root CA that the emitter will use when + # verifying server certificates. + # If left empty, TLS uses the host's root CA set. + # emitter_ca_file: "/path/to/cert/server.pem" + + # Set to true in order to stop autodiscovery in the k8s cluster. It can be useful when running the Pod with a service account + # having limited privileges. + # Default: false + # disable_autodiscovery: false + + # Whether the emitter should skip TLS verification when submitting data. + # Default: false + # emitter_insecure_skip_verify: false + + # Histogram support is based on New Relic's guidelines for higher + # level metrics abstractions https://github.com/newrelic/newrelic-exporter-specs/blob/master/Guidelines.md. + # To better support visualization of this data, percentiles are calculated + # based on the histogram metrics and sent to New Relic. + # By default, the following percentiles are calculated: 50, 95 and 99. + # + # percentiles: + # - 50 + # - 95 + # - 99 + + transformations: [] + # - description: "Custom transformation Example" + # rename_attributes: + # - metric_prefix: "" + # attributes: + # container_name: "containerName" + # pod_name: "podName" + # namespace: "namespaceName" + # node: "nodeName" + # container: "containerName" + # pod: "podName" + # deployment: "deploymentName" + # ignore_metrics: + # # Ignore the following metrics. + # # These metrics are already collected by the New Relic Kubernetes Integration. + # - prefixes: + # - kube_daemonset_ + # - kube_deployment_ + # - kube_endpoint_ + # - kube_namespace_ + # - kube_node_ + # - kube_persistentvolume_ + # - kube_pod_ + # - kube_replicaset_ + # - kube_service_ + # - kube_statefulset_ + # copy_attributes: + # # Copy all the labels from the timeseries with metric name + # # `kube_hpa_labels` into every timeseries with a metric name that + # # starts with `kube_hpa_` only if they share the same `namespace` + # # and `hpa` labels. + # - from_metric: "kube_hpa_labels" + # to_metrics: "kube_hpa_" + # match_by: + # - namespace + # - hpa + # - from_metric: "kube_daemonset_labels" + # to_metrics: "kube_daemonset_" + # match_by: + # - namespace + # - daemonset + # - from_metric: "kube_statefulset_labels" + # to_metrics: "kube_statefulset_" + # match_by: + # - namespace + # - statefulset + # - from_metric: "kube_endpoint_labels" + # to_metrics: "kube_endpoint_" + # match_by: + # - namespace + # - endpoint + # - from_metric: "kube_service_labels" + # to_metrics: "kube_service_" + # match_by: + # - namespace + # - service + # - from_metric: "kube_node_labels" + # to_metrics: "kube_node_" + # match_by: + # - namespace + # - node + +# -- (bool) Reduces number of metrics sent in order to reduce costs. Can be configured also with `global.lowDataMode` +# @default -- false +lowDataMode: + +# -- Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port`. Can be configured also with `global.proxy` +proxy: "" + +# -- (bool) Send the metrics to the staging backend. Requires a valid staging license key. Can be configured also with `global.nrStaging` +# @default -- false +nrStaging: +fedramp: + # fedramp.enabled -- (bool) Enables FedRAMP. Can be configured also with `global.fedramp.enabled` + # @default -- false + enabled: +# -- (bool) Sets the debug logs to this integration or all integrations if it is set globally. Can be configured also with `global.verboseLog` +# @default -- false +verboseLog: diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/Chart.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/Chart.yaml new file mode 100644 index 0000000000..a0ce0a388f --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/Chart.yaml @@ -0,0 +1,4 @@ +apiVersion: v2 +name: pixie-operator-chart +type: application +version: 0.1.6 diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/olm_crd.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/olm_crd.yaml new file mode 100644 index 0000000000..3f5429f789 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/olm_crd.yaml @@ -0,0 +1,9045 @@ +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: catalogsources.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: CatalogSource + listKind: CatalogSourceList + plural: catalogsources + shortNames: + - catsrc + singular: catalogsource + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: The pretty name of the catalog + jsonPath: .spec.displayName + name: Display + type: string + - description: The type of the catalog + jsonPath: .spec.sourceType + name: Type + type: string + - description: The publisher of the catalog + jsonPath: .spec.publisher + name: Publisher + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: CatalogSource is a repository of CSVs, CRDs, and operator packages. + type: object + required: + - metadata + - spec + 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: + type: object + required: + - sourceType + properties: + address: + description: 'Address is a host that OLM can use to connect to a pre-existing registry. Format: : Only used when SourceType = SourceTypeGrpc. Ignored when the Image field is set.' + type: string + configMap: + description: ConfigMap is the name of the ConfigMap to be used to back a configmap-server registry. Only used when SourceType = SourceTypeConfigmap or SourceTypeInternal. + type: string + description: + type: string + displayName: + description: Metadata + type: string + grpcPodConfig: + description: GrpcPodConfig exposes different overrides for the pod spec of the CatalogSource Pod. Only used when SourceType = SourceTypeGrpc and Image is set. + type: object + properties: + affinity: + description: Affinity is the catalog source's pod's affinity. + type: object + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + type: object + 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. + type: array + 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). + type: object + required: + - preference + - weight + properties: + preference: + description: A node selector term, associated with the corresponding weight. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: object + required: + - nodeSelectorTerms + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + type: array + 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. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + type: object + 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. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: array + 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 + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + 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)). + type: object + 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. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: array + 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 + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + extractContent: + description: ExtractContent configures the gRPC catalog Pod to extract catalog metadata from the provided index image and use a well-known version of the `opm` server to expose it. The catalog index image that this CatalogSource is configured to use *must* be using the file-based catalogs in order to utilize this feature. + type: object + required: + - cacheDir + - catalogDir + properties: + cacheDir: + description: CacheDir is the directory storing the pre-calculated API cache. + type: string + catalogDir: + description: CatalogDir is the directory storing the file-based catalog contents. + type: string + memoryTarget: + description: "MemoryTarget configures the $GOMEMLIMIT value for the gRPC catalog Pod. This is a soft memory limit for the server, which the runtime will attempt to meet but makes no guarantees that it will do so. If this value is set, the Pod will have the following modifications made to the container running the server: - the $GOMEMLIMIT environment variable will be set to this value in bytes - the memory request will be set to this value \n This field should be set if it's desired to reduce the footprint of a catalog server as much as possible, or if a catalog being served is very large and needs more than the default allocation. If your index image has a file- system cache, determine a good approximation for this value by doubling the size of the package cache at /tmp/cache/cache/packages.json in the index image. \n This field is best-effort; if unset, no default will be used and no Pod memory limit or $GOMEMLIMIT value will be set." + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + nodeSelector: + description: NodeSelector is a selector which must be true for the pod to fit on a node. Selector which must match a node's labels for the pod to be scheduled on that node. + type: object + additionalProperties: + type: string + priorityClassName: + description: If specified, indicates the pod's priority. If not specified, the pod priority will be default or zero if there is no default. + type: string + securityContextConfig: + description: "SecurityContextConfig can be one of `legacy` or `restricted`. The CatalogSource's pod is either injected with the right pod.spec.securityContext and pod.spec.container[*].securityContext values to allow the pod to run in Pod Security Admission (PSA) `restricted` mode, or doesn't set these values at all, in which case the pod can only be run in PSA `baseline` or `privileged` namespaces. Currently if the SecurityContextConfig is unspecified, the default value of `legacy` is used. Specifying a value other than `legacy` or `restricted` result in a validation error. When using older catalog images, which could not be run in `restricted` mode, the SecurityContextConfig should be set to `legacy`. \n In a future version will the default will be set to `restricted`, catalog maintainers should rebuild their catalogs with a version of opm that supports running catalogSource pods in `restricted` mode to prepare for these changes. \n More information about PSA can be found here: https://kubernetes.io/docs/concepts/security/pod-security-admission/'" + type: string + default: legacy + enum: + - legacy + - restricted + tolerations: + description: Tolerations are the catalog source's pod's tolerations. + type: array + items: + description: The pod this Toleration is attached to tolerates any taint that matches the triple using the matching operator . + type: object + 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. + type: integer + format: int64 + 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 + icon: + type: object + required: + - base64data + - mediatype + properties: + base64data: + type: string + mediatype: + type: string + image: + description: Image is an operator-registry container image to instantiate a registry-server with. Only used when SourceType = SourceTypeGrpc. If present, the address field is ignored. + type: string + priority: + description: 'Priority field assigns a weight to the catalog source to prioritize them so that it can be consumed by the dependency resolver. Usage: Higher weight indicates that this catalog source is preferred over lower weighted catalog sources during dependency resolution. The range of the priority value can go from positive to negative in the range of int32. The default value to a catalog source with unassigned priority would be 0. The catalog source with the same priority values will be ranked lexicographically based on its name.' + type: integer + publisher: + type: string + secrets: + description: Secrets represent set of secrets that can be used to access the contents of the catalog. It is best to keep this list small, since each will need to be tried for every catalog entry. + type: array + items: + type: string + sourceType: + description: SourceType is the type of source + type: string + updateStrategy: + description: UpdateStrategy defines how updated catalog source images can be discovered Consists of an interval that defines polling duration and an embedded strategy type + type: object + properties: + registryPoll: + type: object + properties: + interval: + description: Interval is used to determine the time interval between checks of the latest catalog source version. The catalog operator polls to see if a new version of the catalog source is available. If available, the latest image is pulled and gRPC traffic is directed to the latest catalog source. + type: string + status: + type: object + properties: + conditions: + description: Represents the state of a CatalogSource. Note that Message and Reason represent the original status information, which may be migrated to be conditions based in the future. Any new features introduced will use conditions. + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - lastTransitionTime + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + x-kubernetes-list-map-keys: + - type + x-kubernetes-list-type: map + configMapReference: + type: object + required: + - name + - namespace + properties: + lastUpdateTime: + type: string + format: date-time + name: + type: string + namespace: + type: string + resourceVersion: + type: string + uid: + description: UID is a type that holds unique ID values, including UUIDs. Because we don't ONLY use UUIDs, this is an alias to string. Being a type captures intent and helps make sure that UIDs and names do not get conflated. + type: string + connectionState: + type: object + required: + - lastObservedState + properties: + address: + type: string + lastConnect: + type: string + format: date-time + lastObservedState: + type: string + latestImageRegistryPoll: + description: The last time the CatalogSource image registry has been polled to ensure the image is up-to-date + type: string + format: date-time + message: + description: A human readable message indicating details about why the CatalogSource is in this condition. + type: string + reason: + description: Reason is the reason the CatalogSource was transitioned to its current state. + type: string + registryService: + type: object + properties: + createdAt: + type: string + format: date-time + port: + type: string + protocol: + type: string + serviceName: + type: string + serviceNamespace: + type: string + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: clusterserviceversions.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: ClusterServiceVersion + listKind: ClusterServiceVersionList + plural: clusterserviceversions + shortNames: + - csv + - csvs + singular: clusterserviceversion + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: The name of the CSV + jsonPath: .spec.displayName + name: Display + type: string + - description: The version of the CSV + jsonPath: .spec.version + name: Version + type: string + - description: The name of a CSV that this one replaces + jsonPath: .spec.replaces + name: Replaces + type: string + - jsonPath: .status.phase + name: Phase + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + description: ClusterServiceVersion is a Custom Resource of type `ClusterServiceVersionSpec`. + type: object + required: + - metadata + - spec + 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: ClusterServiceVersionSpec declarations tell OLM how to install an operator that can manage apps for a given version. + type: object + required: + - displayName + - install + properties: + annotations: + description: Annotations is an unstructured key value map stored with a resource that may be set by external tools to store and retrieve arbitrary metadata. + type: object + additionalProperties: + type: string + apiservicedefinitions: + description: APIServiceDefinitions declares all of the extension apis managed or required by an operator being ran by ClusterServiceVersion. + type: object + properties: + owned: + type: array + items: + description: APIServiceDescription provides details to OLM about apis provided via aggregation + type: object + required: + - group + - kind + - name + - version + properties: + actionDescriptors: + type: array + items: + description: ActionDescriptor describes a declarative action that can be performed on a custom resource instance + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + containerPort: + type: integer + format: int32 + deploymentName: + type: string + description: + type: string + displayName: + type: string + group: + type: string + kind: + type: string + name: + type: string + resources: + type: array + items: + description: APIResourceReference is a reference to a Kubernetes resource type that the referrer utilizes. + type: object + required: + - kind + - name + - version + properties: + kind: + description: Kind of the referenced resource type. + type: string + name: + description: Plural name of the referenced resource type (CustomResourceDefinition.Spec.Names[].Plural). Empty string if the referenced resource type is not a custom resource. + type: string + version: + description: API Version of the referenced resource type. + type: string + specDescriptors: + type: array + items: + description: SpecDescriptor describes a field in a spec block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + statusDescriptors: + type: array + items: + description: StatusDescriptor describes a field in a status block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + version: + type: string + required: + type: array + items: + description: APIServiceDescription provides details to OLM about apis provided via aggregation + type: object + required: + - group + - kind + - name + - version + properties: + actionDescriptors: + type: array + items: + description: ActionDescriptor describes a declarative action that can be performed on a custom resource instance + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + containerPort: + type: integer + format: int32 + deploymentName: + type: string + description: + type: string + displayName: + type: string + group: + type: string + kind: + type: string + name: + type: string + resources: + type: array + items: + description: APIResourceReference is a reference to a Kubernetes resource type that the referrer utilizes. + type: object + required: + - kind + - name + - version + properties: + kind: + description: Kind of the referenced resource type. + type: string + name: + description: Plural name of the referenced resource type (CustomResourceDefinition.Spec.Names[].Plural). Empty string if the referenced resource type is not a custom resource. + type: string + version: + description: API Version of the referenced resource type. + type: string + specDescriptors: + type: array + items: + description: SpecDescriptor describes a field in a spec block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + statusDescriptors: + type: array + items: + description: StatusDescriptor describes a field in a status block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + version: + type: string + cleanup: + description: Cleanup specifies the cleanup behaviour when the CSV gets deleted + type: object + required: + - enabled + properties: + enabled: + type: boolean + customresourcedefinitions: + description: "CustomResourceDefinitions declares all of the CRDs managed or required by an operator being ran by ClusterServiceVersion. \n If the CRD is present in the Owned list, it is implicitly required." + type: object + properties: + owned: + type: array + items: + description: CRDDescription provides details to OLM about the CRDs + type: object + required: + - kind + - name + - version + properties: + actionDescriptors: + type: array + items: + description: ActionDescriptor describes a declarative action that can be performed on a custom resource instance + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + description: + type: string + displayName: + type: string + kind: + type: string + name: + type: string + resources: + type: array + items: + description: APIResourceReference is a reference to a Kubernetes resource type that the referrer utilizes. + type: object + required: + - kind + - name + - version + properties: + kind: + description: Kind of the referenced resource type. + type: string + name: + description: Plural name of the referenced resource type (CustomResourceDefinition.Spec.Names[].Plural). Empty string if the referenced resource type is not a custom resource. + type: string + version: + description: API Version of the referenced resource type. + type: string + specDescriptors: + type: array + items: + description: SpecDescriptor describes a field in a spec block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + statusDescriptors: + type: array + items: + description: StatusDescriptor describes a field in a status block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + version: + type: string + required: + type: array + items: + description: CRDDescription provides details to OLM about the CRDs + type: object + required: + - kind + - name + - version + properties: + actionDescriptors: + type: array + items: + description: ActionDescriptor describes a declarative action that can be performed on a custom resource instance + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + description: + type: string + displayName: + type: string + kind: + type: string + name: + type: string + resources: + type: array + items: + description: APIResourceReference is a reference to a Kubernetes resource type that the referrer utilizes. + type: object + required: + - kind + - name + - version + properties: + kind: + description: Kind of the referenced resource type. + type: string + name: + description: Plural name of the referenced resource type (CustomResourceDefinition.Spec.Names[].Plural). Empty string if the referenced resource type is not a custom resource. + type: string + version: + description: API Version of the referenced resource type. + type: string + specDescriptors: + type: array + items: + description: SpecDescriptor describes a field in a spec block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + statusDescriptors: + type: array + items: + description: StatusDescriptor describes a field in a status block of a CRD so that OLM can consume it + type: object + required: + - path + properties: + description: + type: string + displayName: + type: string + path: + type: string + value: + description: RawMessage is a raw encoded JSON value. It implements Marshaler and Unmarshaler and can be used to delay JSON decoding or precompute a JSON encoding. + type: string + format: byte + x-descriptors: + type: array + items: + type: string + version: + type: string + description: + description: Description of the operator. Can include the features, limitations or use-cases of the operator. + type: string + displayName: + description: The name of the operator in display format. + type: string + icon: + description: The icon for this operator. + type: array + items: + type: object + required: + - base64data + - mediatype + properties: + base64data: + type: string + mediatype: + type: string + install: + description: NamedInstallStrategy represents the block of an ClusterServiceVersion resource where the install strategy is specified. + type: object + required: + - strategy + properties: + spec: + description: StrategyDetailsDeployment represents the parsed details of a Deployment InstallStrategy. + type: object + required: + - deployments + properties: + clusterPermissions: + type: array + items: + description: StrategyDeploymentPermissions describe the rbac rules and service account needed by the install strategy + type: object + required: + - rules + - serviceAccountName + properties: + rules: + type: array + items: + description: PolicyRule holds information that describes a policy rule, but does not contain information about who the rule applies to or which namespace the rule applies to. + type: object + required: + - verbs + properties: + apiGroups: + description: APIGroups is the name of the APIGroup that contains the resources. If multiple API groups are specified, any action requested against one of the enumerated resources in any API group will be allowed. "" represents the core API group and "*" represents all API groups. + type: array + items: + type: string + nonResourceURLs: + description: NonResourceURLs is a set of partial urls that a user should have access to. *s are allowed, but only as the full, final step in the path Since non-resource URLs are not namespaced, this field is only applicable for ClusterRoles referenced from a ClusterRoleBinding. Rules can either apply to API resources (such as "pods" or "secrets") or non-resource URL paths (such as "/api"), but not both. + type: array + items: + type: string + resourceNames: + description: ResourceNames is an optional white list of names that the rule applies to. An empty set means that everything is allowed. + type: array + items: + type: string + resources: + description: Resources is a list of resources this rule applies to. '*' represents all resources. + type: array + items: + type: string + verbs: + description: Verbs is a list of Verbs that apply to ALL the ResourceKinds contained in this rule. '*' represents all verbs. + type: array + items: + type: string + serviceAccountName: + type: string + deployments: + type: array + items: + description: StrategyDeploymentSpec contains the name, spec and labels for the deployment ALM should create + type: object + required: + - name + - spec + properties: + label: + description: Set is a map of label:value. It implements Labels. + type: object + additionalProperties: + type: string + name: + type: string + spec: + description: DeploymentSpec is the specification of the desired behavior of the Deployment. + type: object + required: + - selector + - template + properties: + minReadySeconds: + description: 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) + type: integer + format: int32 + paused: + description: Indicates that the deployment is paused. + type: boolean + progressDeadlineSeconds: + description: The maximum time in seconds for a deployment to make progress before it is considered to be failed. The deployment controller will continue to process failed deployments and a condition with a ProgressDeadlineExceeded reason will be surfaced in the deployment status. Note that progress will not be estimated during the time a deployment is paused. Defaults to 600s. + type: integer + format: int32 + replicas: + description: Number of desired pods. This is a pointer to distinguish between explicit zero and not specified. Defaults to 1. + type: integer + format: int32 + revisionHistoryLimit: + description: The number of old ReplicaSets to retain to allow rollback. This is a pointer to distinguish between explicit zero and not specified. Defaults to 10. + type: integer + format: int32 + selector: + description: Label selector for pods. Existing ReplicaSets whose pods are selected by this will be the ones affected by this deployment. It must match the pod template's labels. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + strategy: + description: The deployment strategy to use to replace existing pods with new ones. + type: object + properties: + rollingUpdate: + description: 'Rolling update config params. Present only if DeploymentStrategyType = RollingUpdate. --- TODO: Update this to follow our convention for oneOf, whatever we decide it to be.' + type: object + properties: + maxSurge: + description: 'The maximum number of pods that can be scheduled above the desired number of pods. Value can be an absolute number (ex: 5) or a percentage of desired pods (ex: 10%). This can not be 0 if MaxUnavailable is 0. Absolute number is calculated from percentage by rounding up. Defaults to 25%. Example: when this is set to 30%, the new ReplicaSet can be scaled up immediately when the rolling update starts, such that the total number of old and new pods do not exceed 130% of desired pods. Once old pods have been killed, new ReplicaSet can be scaled up further, ensuring that total number of pods running at any time during the update is at most 130% of desired pods.' + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + maxUnavailable: + description: 'The maximum number of pods that can be unavailable during the update. Value can be an absolute number (ex: 5) or a percentage of desired pods (ex: 10%). Absolute number is calculated from percentage by rounding down. This can not be 0 if MaxSurge is 0. Defaults to 25%. Example: when this is set to 30%, the old ReplicaSet can be scaled down to 70% of desired pods immediately when the rolling update starts. Once new pods are ready, old ReplicaSet can be scaled down further, followed by scaling up the new ReplicaSet, ensuring that the total number of pods available at all times during the update is at least 70% of desired pods.' + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + type: + description: Type of deployment. Can be "Recreate" or "RollingUpdate". Default is RollingUpdate. + type: string + template: + description: Template describes the pods that will be created. The only allowed template.spec.restartPolicy value is "Always". + type: object + properties: + metadata: + description: 'Standard object''s metadata. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#metadata' + type: object + x-kubernetes-preserve-unknown-fields: true + spec: + description: 'Specification of the desired behavior of the pod. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status' + type: object + required: + - containers + properties: + activeDeadlineSeconds: + description: Optional duration in seconds the pod may be active on the node relative to StartTime before the system will actively try to mark it failed and kill associated containers. Value must be a positive integer. + type: integer + format: int64 + affinity: + description: If specified, the pod's scheduling constraints + type: object + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + type: object + 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. + type: array + 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). + type: object + required: + - preference + - weight + properties: + preference: + description: A node selector term, associated with the corresponding weight. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: object + required: + - nodeSelectorTerms + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + type: array + 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. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + type: object + 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. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: array + 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 + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + 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)). + type: object + 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. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: array + 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 + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + automountServiceAccountToken: + description: AutomountServiceAccountToken indicates whether a service account token should be automatically mounted. + type: boolean + containers: + description: List of containers belonging to the pod. Containers cannot currently be added or removed. There must be at least one container in a Pod. Cannot be updated. + type: array + items: + description: A single application container that you want to run within a pod. + type: object + required: + - name + properties: + args: + description: 'Arguments to the entrypoint. The container image''s CMD is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container''s environment. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated. More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell' + type: array + items: + type: string + command: + description: 'Entrypoint array. Not executed within a shell. The container image''s ENTRYPOINT is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container''s environment. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated. More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell' + type: array + items: + type: string + env: + description: List of environment variables to set in the container. Cannot be updated. + type: array + items: + description: EnvVar represents an environment variable present in a Container. + type: object + required: + - name + properties: + name: + description: Name of the environment variable. Must be a C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded using the previously defined environment variables in the container and any service environment variables. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. Cannot be used if value is not empty. + type: object + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + type: object + required: + - key + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap or its key must be defined + type: boolean + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`, spec.nodeName, spec.serviceAccountName, status.hostIP, status.podIP, status.podIPs.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage, requests.cpu, requests.memory and requests.ephemeral-storage) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + secretKeyRef: + description: Selects a key of a secret in the pod's namespace + type: object + required: + - key + properties: + key: + description: The key of the secret to select from. Must be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret or its key must be defined + type: boolean + envFrom: + description: List of sources to populate environment variables in the container. The keys defined within a source must be a C_IDENTIFIER. All invalid keys will be reported as an event when the container is starting. When a key exists in multiple sources, the value associated with the last source will take precedence. Values defined by an Env with a duplicate key will take precedence. Cannot be updated. + type: array + items: + description: EnvFromSource represents the source of a set of ConfigMaps + type: object + properties: + configMapRef: + description: The ConfigMap to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap must be defined + type: boolean + prefix: + description: An optional identifier to prepend to each key in the ConfigMap. Must be a C_IDENTIFIER. + type: string + secretRef: + description: The Secret to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret must be defined + type: boolean + image: + description: 'Container image name. More info: https://kubernetes.io/docs/concepts/containers/images This field is optional to allow higher level config management to default or override container images in workload controllers like Deployments and StatefulSets.' + type: string + imagePullPolicy: + description: 'Image pull policy. One of Always, Never, IfNotPresent. Defaults to Always if :latest tag is specified, or IfNotPresent otherwise. Cannot be updated. More info: https://kubernetes.io/docs/concepts/containers/images#updating-images' + type: string + lifecycle: + description: Actions that the management system should take in response to container lifecycle events. Cannot be updated. + type: object + properties: + postStart: + description: 'PostStart is called immediately after a container is created. If the handler fails, the container is terminated and restarted according to its restart policy. Other management of the container blocks until the hook completes. More info: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + tcpSocket: + description: Deprecated. TCPSocket is NOT supported as a LifecycleHandler and kept for the backward compatibility. There are no validation of this field and lifecycle hooks will fail in runtime when tcp handler is specified. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + preStop: + description: 'PreStop is called immediately before a container is terminated due to an API request or management event such as liveness/startup probe failure, preemption, resource contention, etc. The handler is not called if the container crashes or exits. The Pod''s termination grace period countdown begins before the PreStop hook is executed. Regardless of the outcome of the handler, the container will eventually terminate within the Pod''s termination grace period (unless delayed by finalizers). Other management of the container blocks until the hook completes or until the termination grace period is reached. More info: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + tcpSocket: + description: Deprecated. TCPSocket is NOT supported as a LifecycleHandler and kept for the backward compatibility. There are no validation of this field and lifecycle hooks will fail in runtime when tcp handler is specified. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + livenessProbe: + description: 'Periodic probe of container liveness. Container will be restarted if the probe fails. Cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + name: + description: Name of the container specified as a DNS_LABEL. Each container in a pod must have a unique name (DNS_LABEL). Cannot be updated. + type: string + ports: + description: List of ports to expose from the container. Not specifying a port here DOES NOT prevent that port from being exposed. Any port which is listening on the default "0.0.0.0" address inside a container will be accessible from the network. Modifying this array with strategic merge patch may corrupt the data. For more information See https://github.com/kubernetes/kubernetes/issues/108255. Cannot be updated. + type: array + items: + description: ContainerPort represents a network port in a single container. + type: object + required: + - containerPort + properties: + containerPort: + description: Number of port to expose on the pod's IP address. This must be a valid port number, 0 < x < 65536. + type: integer + format: int32 + hostIP: + description: What host IP to bind the external port to. + type: string + hostPort: + description: Number of port to expose on the host. If specified, this must be a valid port number, 0 < x < 65536. If HostNetwork is specified, this must match ContainerPort. Most containers do not need this. + type: integer + format: int32 + name: + description: If specified, this must be an IANA_SVC_NAME and unique within the pod. Each named port in a pod must have a unique name. Name for the port that can be referred to by services. + type: string + protocol: + description: Protocol for port. Must be UDP, TCP, or SCTP. Defaults to "TCP". + type: string + default: TCP + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + description: 'Periodic probe of container service readiness. Container will be removed from service endpoints if the probe fails. Cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + resizePolicy: + description: Resources resize policy for the container. + type: array + items: + description: ContainerResizePolicy represents resource resize policy for the container. + type: object + required: + - resourceName + - restartPolicy + properties: + resourceName: + description: 'Name of the resource to which this resource resize policy applies. Supported values: cpu, memory.' + type: string + restartPolicy: + description: Restart policy to apply when specified resource is resized. If not specified, it defaults to NotRequired. + type: string + x-kubernetes-list-type: atomic + resources: + description: 'Compute Resources required by this container. Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + properties: + claims: + description: "Claims lists the names of resources, defined in spec.resourceClaims, that are used by this container. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable. It can only be set for containers." + type: array + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + type: object + required: + - name + 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 + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + description: 'Limits describes the maximum amount of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + requests: + 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 + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + restartPolicy: + description: 'RestartPolicy defines the restart behavior of individual containers in a pod. This field may only be set for init containers, and the only allowed value is "Always". For non-init containers or when this field is not specified, the restart behavior is defined by the Pod''s restart policy and the container type. Setting the RestartPolicy as "Always" for the init container will have the following effect: this init container will be continually restarted on exit until all regular containers have terminated. Once all regular containers have completed, all init containers with restartPolicy "Always" will be shut down. This lifecycle differs from normal init containers and is often referred to as a "sidecar" container. Although this init container still starts in the init container sequence, it does not wait for the container to complete before proceeding to the next init container. Instead, the next init container starts immediately after this init container is started, or after any startupProbe has successfully completed.' + type: string + securityContext: + description: 'SecurityContext defines the security options the container should be run with. If set, the fields of SecurityContext override the equivalent fields of PodSecurityContext. More info: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/' + type: object + properties: + allowPrivilegeEscalation: + description: 'AllowPrivilegeEscalation controls whether a process can gain more privileges than its parent process. This bool directly controls if the no_new_privs flag will be set on the container process. AllowPrivilegeEscalation is true always when the container is: 1) run as Privileged 2) has CAP_SYS_ADMIN Note that this field cannot be set when spec.os.name is windows.' + type: boolean + capabilities: + description: The capabilities to add/drop when running containers. Defaults to the default set of capabilities granted by the container runtime. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + add: + description: Added capabilities + type: array + items: + description: Capability represent POSIX capabilities type + type: string + drop: + description: Removed capabilities + type: array + items: + description: Capability represent POSIX capabilities type + type: string + privileged: + description: Run container in privileged mode. Processes in privileged containers are essentially equivalent to root on the host. Defaults to false. Note that this field cannot be set when spec.os.name is windows. + type: boolean + procMount: + description: procMount denotes the type of proc mount to use for the containers. The default is DefaultProcMount which uses the container runtime defaults for readonly paths and masked paths. This requires the ProcMountType feature flag to be enabled. Note that this field cannot be set when spec.os.name is windows. + type: string + readOnlyRootFilesystem: + description: Whether this container has a read-only root filesystem. Default is false. Note that this field cannot be set when spec.os.name is windows. + type: boolean + runAsGroup: + description: The GID to run the entrypoint of the container process. Uses runtime default if unset. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + runAsNonRoot: + description: Indicates that the container must run as a non-root user. If true, the Kubelet will validate the image at runtime to ensure that it does not run as UID 0 (root) and fail to start the container if it does. If unset or false, no such validation will be performed. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: boolean + runAsUser: + description: The UID to run the entrypoint of the container process. Defaults to user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + seLinuxOptions: + description: The SELinux context to be applied to the container. If unspecified, the container runtime will allocate a random SELinux context for each container. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + level: + description: Level is SELinux level label that applies to the container. + type: string + role: + description: Role is a SELinux role label that applies to the container. + type: string + type: + description: Type is a SELinux type label that applies to the container. + type: string + user: + description: User is a SELinux user label that applies to the container. + type: string + seccompProfile: + description: The seccomp options to use by this container. If seccomp options are provided at both the pod & container level, the container options override the pod options. Note that this field cannot be set when spec.os.name is windows. + type: object + required: + - type + properties: + localhostProfile: + description: localhostProfile indicates a profile defined in a file on the node should be used. The profile must be preconfigured on the node to work. Must be a descending path, relative to the kubelet's configured seccomp profile location. Must be set if type is "Localhost". Must NOT be set for any other type. + type: string + type: + description: "type indicates which kind of seccomp profile will be applied. Valid options are: \n Localhost - a profile defined in a file on the node should be used. RuntimeDefault - the container runtime default profile should be used. Unconfined - no profile should be applied." + type: string + windowsOptions: + description: The Windows specific settings applied to all containers. If unspecified, the options from the PodSecurityContext will be used. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is linux. + type: object + properties: + gmsaCredentialSpec: + description: GMSACredentialSpec is where the GMSA admission webhook (https://github.com/kubernetes-sigs/windows-gmsa) inlines the contents of the GMSA credential spec named by the GMSACredentialSpecName field. + type: string + gmsaCredentialSpecName: + description: GMSACredentialSpecName is the name of the GMSA credential spec to use. + type: string + hostProcess: + description: HostProcess determines if a container should be run as a 'Host Process' container. All of a Pod's containers must have the same effective HostProcess value (it is not allowed to have a mix of HostProcess containers and non-HostProcess containers). In addition, if HostProcess is true then HostNetwork must also be set to true. + type: boolean + runAsUserName: + description: The UserName in Windows to run the entrypoint of the container process. Defaults to the user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: string + startupProbe: + description: 'StartupProbe indicates that the Pod has successfully initialized. If specified, no other probes are executed until this completes successfully. If this probe fails, the Pod will be restarted, just as if the livenessProbe failed. This can be used to provide different probe parameters at the beginning of a Pod''s lifecycle, when it might take a long time to load data or warm a cache, than during steady-state operation. This cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + stdin: + description: Whether this container should allocate a buffer for stdin in the container runtime. If this is not set, reads from stdin in the container will always result in EOF. Default is false. + type: boolean + stdinOnce: + description: Whether the container runtime should close the stdin channel after it has been opened by a single attach. When stdin is true the stdin stream will remain open across multiple attach sessions. If stdinOnce is set to true, stdin is opened on container start, is empty until the first client attaches to stdin, and then remains open and accepts data until the client disconnects, at which time stdin is closed and remains closed until the container is restarted. If this flag is false, a container processes that reads from stdin will never receive an EOF. Default is false + type: boolean + terminationMessagePath: + description: 'Optional: Path at which the file to which the container''s termination message will be written is mounted into the container''s filesystem. Message written is intended to be brief final status, such as an assertion failure message. Will be truncated by the node if greater than 4096 bytes. The total message length across all containers will be limited to 12kb. Defaults to /dev/termination-log. Cannot be updated.' + type: string + terminationMessagePolicy: + description: Indicate how the termination message should be populated. File will use the contents of terminationMessagePath to populate the container status message on both success and failure. FallbackToLogsOnError will use the last chunk of container log output if the termination message file is empty and the container exited with an error. The log output is limited to 2048 bytes or 80 lines, whichever is smaller. Defaults to File. Cannot be updated. + type: string + tty: + description: Whether this container should allocate a TTY for itself, also requires 'stdin' to be true. Default is false. + type: boolean + volumeDevices: + description: volumeDevices is the list of block devices to be used by the container. + type: array + items: + description: volumeDevice describes a mapping of a raw block device within a container. + type: object + required: + - devicePath + - name + properties: + devicePath: + description: devicePath is the path inside of the container that the device will be mapped to. + type: string + name: + description: name must match the name of a persistentVolumeClaim in the pod + type: string + volumeMounts: + description: Pod volumes to mount into the container's filesystem. Cannot be updated. + type: array + items: + description: VolumeMount describes a mounting of a Volume within a container. + type: object + required: + - mountPath + - name + properties: + mountPath: + description: Path within the container at which the volume should be mounted. Must not contain ':'. + type: string + mountPropagation: + description: mountPropagation determines how mounts are propagated from the host to container and the other way around. When not set, MountPropagationNone is used. This field is beta in 1.10. + type: string + name: + description: This must match the Name of a Volume. + type: string + readOnly: + description: Mounted read-only if true, read-write otherwise (false or unspecified). Defaults to false. + type: boolean + subPath: + description: Path within the volume from which the container's volume should be mounted. Defaults to "" (volume's root). + type: string + subPathExpr: + description: Expanded path within the volume from which the container's volume should be mounted. Behaves similarly to SubPath but environment variable references $(VAR_NAME) are expanded using the container's environment. Defaults to "" (volume's root). SubPathExpr and SubPath are mutually exclusive. + type: string + workingDir: + description: Container's working directory. If not specified, the container runtime's default will be used, which might be configured in the container image. Cannot be updated. + type: string + dnsConfig: + description: Specifies the DNS parameters of a pod. Parameters specified here will be merged to the generated DNS configuration based on DNSPolicy. + type: object + properties: + nameservers: + description: A list of DNS name server IP addresses. This will be appended to the base nameservers generated from DNSPolicy. Duplicated nameservers will be removed. + type: array + items: + type: string + options: + description: A list of DNS resolver options. This will be merged with the base options generated from DNSPolicy. Duplicated entries will be removed. Resolution options given in Options will override those that appear in the base DNSPolicy. + type: array + items: + description: PodDNSConfigOption defines DNS resolver options of a pod. + type: object + properties: + name: + description: Required. + type: string + value: + type: string + searches: + description: A list of DNS search domains for host-name lookup. This will be appended to the base search paths generated from DNSPolicy. Duplicated search paths will be removed. + type: array + items: + type: string + dnsPolicy: + description: Set DNS policy for the pod. Defaults to "ClusterFirst". Valid values are 'ClusterFirstWithHostNet', 'ClusterFirst', 'Default' or 'None'. DNS parameters given in DNSConfig will be merged with the policy selected with DNSPolicy. To have DNS options set along with hostNetwork, you have to specify DNS policy explicitly to 'ClusterFirstWithHostNet'. + type: string + enableServiceLinks: + description: 'EnableServiceLinks indicates whether information about services should be injected into pod''s environment variables, matching the syntax of Docker links. Optional: Defaults to true.' + type: boolean + ephemeralContainers: + description: List of ephemeral containers run in this pod. Ephemeral containers may be run in an existing pod to perform user-initiated actions such as debugging. This list cannot be specified when creating a pod, and it cannot be modified by updating the pod spec. In order to add an ephemeral container to an existing pod, use the pod's ephemeralcontainers subresource. + type: array + items: + description: "An EphemeralContainer is a temporary container that you may add to an existing Pod for user-initiated activities such as debugging. Ephemeral containers have no resource or scheduling guarantees, and they will not be restarted when they exit or when a Pod is removed or restarted. The kubelet may evict a Pod if an ephemeral container causes the Pod to exceed its resource allocation. \n To add an ephemeral container, use the ephemeralcontainers subresource of an existing Pod. Ephemeral containers may not be removed or restarted." + type: object + required: + - name + properties: + args: + description: 'Arguments to the entrypoint. The image''s CMD is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container''s environment. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated. More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell' + type: array + items: + type: string + command: + description: 'Entrypoint array. Not executed within a shell. The image''s ENTRYPOINT is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container''s environment. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated. More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell' + type: array + items: + type: string + env: + description: List of environment variables to set in the container. Cannot be updated. + type: array + items: + description: EnvVar represents an environment variable present in a Container. + type: object + required: + - name + properties: + name: + description: Name of the environment variable. Must be a C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded using the previously defined environment variables in the container and any service environment variables. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. Cannot be used if value is not empty. + type: object + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + type: object + required: + - key + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap or its key must be defined + type: boolean + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`, spec.nodeName, spec.serviceAccountName, status.hostIP, status.podIP, status.podIPs.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage, requests.cpu, requests.memory and requests.ephemeral-storage) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + secretKeyRef: + description: Selects a key of a secret in the pod's namespace + type: object + required: + - key + properties: + key: + description: The key of the secret to select from. Must be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret or its key must be defined + type: boolean + envFrom: + description: List of sources to populate environment variables in the container. The keys defined within a source must be a C_IDENTIFIER. All invalid keys will be reported as an event when the container is starting. When a key exists in multiple sources, the value associated with the last source will take precedence. Values defined by an Env with a duplicate key will take precedence. Cannot be updated. + type: array + items: + description: EnvFromSource represents the source of a set of ConfigMaps + type: object + properties: + configMapRef: + description: The ConfigMap to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap must be defined + type: boolean + prefix: + description: An optional identifier to prepend to each key in the ConfigMap. Must be a C_IDENTIFIER. + type: string + secretRef: + description: The Secret to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret must be defined + type: boolean + image: + description: 'Container image name. More info: https://kubernetes.io/docs/concepts/containers/images' + type: string + imagePullPolicy: + description: 'Image pull policy. One of Always, Never, IfNotPresent. Defaults to Always if :latest tag is specified, or IfNotPresent otherwise. Cannot be updated. More info: https://kubernetes.io/docs/concepts/containers/images#updating-images' + type: string + lifecycle: + description: Lifecycle is not allowed for ephemeral containers. + type: object + properties: + postStart: + description: 'PostStart is called immediately after a container is created. If the handler fails, the container is terminated and restarted according to its restart policy. Other management of the container blocks until the hook completes. More info: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + tcpSocket: + description: Deprecated. TCPSocket is NOT supported as a LifecycleHandler and kept for the backward compatibility. There are no validation of this field and lifecycle hooks will fail in runtime when tcp handler is specified. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + preStop: + description: 'PreStop is called immediately before a container is terminated due to an API request or management event such as liveness/startup probe failure, preemption, resource contention, etc. The handler is not called if the container crashes or exits. The Pod''s termination grace period countdown begins before the PreStop hook is executed. Regardless of the outcome of the handler, the container will eventually terminate within the Pod''s termination grace period (unless delayed by finalizers). Other management of the container blocks until the hook completes or until the termination grace period is reached. More info: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + tcpSocket: + description: Deprecated. TCPSocket is NOT supported as a LifecycleHandler and kept for the backward compatibility. There are no validation of this field and lifecycle hooks will fail in runtime when tcp handler is specified. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + livenessProbe: + description: Probes are not allowed for ephemeral containers. + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + name: + description: Name of the ephemeral container specified as a DNS_LABEL. This name must be unique among all containers, init containers and ephemeral containers. + type: string + ports: + description: Ports are not allowed for ephemeral containers. + type: array + items: + description: ContainerPort represents a network port in a single container. + type: object + required: + - containerPort + properties: + containerPort: + description: Number of port to expose on the pod's IP address. This must be a valid port number, 0 < x < 65536. + type: integer + format: int32 + hostIP: + description: What host IP to bind the external port to. + type: string + hostPort: + description: Number of port to expose on the host. If specified, this must be a valid port number, 0 < x < 65536. If HostNetwork is specified, this must match ContainerPort. Most containers do not need this. + type: integer + format: int32 + name: + description: If specified, this must be an IANA_SVC_NAME and unique within the pod. Each named port in a pod must have a unique name. Name for the port that can be referred to by services. + type: string + protocol: + description: Protocol for port. Must be UDP, TCP, or SCTP. Defaults to "TCP". + type: string + default: TCP + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + description: Probes are not allowed for ephemeral containers. + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + resizePolicy: + description: Resources resize policy for the container. + type: array + items: + description: ContainerResizePolicy represents resource resize policy for the container. + type: object + required: + - resourceName + - restartPolicy + properties: + resourceName: + description: 'Name of the resource to which this resource resize policy applies. Supported values: cpu, memory.' + type: string + restartPolicy: + description: Restart policy to apply when specified resource is resized. If not specified, it defaults to NotRequired. + type: string + x-kubernetes-list-type: atomic + resources: + description: Resources are not allowed for ephemeral containers. Ephemeral containers use spare resources already allocated to the pod. + type: object + properties: + claims: + description: "Claims lists the names of resources, defined in spec.resourceClaims, that are used by this container. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable. It can only be set for containers." + type: array + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + type: object + required: + - name + 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 + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + description: 'Limits describes the maximum amount of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + requests: + 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 + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + restartPolicy: + description: Restart policy for the container to manage the restart behavior of each container within a pod. This may only be set for init containers. You cannot set this field on ephemeral containers. + type: string + securityContext: + description: 'Optional: SecurityContext defines the security options the ephemeral container should be run with. If set, the fields of SecurityContext override the equivalent fields of PodSecurityContext.' + type: object + properties: + allowPrivilegeEscalation: + description: 'AllowPrivilegeEscalation controls whether a process can gain more privileges than its parent process. This bool directly controls if the no_new_privs flag will be set on the container process. AllowPrivilegeEscalation is true always when the container is: 1) run as Privileged 2) has CAP_SYS_ADMIN Note that this field cannot be set when spec.os.name is windows.' + type: boolean + capabilities: + description: The capabilities to add/drop when running containers. Defaults to the default set of capabilities granted by the container runtime. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + add: + description: Added capabilities + type: array + items: + description: Capability represent POSIX capabilities type + type: string + drop: + description: Removed capabilities + type: array + items: + description: Capability represent POSIX capabilities type + type: string + privileged: + description: Run container in privileged mode. Processes in privileged containers are essentially equivalent to root on the host. Defaults to false. Note that this field cannot be set when spec.os.name is windows. + type: boolean + procMount: + description: procMount denotes the type of proc mount to use for the containers. The default is DefaultProcMount which uses the container runtime defaults for readonly paths and masked paths. This requires the ProcMountType feature flag to be enabled. Note that this field cannot be set when spec.os.name is windows. + type: string + readOnlyRootFilesystem: + description: Whether this container has a read-only root filesystem. Default is false. Note that this field cannot be set when spec.os.name is windows. + type: boolean + runAsGroup: + description: The GID to run the entrypoint of the container process. Uses runtime default if unset. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + runAsNonRoot: + description: Indicates that the container must run as a non-root user. If true, the Kubelet will validate the image at runtime to ensure that it does not run as UID 0 (root) and fail to start the container if it does. If unset or false, no such validation will be performed. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: boolean + runAsUser: + description: The UID to run the entrypoint of the container process. Defaults to user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + seLinuxOptions: + description: The SELinux context to be applied to the container. If unspecified, the container runtime will allocate a random SELinux context for each container. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + level: + description: Level is SELinux level label that applies to the container. + type: string + role: + description: Role is a SELinux role label that applies to the container. + type: string + type: + description: Type is a SELinux type label that applies to the container. + type: string + user: + description: User is a SELinux user label that applies to the container. + type: string + seccompProfile: + description: The seccomp options to use by this container. If seccomp options are provided at both the pod & container level, the container options override the pod options. Note that this field cannot be set when spec.os.name is windows. + type: object + required: + - type + properties: + localhostProfile: + description: localhostProfile indicates a profile defined in a file on the node should be used. The profile must be preconfigured on the node to work. Must be a descending path, relative to the kubelet's configured seccomp profile location. Must be set if type is "Localhost". Must NOT be set for any other type. + type: string + type: + description: "type indicates which kind of seccomp profile will be applied. Valid options are: \n Localhost - a profile defined in a file on the node should be used. RuntimeDefault - the container runtime default profile should be used. Unconfined - no profile should be applied." + type: string + windowsOptions: + description: The Windows specific settings applied to all containers. If unspecified, the options from the PodSecurityContext will be used. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is linux. + type: object + properties: + gmsaCredentialSpec: + description: GMSACredentialSpec is where the GMSA admission webhook (https://github.com/kubernetes-sigs/windows-gmsa) inlines the contents of the GMSA credential spec named by the GMSACredentialSpecName field. + type: string + gmsaCredentialSpecName: + description: GMSACredentialSpecName is the name of the GMSA credential spec to use. + type: string + hostProcess: + description: HostProcess determines if a container should be run as a 'Host Process' container. All of a Pod's containers must have the same effective HostProcess value (it is not allowed to have a mix of HostProcess containers and non-HostProcess containers). In addition, if HostProcess is true then HostNetwork must also be set to true. + type: boolean + runAsUserName: + description: The UserName in Windows to run the entrypoint of the container process. Defaults to the user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: string + startupProbe: + description: Probes are not allowed for ephemeral containers. + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + stdin: + description: Whether this container should allocate a buffer for stdin in the container runtime. If this is not set, reads from stdin in the container will always result in EOF. Default is false. + type: boolean + stdinOnce: + description: Whether the container runtime should close the stdin channel after it has been opened by a single attach. When stdin is true the stdin stream will remain open across multiple attach sessions. If stdinOnce is set to true, stdin is opened on container start, is empty until the first client attaches to stdin, and then remains open and accepts data until the client disconnects, at which time stdin is closed and remains closed until the container is restarted. If this flag is false, a container processes that reads from stdin will never receive an EOF. Default is false + type: boolean + targetContainerName: + description: "If set, the name of the container from PodSpec that this ephemeral container targets. The ephemeral container will be run in the namespaces (IPC, PID, etc) of this container. If not set then the ephemeral container uses the namespaces configured in the Pod spec. \n The container runtime must implement support for this feature. If the runtime does not support namespace targeting then the result of setting this field is undefined." + type: string + terminationMessagePath: + description: 'Optional: Path at which the file to which the container''s termination message will be written is mounted into the container''s filesystem. Message written is intended to be brief final status, such as an assertion failure message. Will be truncated by the node if greater than 4096 bytes. The total message length across all containers will be limited to 12kb. Defaults to /dev/termination-log. Cannot be updated.' + type: string + terminationMessagePolicy: + description: Indicate how the termination message should be populated. File will use the contents of terminationMessagePath to populate the container status message on both success and failure. FallbackToLogsOnError will use the last chunk of container log output if the termination message file is empty and the container exited with an error. The log output is limited to 2048 bytes or 80 lines, whichever is smaller. Defaults to File. Cannot be updated. + type: string + tty: + description: Whether this container should allocate a TTY for itself, also requires 'stdin' to be true. Default is false. + type: boolean + volumeDevices: + description: volumeDevices is the list of block devices to be used by the container. + type: array + items: + description: volumeDevice describes a mapping of a raw block device within a container. + type: object + required: + - devicePath + - name + properties: + devicePath: + description: devicePath is the path inside of the container that the device will be mapped to. + type: string + name: + description: name must match the name of a persistentVolumeClaim in the pod + type: string + volumeMounts: + description: Pod volumes to mount into the container's filesystem. Subpath mounts are not allowed for ephemeral containers. Cannot be updated. + type: array + items: + description: VolumeMount describes a mounting of a Volume within a container. + type: object + required: + - mountPath + - name + properties: + mountPath: + description: Path within the container at which the volume should be mounted. Must not contain ':'. + type: string + mountPropagation: + description: mountPropagation determines how mounts are propagated from the host to container and the other way around. When not set, MountPropagationNone is used. This field is beta in 1.10. + type: string + name: + description: This must match the Name of a Volume. + type: string + readOnly: + description: Mounted read-only if true, read-write otherwise (false or unspecified). Defaults to false. + type: boolean + subPath: + description: Path within the volume from which the container's volume should be mounted. Defaults to "" (volume's root). + type: string + subPathExpr: + description: Expanded path within the volume from which the container's volume should be mounted. Behaves similarly to SubPath but environment variable references $(VAR_NAME) are expanded using the container's environment. Defaults to "" (volume's root). SubPathExpr and SubPath are mutually exclusive. + type: string + workingDir: + description: Container's working directory. If not specified, the container runtime's default will be used, which might be configured in the container image. Cannot be updated. + type: string + hostAliases: + description: HostAliases is an optional list of hosts and IPs that will be injected into the pod's hosts file if specified. This is only valid for non-hostNetwork pods. + type: array + items: + description: HostAlias holds the mapping between IP and hostnames that will be injected as an entry in the pod's hosts file. + type: object + properties: + hostnames: + description: Hostnames for the above IP address. + type: array + items: + type: string + ip: + description: IP address of the host file entry. + type: string + hostIPC: + description: 'Use the host''s ipc namespace. Optional: Default to false.' + type: boolean + hostNetwork: + description: Host networking requested for this pod. Use the host's network namespace. If this option is set, the ports that will be used must be specified. Default to false. + type: boolean + hostPID: + description: 'Use the host''s pid namespace. Optional: Default to false.' + type: boolean + hostUsers: + description: 'Use the host''s user namespace. Optional: Default to true. If set to true or not present, the pod will be run in the host user namespace, useful for when the pod needs a feature only available to the host user namespace, such as loading a kernel module with CAP_SYS_MODULE. When set to false, a new userns is created for the pod. Setting false is useful for mitigating container breakout vulnerabilities even allowing users to run their containers as root without actually having root privileges on the host. This field is alpha-level and is only honored by servers that enable the UserNamespacesSupport feature.' + type: boolean + hostname: + description: Specifies the hostname of the Pod If not specified, the pod's hostname will be set to a system-defined value. + type: string + imagePullSecrets: + description: 'ImagePullSecrets is an optional list of references to secrets in the same namespace to use for pulling any of the images used by this PodSpec. If specified, these secrets will be passed to individual puller implementations for them to use. More info: https://kubernetes.io/docs/concepts/containers/images#specifying-imagepullsecrets-on-a-pod' + type: array + items: + description: LocalObjectReference contains enough information to let you locate the referenced object inside the same namespace. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + initContainers: + description: 'List of initialization containers belonging to the pod. Init containers are executed in order prior to containers being started. If any init container fails, the pod is considered to have failed and is handled according to its restartPolicy. The name for an init container or normal container must be unique among all containers. Init containers may not have Lifecycle actions, Readiness probes, Liveness probes, or Startup probes. The resourceRequirements of an init container are taken into account during scheduling by finding the highest request/limit for each resource type, and then using the max of of that value or the sum of the normal containers. Limits are applied to init containers in a similar fashion. Init containers cannot currently be added or removed. Cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/init-containers/' + type: array + items: + description: A single application container that you want to run within a pod. + type: object + required: + - name + properties: + args: + description: 'Arguments to the entrypoint. The container image''s CMD is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container''s environment. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated. More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell' + type: array + items: + type: string + command: + description: 'Entrypoint array. Not executed within a shell. The container image''s ENTRYPOINT is used if this is not provided. Variable references $(VAR_NAME) are expanded using the container''s environment. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Cannot be updated. More info: https://kubernetes.io/docs/tasks/inject-data-application/define-command-argument-container/#running-a-command-in-a-shell' + type: array + items: + type: string + env: + description: List of environment variables to set in the container. Cannot be updated. + type: array + items: + description: EnvVar represents an environment variable present in a Container. + type: object + required: + - name + properties: + name: + description: Name of the environment variable. Must be a C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded using the previously defined environment variables in the container and any service environment variables. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. Cannot be used if value is not empty. + type: object + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + type: object + required: + - key + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap or its key must be defined + type: boolean + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`, spec.nodeName, spec.serviceAccountName, status.hostIP, status.podIP, status.podIPs.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage, requests.cpu, requests.memory and requests.ephemeral-storage) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + secretKeyRef: + description: Selects a key of a secret in the pod's namespace + type: object + required: + - key + properties: + key: + description: The key of the secret to select from. Must be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret or its key must be defined + type: boolean + envFrom: + description: List of sources to populate environment variables in the container. The keys defined within a source must be a C_IDENTIFIER. All invalid keys will be reported as an event when the container is starting. When a key exists in multiple sources, the value associated with the last source will take precedence. Values defined by an Env with a duplicate key will take precedence. Cannot be updated. + type: array + items: + description: EnvFromSource represents the source of a set of ConfigMaps + type: object + properties: + configMapRef: + description: The ConfigMap to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap must be defined + type: boolean + prefix: + description: An optional identifier to prepend to each key in the ConfigMap. Must be a C_IDENTIFIER. + type: string + secretRef: + description: The Secret to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret must be defined + type: boolean + image: + description: 'Container image name. More info: https://kubernetes.io/docs/concepts/containers/images This field is optional to allow higher level config management to default or override container images in workload controllers like Deployments and StatefulSets.' + type: string + imagePullPolicy: + description: 'Image pull policy. One of Always, Never, IfNotPresent. Defaults to Always if :latest tag is specified, or IfNotPresent otherwise. Cannot be updated. More info: https://kubernetes.io/docs/concepts/containers/images#updating-images' + type: string + lifecycle: + description: Actions that the management system should take in response to container lifecycle events. Cannot be updated. + type: object + properties: + postStart: + description: 'PostStart is called immediately after a container is created. If the handler fails, the container is terminated and restarted according to its restart policy. Other management of the container blocks until the hook completes. More info: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + tcpSocket: + description: Deprecated. TCPSocket is NOT supported as a LifecycleHandler and kept for the backward compatibility. There are no validation of this field and lifecycle hooks will fail in runtime when tcp handler is specified. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + preStop: + description: 'PreStop is called immediately before a container is terminated due to an API request or management event such as liveness/startup probe failure, preemption, resource contention, etc. The handler is not called if the container crashes or exits. The Pod''s termination grace period countdown begins before the PreStop hook is executed. Regardless of the outcome of the handler, the container will eventually terminate within the Pod''s termination grace period (unless delayed by finalizers). Other management of the container blocks until the hook completes or until the termination grace period is reached. More info: https://kubernetes.io/docs/concepts/containers/container-lifecycle-hooks/#container-hooks' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + tcpSocket: + description: Deprecated. TCPSocket is NOT supported as a LifecycleHandler and kept for the backward compatibility. There are no validation of this field and lifecycle hooks will fail in runtime when tcp handler is specified. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + livenessProbe: + description: 'Periodic probe of container liveness. Container will be restarted if the probe fails. Cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + name: + description: Name of the container specified as a DNS_LABEL. Each container in a pod must have a unique name (DNS_LABEL). Cannot be updated. + type: string + ports: + description: List of ports to expose from the container. Not specifying a port here DOES NOT prevent that port from being exposed. Any port which is listening on the default "0.0.0.0" address inside a container will be accessible from the network. Modifying this array with strategic merge patch may corrupt the data. For more information See https://github.com/kubernetes/kubernetes/issues/108255. Cannot be updated. + type: array + items: + description: ContainerPort represents a network port in a single container. + type: object + required: + - containerPort + properties: + containerPort: + description: Number of port to expose on the pod's IP address. This must be a valid port number, 0 < x < 65536. + type: integer + format: int32 + hostIP: + description: What host IP to bind the external port to. + type: string + hostPort: + description: Number of port to expose on the host. If specified, this must be a valid port number, 0 < x < 65536. If HostNetwork is specified, this must match ContainerPort. Most containers do not need this. + type: integer + format: int32 + name: + description: If specified, this must be an IANA_SVC_NAME and unique within the pod. Each named port in a pod must have a unique name. Name for the port that can be referred to by services. + type: string + protocol: + description: Protocol for port. Must be UDP, TCP, or SCTP. Defaults to "TCP". + type: string + default: TCP + x-kubernetes-list-map-keys: + - containerPort + - protocol + x-kubernetes-list-type: map + readinessProbe: + description: 'Periodic probe of container service readiness. Container will be removed from service endpoints if the probe fails. Cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + resizePolicy: + description: Resources resize policy for the container. + type: array + items: + description: ContainerResizePolicy represents resource resize policy for the container. + type: object + required: + - resourceName + - restartPolicy + properties: + resourceName: + description: 'Name of the resource to which this resource resize policy applies. Supported values: cpu, memory.' + type: string + restartPolicy: + description: Restart policy to apply when specified resource is resized. If not specified, it defaults to NotRequired. + type: string + x-kubernetes-list-type: atomic + resources: + description: 'Compute Resources required by this container. Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + properties: + claims: + description: "Claims lists the names of resources, defined in spec.resourceClaims, that are used by this container. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable. It can only be set for containers." + type: array + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + type: object + required: + - name + 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 + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + description: 'Limits describes the maximum amount of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + requests: + 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 + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + restartPolicy: + description: 'RestartPolicy defines the restart behavior of individual containers in a pod. This field may only be set for init containers, and the only allowed value is "Always". For non-init containers or when this field is not specified, the restart behavior is defined by the Pod''s restart policy and the container type. Setting the RestartPolicy as "Always" for the init container will have the following effect: this init container will be continually restarted on exit until all regular containers have terminated. Once all regular containers have completed, all init containers with restartPolicy "Always" will be shut down. This lifecycle differs from normal init containers and is often referred to as a "sidecar" container. Although this init container still starts in the init container sequence, it does not wait for the container to complete before proceeding to the next init container. Instead, the next init container starts immediately after this init container is started, or after any startupProbe has successfully completed.' + type: string + securityContext: + description: 'SecurityContext defines the security options the container should be run with. If set, the fields of SecurityContext override the equivalent fields of PodSecurityContext. More info: https://kubernetes.io/docs/tasks/configure-pod-container/security-context/' + type: object + properties: + allowPrivilegeEscalation: + description: 'AllowPrivilegeEscalation controls whether a process can gain more privileges than its parent process. This bool directly controls if the no_new_privs flag will be set on the container process. AllowPrivilegeEscalation is true always when the container is: 1) run as Privileged 2) has CAP_SYS_ADMIN Note that this field cannot be set when spec.os.name is windows.' + type: boolean + capabilities: + description: The capabilities to add/drop when running containers. Defaults to the default set of capabilities granted by the container runtime. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + add: + description: Added capabilities + type: array + items: + description: Capability represent POSIX capabilities type + type: string + drop: + description: Removed capabilities + type: array + items: + description: Capability represent POSIX capabilities type + type: string + privileged: + description: Run container in privileged mode. Processes in privileged containers are essentially equivalent to root on the host. Defaults to false. Note that this field cannot be set when spec.os.name is windows. + type: boolean + procMount: + description: procMount denotes the type of proc mount to use for the containers. The default is DefaultProcMount which uses the container runtime defaults for readonly paths and masked paths. This requires the ProcMountType feature flag to be enabled. Note that this field cannot be set when spec.os.name is windows. + type: string + readOnlyRootFilesystem: + description: Whether this container has a read-only root filesystem. Default is false. Note that this field cannot be set when spec.os.name is windows. + type: boolean + runAsGroup: + description: The GID to run the entrypoint of the container process. Uses runtime default if unset. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + runAsNonRoot: + description: Indicates that the container must run as a non-root user. If true, the Kubelet will validate the image at runtime to ensure that it does not run as UID 0 (root) and fail to start the container if it does. If unset or false, no such validation will be performed. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: boolean + runAsUser: + description: The UID to run the entrypoint of the container process. Defaults to user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + seLinuxOptions: + description: The SELinux context to be applied to the container. If unspecified, the container runtime will allocate a random SELinux context for each container. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + level: + description: Level is SELinux level label that applies to the container. + type: string + role: + description: Role is a SELinux role label that applies to the container. + type: string + type: + description: Type is a SELinux type label that applies to the container. + type: string + user: + description: User is a SELinux user label that applies to the container. + type: string + seccompProfile: + description: The seccomp options to use by this container. If seccomp options are provided at both the pod & container level, the container options override the pod options. Note that this field cannot be set when spec.os.name is windows. + type: object + required: + - type + properties: + localhostProfile: + description: localhostProfile indicates a profile defined in a file on the node should be used. The profile must be preconfigured on the node to work. Must be a descending path, relative to the kubelet's configured seccomp profile location. Must be set if type is "Localhost". Must NOT be set for any other type. + type: string + type: + description: "type indicates which kind of seccomp profile will be applied. Valid options are: \n Localhost - a profile defined in a file on the node should be used. RuntimeDefault - the container runtime default profile should be used. Unconfined - no profile should be applied." + type: string + windowsOptions: + description: The Windows specific settings applied to all containers. If unspecified, the options from the PodSecurityContext will be used. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is linux. + type: object + properties: + gmsaCredentialSpec: + description: GMSACredentialSpec is where the GMSA admission webhook (https://github.com/kubernetes-sigs/windows-gmsa) inlines the contents of the GMSA credential spec named by the GMSACredentialSpecName field. + type: string + gmsaCredentialSpecName: + description: GMSACredentialSpecName is the name of the GMSA credential spec to use. + type: string + hostProcess: + description: HostProcess determines if a container should be run as a 'Host Process' container. All of a Pod's containers must have the same effective HostProcess value (it is not allowed to have a mix of HostProcess containers and non-HostProcess containers). In addition, if HostProcess is true then HostNetwork must also be set to true. + type: boolean + runAsUserName: + description: The UserName in Windows to run the entrypoint of the container process. Defaults to the user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: string + startupProbe: + description: 'StartupProbe indicates that the Pod has successfully initialized. If specified, no other probes are executed until this completes successfully. If this probe fails, the Pod will be restarted, just as if the livenessProbe failed. This can be used to provide different probe parameters at the beginning of a Pod''s lifecycle, when it might take a long time to load data or warm a cache, than during steady-state operation. This cannot be updated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: object + properties: + exec: + description: Exec specifies the action to take. + type: object + properties: + command: + description: Command is the command line to execute inside the container, the working directory for the command is root ('/') in the container's filesystem. The command is simply exec'd, it is not run inside a shell, so traditional shell instructions ('|', etc) won't work. To use a shell, you need to explicitly call out to that shell. Exit status of 0 is treated as live/healthy and non-zero is unhealthy. + type: array + items: + type: string + failureThreshold: + description: Minimum consecutive failures for the probe to be considered failed after having succeeded. Defaults to 3. Minimum value is 1. + type: integer + format: int32 + grpc: + description: GRPC specifies an action involving a GRPC port. + type: object + required: + - port + properties: + port: + description: Port number of the gRPC service. Number must be in the range 1 to 65535. + type: integer + format: int32 + service: + description: "Service is the name of the service to place in the gRPC HealthCheckRequest (see https://github.com/grpc/grpc/blob/master/doc/health-checking.md). \n If this is not specified, the default behavior is defined by gRPC." + type: string + httpGet: + description: HTTPGet specifies the http request to perform. + type: object + required: + - port + properties: + host: + description: Host name to connect to, defaults to the pod IP. You probably want to set "Host" in httpHeaders instead. + type: string + httpHeaders: + description: Custom headers to set in the request. HTTP allows repeated headers. + type: array + items: + description: HTTPHeader describes a custom header to be used in HTTP probes + type: object + required: + - name + - value + properties: + name: + description: The header field name. This will be canonicalized upon output, so case-variant names will be understood as the same header. + type: string + value: + description: The header field value + type: string + path: + description: Path to access on the HTTP server. + type: string + port: + description: Name or number of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + scheme: + description: Scheme to use for connecting to the host. Defaults to HTTP. + type: string + initialDelaySeconds: + description: 'Number of seconds after the container has started before liveness probes are initiated. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + periodSeconds: + description: How often (in seconds) to perform the probe. Default to 10 seconds. Minimum value is 1. + type: integer + format: int32 + successThreshold: + description: Minimum consecutive successes for the probe to be considered successful after having failed. Defaults to 1. Must be 1 for liveness and startup. Minimum value is 1. + type: integer + format: int32 + tcpSocket: + description: TCPSocket specifies an action involving a TCP port. + type: object + required: + - port + properties: + host: + description: 'Optional: Host name to connect to, defaults to the pod IP.' + type: string + port: + description: Number or name of the port to access on the container. Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME. + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully upon probe failure. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. If this value is nil, the pod's terminationGracePeriodSeconds will be used. Otherwise, this value overrides the value provided by the pod spec. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). This is a beta field and requires enabling ProbeTerminationGracePeriod feature gate. Minimum value is 1. spec.terminationGracePeriodSeconds is used if unset. + type: integer + format: int64 + timeoutSeconds: + description: 'Number of seconds after which the probe times out. Defaults to 1 second. Minimum value is 1. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes' + type: integer + format: int32 + stdin: + description: Whether this container should allocate a buffer for stdin in the container runtime. If this is not set, reads from stdin in the container will always result in EOF. Default is false. + type: boolean + stdinOnce: + description: Whether the container runtime should close the stdin channel after it has been opened by a single attach. When stdin is true the stdin stream will remain open across multiple attach sessions. If stdinOnce is set to true, stdin is opened on container start, is empty until the first client attaches to stdin, and then remains open and accepts data until the client disconnects, at which time stdin is closed and remains closed until the container is restarted. If this flag is false, a container processes that reads from stdin will never receive an EOF. Default is false + type: boolean + terminationMessagePath: + description: 'Optional: Path at which the file to which the container''s termination message will be written is mounted into the container''s filesystem. Message written is intended to be brief final status, such as an assertion failure message. Will be truncated by the node if greater than 4096 bytes. The total message length across all containers will be limited to 12kb. Defaults to /dev/termination-log. Cannot be updated.' + type: string + terminationMessagePolicy: + description: Indicate how the termination message should be populated. File will use the contents of terminationMessagePath to populate the container status message on both success and failure. FallbackToLogsOnError will use the last chunk of container log output if the termination message file is empty and the container exited with an error. The log output is limited to 2048 bytes or 80 lines, whichever is smaller. Defaults to File. Cannot be updated. + type: string + tty: + description: Whether this container should allocate a TTY for itself, also requires 'stdin' to be true. Default is false. + type: boolean + volumeDevices: + description: volumeDevices is the list of block devices to be used by the container. + type: array + items: + description: volumeDevice describes a mapping of a raw block device within a container. + type: object + required: + - devicePath + - name + properties: + devicePath: + description: devicePath is the path inside of the container that the device will be mapped to. + type: string + name: + description: name must match the name of a persistentVolumeClaim in the pod + type: string + volumeMounts: + description: Pod volumes to mount into the container's filesystem. Cannot be updated. + type: array + items: + description: VolumeMount describes a mounting of a Volume within a container. + type: object + required: + - mountPath + - name + properties: + mountPath: + description: Path within the container at which the volume should be mounted. Must not contain ':'. + type: string + mountPropagation: + description: mountPropagation determines how mounts are propagated from the host to container and the other way around. When not set, MountPropagationNone is used. This field is beta in 1.10. + type: string + name: + description: This must match the Name of a Volume. + type: string + readOnly: + description: Mounted read-only if true, read-write otherwise (false or unspecified). Defaults to false. + type: boolean + subPath: + description: Path within the volume from which the container's volume should be mounted. Defaults to "" (volume's root). + type: string + subPathExpr: + description: Expanded path within the volume from which the container's volume should be mounted. Behaves similarly to SubPath but environment variable references $(VAR_NAME) are expanded using the container's environment. Defaults to "" (volume's root). SubPathExpr and SubPath are mutually exclusive. + type: string + workingDir: + description: Container's working directory. If not specified, the container runtime's default will be used, which might be configured in the container image. Cannot be updated. + type: string + nodeName: + description: NodeName is a request to schedule this pod onto a specific node. If it is non-empty, the scheduler simply schedules this pod onto that node, assuming that it fits resource requirements. + type: string + nodeSelector: + description: 'NodeSelector is a selector which must be true for the pod to fit on a node. Selector which must match a node''s labels for the pod to be scheduled on that node. More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/' + type: object + additionalProperties: + type: string + x-kubernetes-map-type: atomic + os: + description: "Specifies the OS of the containers in the pod. Some pod and container fields are restricted if this is set. \n If the OS field is set to linux, the following fields must be unset: -securityContext.windowsOptions \n If the OS field is set to windows, following fields must be unset: - spec.hostPID - spec.hostIPC - spec.hostUsers - spec.securityContext.seLinuxOptions - spec.securityContext.seccompProfile - spec.securityContext.fsGroup - spec.securityContext.fsGroupChangePolicy - spec.securityContext.sysctls - spec.shareProcessNamespace - spec.securityContext.runAsUser - spec.securityContext.runAsGroup - spec.securityContext.supplementalGroups - spec.containers[*].securityContext.seLinuxOptions - spec.containers[*].securityContext.seccompProfile - spec.containers[*].securityContext.capabilities - spec.containers[*].securityContext.readOnlyRootFilesystem - spec.containers[*].securityContext.privileged - spec.containers[*].securityContext.allowPrivilegeEscalation - spec.containers[*].securityContext.procMount - spec.containers[*].securityContext.runAsUser - spec.containers[*].securityContext.runAsGroup" + type: object + required: + - name + properties: + name: + description: 'Name is the name of the operating system. The currently supported values are linux and windows. Additional value may be defined in future and can be one of: https://github.com/opencontainers/runtime-spec/blob/master/config.md#platform-specific-configuration Clients should expect to handle additional values and treat unrecognized values in this field as os: null' + type: string + overhead: + description: 'Overhead represents the resource overhead associated with running a pod for a given RuntimeClass. This field will be autopopulated at admission time by the RuntimeClass admission controller. If the RuntimeClass admission controller is enabled, overhead must not be set in Pod create requests. The RuntimeClass admission controller will reject Pod create requests which have the overhead already set. If RuntimeClass is configured and selected in the PodSpec, Overhead will be set to the value defined in the corresponding RuntimeClass, otherwise it will remain unset and treated as zero. More info: https://git.k8s.io/enhancements/keps/sig-node/688-pod-overhead/README.md' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + preemptionPolicy: + description: PreemptionPolicy is the Policy for preempting pods with lower priority. One of Never, PreemptLowerPriority. Defaults to PreemptLowerPriority if unset. + type: string + priority: + description: The priority value. Various system components use this field to find the priority of the pod. When Priority Admission Controller is enabled, it prevents users from setting this field. The admission controller populates this field from PriorityClassName. The higher the value, the higher the priority. + type: integer + format: int32 + priorityClassName: + description: If specified, indicates the pod's priority. "system-node-critical" and "system-cluster-critical" are two special keywords which indicate the highest priorities with the former being the highest priority. Any other name must be defined by creating a PriorityClass object with that name. If not specified, the pod priority will be default or zero if there is no default. + type: string + readinessGates: + description: 'If specified, all readiness gates will be evaluated for pod readiness. A pod is ready when all its containers are ready AND all conditions specified in the readiness gates have status equal to "True" More info: https://git.k8s.io/enhancements/keps/sig-network/580-pod-readiness-gates' + type: array + items: + description: PodReadinessGate contains the reference to a pod condition + type: object + required: + - conditionType + properties: + conditionType: + description: ConditionType refers to a condition in the pod's condition list with matching type. + type: string + resourceClaims: + description: "ResourceClaims defines which ResourceClaims must be allocated and reserved before the Pod is allowed to start. The resources will be made available to those containers which consume them by name. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable." + type: array + items: + description: PodResourceClaim references exactly one ResourceClaim through a ClaimSource. It adds a name to it that uniquely identifies the ResourceClaim inside the Pod. Containers that need access to the ResourceClaim reference it with this name. + type: object + required: + - name + properties: + name: + description: Name uniquely identifies this resource claim inside the pod. This must be a DNS_LABEL. + type: string + source: + description: Source describes where to find the ResourceClaim. + type: object + properties: + resourceClaimName: + description: ResourceClaimName is the name of a ResourceClaim object in the same namespace as this pod. + type: string + resourceClaimTemplateName: + description: "ResourceClaimTemplateName is the name of a ResourceClaimTemplate object in the same namespace as this pod. \n The template will be used to create a new ResourceClaim, which will be bound to this pod. When this pod is deleted, the ResourceClaim will also be deleted. The pod name and resource name, along with a generated component, will be used to form a unique name for the ResourceClaim, which will be recorded in pod.status.resourceClaimStatuses. \n This field is immutable and no changes will be made to the corresponding ResourceClaim by the control plane after creating the ResourceClaim." + type: string + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + restartPolicy: + description: 'Restart policy for all containers within the pod. One of Always, OnFailure, Never. In some contexts, only a subset of those values may be permitted. Default to Always. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy' + type: string + runtimeClassName: + description: 'RuntimeClassName refers to a RuntimeClass object in the node.k8s.io group, which should be used to run this pod. If no RuntimeClass resource matches the named class, the pod will not be run. If unset or empty, the "legacy" RuntimeClass will be used, which is an implicit class with an empty definition that uses the default runtime handler. More info: https://git.k8s.io/enhancements/keps/sig-node/585-runtime-class' + type: string + schedulerName: + description: If specified, the pod will be dispatched by specified scheduler. If not specified, the pod will be dispatched by default scheduler. + type: string + schedulingGates: + description: "SchedulingGates is an opaque list of values that if specified will block scheduling the pod. If schedulingGates is not empty, the pod will stay in the SchedulingGated state and the scheduler will not attempt to schedule the pod. \n SchedulingGates can only be set at pod creation time, and be removed only afterwards. \n This is a beta feature enabled by the PodSchedulingReadiness feature gate." + type: array + items: + description: PodSchedulingGate is associated to a Pod to guard its scheduling. + type: object + required: + - name + properties: + name: + description: Name of the scheduling gate. Each scheduling gate must have a unique name field. + type: string + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + securityContext: + description: 'SecurityContext holds pod-level security attributes and common container settings. Optional: Defaults to empty. See type description for default values of each field.' + type: object + properties: + fsGroup: + description: "A special supplemental group that applies to all containers in a pod. Some volume types allow the Kubelet to change the ownership of that volume to be owned by the pod: \n 1. The owning GID will be the FSGroup 2. The setgid bit is set (new files created in the volume will be owned by FSGroup) 3. The permission bits are OR'd with rw-rw---- \n If unset, the Kubelet will not modify the ownership and permissions of any volume. Note that this field cannot be set when spec.os.name is windows." + type: integer + format: int64 + fsGroupChangePolicy: + description: 'fsGroupChangePolicy defines behavior of changing ownership and permission of the volume before being exposed inside Pod. This field will only apply to volume types which support fsGroup based ownership(and permissions). It will have no effect on ephemeral volume types such as: secret, configmaps and emptydir. Valid values are "OnRootMismatch" and "Always". If not specified, "Always" is used. Note that this field cannot be set when spec.os.name is windows.' + type: string + runAsGroup: + description: The GID to run the entrypoint of the container process. Uses runtime default if unset. May also be set in SecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence for that container. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + runAsNonRoot: + description: Indicates that the container must run as a non-root user. If true, the Kubelet will validate the image at runtime to ensure that it does not run as UID 0 (root) and fail to start the container if it does. If unset or false, no such validation will be performed. May also be set in SecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: boolean + runAsUser: + description: The UID to run the entrypoint of the container process. Defaults to user specified in image metadata if unspecified. May also be set in SecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence for that container. Note that this field cannot be set when spec.os.name is windows. + type: integer + format: int64 + seLinuxOptions: + description: The SELinux context to be applied to all containers. If unspecified, the container runtime will allocate a random SELinux context for each container. May also be set in SecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence for that container. Note that this field cannot be set when spec.os.name is windows. + type: object + properties: + level: + description: Level is SELinux level label that applies to the container. + type: string + role: + description: Role is a SELinux role label that applies to the container. + type: string + type: + description: Type is a SELinux type label that applies to the container. + type: string + user: + description: User is a SELinux user label that applies to the container. + type: string + seccompProfile: + description: The seccomp options to use by the containers in this pod. Note that this field cannot be set when spec.os.name is windows. + type: object + required: + - type + properties: + localhostProfile: + description: localhostProfile indicates a profile defined in a file on the node should be used. The profile must be preconfigured on the node to work. Must be a descending path, relative to the kubelet's configured seccomp profile location. Must be set if type is "Localhost". Must NOT be set for any other type. + type: string + type: + description: "type indicates which kind of seccomp profile will be applied. Valid options are: \n Localhost - a profile defined in a file on the node should be used. RuntimeDefault - the container runtime default profile should be used. Unconfined - no profile should be applied." + type: string + supplementalGroups: + description: A list of groups applied to the first process run in each container, in addition to the container's primary GID, the fsGroup (if specified), and group memberships defined in the container image for the uid of the container process. If unspecified, no additional groups are added to any container. Note that group memberships defined in the container image for the uid of the container process are still effective, even if they are not included in this list. Note that this field cannot be set when spec.os.name is windows. + type: array + items: + type: integer + format: int64 + sysctls: + description: Sysctls hold a list of namespaced sysctls used for the pod. Pods with unsupported sysctls (by the container runtime) might fail to launch. Note that this field cannot be set when spec.os.name is windows. + type: array + items: + description: Sysctl defines a kernel parameter to be set + type: object + required: + - name + - value + properties: + name: + description: Name of a property to set + type: string + value: + description: Value of a property to set + type: string + windowsOptions: + description: The Windows specific settings applied to all containers. If unspecified, the options within a container's SecurityContext will be used. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. Note that this field cannot be set when spec.os.name is linux. + type: object + properties: + gmsaCredentialSpec: + description: GMSACredentialSpec is where the GMSA admission webhook (https://github.com/kubernetes-sigs/windows-gmsa) inlines the contents of the GMSA credential spec named by the GMSACredentialSpecName field. + type: string + gmsaCredentialSpecName: + description: GMSACredentialSpecName is the name of the GMSA credential spec to use. + type: string + hostProcess: + description: HostProcess determines if a container should be run as a 'Host Process' container. All of a Pod's containers must have the same effective HostProcess value (it is not allowed to have a mix of HostProcess containers and non-HostProcess containers). In addition, if HostProcess is true then HostNetwork must also be set to true. + type: boolean + runAsUserName: + description: The UserName in Windows to run the entrypoint of the container process. Defaults to the user specified in image metadata if unspecified. May also be set in PodSecurityContext. If set in both SecurityContext and PodSecurityContext, the value specified in SecurityContext takes precedence. + type: string + serviceAccount: + description: 'DeprecatedServiceAccount is a depreciated alias for ServiceAccountName. Deprecated: Use serviceAccountName instead.' + type: string + serviceAccountName: + description: 'ServiceAccountName is the name of the ServiceAccount to use to run this pod. More info: https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/' + type: string + setHostnameAsFQDN: + description: If true the pod's hostname will be configured as the pod's FQDN, rather than the leaf name (the default). In Linux containers, this means setting the FQDN in the hostname field of the kernel (the nodename field of struct utsname). In Windows containers, this means setting the registry value of hostname for the registry key HKEY_LOCAL_MACHINE\\SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters to FQDN. If a pod does not have FQDN, this has no effect. Default to false. + type: boolean + shareProcessNamespace: + description: 'Share a single process namespace between all of the containers in a pod. When this is set containers will be able to view and signal processes from other containers in the same pod, and the first process in each container will not be assigned PID 1. HostPID and ShareProcessNamespace cannot both be set. Optional: Default to false.' + type: boolean + subdomain: + description: If specified, the fully qualified Pod hostname will be "...svc.". If not specified, the pod will not have a domainname at all. + type: string + terminationGracePeriodSeconds: + description: Optional duration in seconds the pod needs to terminate gracefully. May be decreased in delete request. Value must be non-negative integer. The value zero indicates stop immediately via the kill signal (no opportunity to shut down). If this value is nil, the default grace period will be used instead. The grace period is the duration in seconds after the processes running in the pod are sent a termination signal and the time when the processes are forcibly halted with a kill signal. Set this value longer than the expected cleanup time for your process. Defaults to 30 seconds. + type: integer + format: int64 + tolerations: + description: If specified, the pod's tolerations. + type: array + items: + description: The pod this Toleration is attached to tolerates any taint that matches the triple using the matching operator . + type: object + 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. + type: integer + format: int64 + 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 + topologySpreadConstraints: + description: TopologySpreadConstraints describes how a group of pods ought to spread across topology domains. Scheduler will schedule pods in a way which abides by the constraints. All topologySpreadConstraints are ANDed. + type: array + items: + description: TopologySpreadConstraint specifies how to spread matching pods among the given topology. + type: object + required: + - maxSkew + - topologyKey + - whenUnsatisfiable + properties: + labelSelector: + description: LabelSelector is used to find matching pods. Pods that match this label selector are counted to determine the number of pods in their corresponding topology domain. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + matchLabelKeys: + description: "MatchLabelKeys is a set of pod label keys to select the pods over which spreading will be calculated. The keys are used to lookup values from the incoming pod labels, those key-value labels are ANDed with labelSelector to select the group of existing pods over which spreading will be calculated for the incoming pod. The same key is forbidden to exist in both MatchLabelKeys and LabelSelector. MatchLabelKeys cannot be set when LabelSelector isn't set. Keys that don't exist in the incoming pod labels will be ignored. A null or empty list means only match against labelSelector. \n This is a beta field and requires the MatchLabelKeysInPodTopologySpread feature gate to be enabled (enabled by default)." + type: array + items: + type: string + x-kubernetes-list-type: atomic + maxSkew: + description: 'MaxSkew describes the degree to which pods may be unevenly distributed. When `whenUnsatisfiable=DoNotSchedule`, it is the maximum permitted difference between the number of matching pods in the target topology and the global minimum. The global minimum is the minimum number of matching pods in an eligible domain or zero if the number of eligible domains is less than MinDomains. For example, in a 3-zone cluster, MaxSkew is set to 1, and pods with the same labelSelector spread as 2/2/1: In this case, the global minimum is 1. | zone1 | zone2 | zone3 | | P P | P P | P | - if MaxSkew is 1, incoming pod can only be scheduled to zone3 to become 2/2/2; scheduling it onto zone1(zone2) would make the ActualSkew(3-1) on zone1(zone2) violate MaxSkew(1). - if MaxSkew is 2, incoming pod can be scheduled onto any zone. When `whenUnsatisfiable=ScheduleAnyway`, it is used to give higher precedence to topologies that satisfy it. It''s a required field. Default value is 1 and 0 is not allowed.' + type: integer + format: int32 + minDomains: + description: "MinDomains indicates a minimum number of eligible domains. When the number of eligible domains with matching topology keys is less than minDomains, Pod Topology Spread treats \"global minimum\" as 0, and then the calculation of Skew is performed. And when the number of eligible domains with matching topology keys equals or greater than minDomains, this value has no effect on scheduling. As a result, when the number of eligible domains is less than minDomains, scheduler won't schedule more than maxSkew Pods to those domains. If value is nil, the constraint behaves as if MinDomains is equal to 1. Valid values are integers greater than 0. When value is not nil, WhenUnsatisfiable must be DoNotSchedule. \n For example, in a 3-zone cluster, MaxSkew is set to 2, MinDomains is set to 5 and pods with the same labelSelector spread as 2/2/2: | zone1 | zone2 | zone3 | | P P | P P | P P | The number of domains is less than 5(MinDomains), so \"global minimum\" is treated as 0. In this situation, new pod with the same labelSelector cannot be scheduled, because computed skew will be 3(3 - 0) if new Pod is scheduled to any of the three zones, it will violate MaxSkew. \n This is a beta field and requires the MinDomainsInPodTopologySpread feature gate to be enabled (enabled by default)." + type: integer + format: int32 + nodeAffinityPolicy: + description: "NodeAffinityPolicy indicates how we will treat Pod's nodeAffinity/nodeSelector when calculating pod topology spread skew. Options are: - Honor: only nodes matching nodeAffinity/nodeSelector are included in the calculations. - Ignore: nodeAffinity/nodeSelector are ignored. All nodes are included in the calculations. \n If this value is nil, the behavior is equivalent to the Honor policy. This is a beta-level feature default enabled by the NodeInclusionPolicyInPodTopologySpread feature flag." + type: string + nodeTaintsPolicy: + description: "NodeTaintsPolicy indicates how we will treat node taints when calculating pod topology spread skew. Options are: - Honor: nodes without taints, along with tainted nodes for which the incoming pod has a toleration, are included. - Ignore: node taints are ignored. All nodes are included. \n If this value is nil, the behavior is equivalent to the Ignore policy. This is a beta-level feature default enabled by the NodeInclusionPolicyInPodTopologySpread feature flag." + type: string + topologyKey: + description: TopologyKey is the key of node labels. Nodes that have a label with this key and identical values are considered to be in the same topology. We consider each as a "bucket", and try to put balanced number of pods into each bucket. We define a domain as a particular instance of a topology. Also, we define an eligible domain as a domain whose nodes meet the requirements of nodeAffinityPolicy and nodeTaintsPolicy. e.g. If TopologyKey is "kubernetes.io/hostname", each Node is a domain of that topology. And, if TopologyKey is "topology.kubernetes.io/zone", each zone is a domain of that topology. It's a required field. + type: string + whenUnsatisfiable: + description: 'WhenUnsatisfiable indicates how to deal with a pod if it doesn''t satisfy the spread constraint. - DoNotSchedule (default) tells the scheduler not to schedule it. - ScheduleAnyway tells the scheduler to schedule the pod in any location, but giving higher precedence to topologies that would help reduce the skew. A constraint is considered "Unsatisfiable" for an incoming pod if and only if every possible node assignment for that pod would violate "MaxSkew" on some topology. For example, in a 3-zone cluster, MaxSkew is set to 1, and pods with the same labelSelector spread as 3/1/1: | zone1 | zone2 | zone3 | | P P P | P | P | If WhenUnsatisfiable is set to DoNotSchedule, incoming pod can only be scheduled to zone2(zone3) to become 3/2/1(3/1/2) as ActualSkew(2-1) on zone2(zone3) satisfies MaxSkew(1). In other words, the cluster can still be imbalanced, but scheduler won''t make it *more* imbalanced. It''s a required field.' + type: string + x-kubernetes-list-map-keys: + - topologyKey + - whenUnsatisfiable + x-kubernetes-list-type: map + volumes: + description: 'List of volumes that can be mounted by containers belonging to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes' + type: array + items: + description: Volume represents a named volume in a pod that may be accessed by any container in the pod. + type: object + required: + - name + properties: + awsElasticBlockStore: + description: 'awsElasticBlockStore represents an AWS Disk resource that is attached to a kubelet''s host machine and then exposed to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore' + type: object + required: + - volumeID + properties: + fsType: + description: 'fsType is the filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + partition: + description: 'partition is the partition in the volume that you want to mount. If omitted, the default is to mount by volume name. Examples: For volume /dev/sda1, you specify the partition as "1". Similarly, the volume partition for /dev/sda is "0" (or you can leave the property empty).' + type: integer + format: int32 + readOnly: + description: 'readOnly value true will force the readOnly setting in VolumeMounts. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore' + type: boolean + volumeID: + description: 'volumeID is unique ID of the persistent disk resource in AWS (Amazon EBS volume). More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore' + type: string + azureDisk: + description: azureDisk represents an Azure Data Disk mount on the host and bind mount to the pod. + type: object + required: + - diskName + - diskURI + properties: + cachingMode: + description: 'cachingMode is the Host Caching mode: None, Read Only, Read Write.' + type: string + diskName: + description: diskName is the Name of the data disk in the blob storage + type: string + diskURI: + description: diskURI is the URI of data disk in the blob storage + type: string + fsType: + description: fsType is Filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + kind: + description: 'kind expected values are Shared: multiple blob disks per storage account Dedicated: single blob disk per storage account Managed: azure managed data disk (only in managed availability set). defaults to shared' + type: string + readOnly: + description: readOnly Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + azureFile: + description: azureFile represents an Azure File Service mount on the host and bind mount to the pod. + type: object + required: + - secretName + - shareName + properties: + readOnly: + description: readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + secretName: + description: secretName is the name of secret that contains Azure Storage Account Name and Key + type: string + shareName: + description: shareName is the azure share Name + type: string + cephfs: + description: cephFS represents a Ceph FS mount on the host that shares a pod's lifetime + type: object + required: + - monitors + properties: + monitors: + description: 'monitors is Required: Monitors is a collection of Ceph monitors More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: array + items: + type: string + path: + description: 'path is Optional: Used as the mounted root, rather than the full Ceph tree, default is /' + type: string + readOnly: + description: 'readOnly is Optional: Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: boolean + secretFile: + description: 'secretFile is Optional: SecretFile is the path to key ring for User, default is /etc/ceph/user.secret More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: string + secretRef: + description: 'secretRef is Optional: SecretRef is reference to the authentication secret for User, default is empty. More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + user: + description: 'user is optional: User is the rados user name, default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: string + cinder: + description: 'cinder represents a cinder volume attached and mounted on kubelets host machine. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: object + required: + - volumeID + properties: + fsType: + description: 'fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: string + readOnly: + description: 'readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: boolean + secretRef: + description: 'secretRef is optional: points to a secret object containing parameters used to connect to OpenStack.' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + volumeID: + description: 'volumeID used to identify the volume in cinder. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: string + configMap: + description: configMap represents a configMap that should populate this volume + type: object + properties: + defaultMode: + description: 'defaultMode is optional: mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Defaults to 0644. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + items: + description: items if unspecified, each key-value pair in the Data field of the referenced ConfigMap will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the ConfigMap, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: optional specify whether the ConfigMap or its keys must be defined + type: boolean + csi: + description: csi (Container Storage Interface) represents ephemeral storage that is handled by certain external CSI drivers (Beta feature). + type: object + required: + - driver + properties: + driver: + description: driver is the name of the CSI driver that handles this volume. Consult with your admin for the correct name as registered in the cluster. + type: string + fsType: + description: fsType to mount. Ex. "ext4", "xfs", "ntfs". If not provided, the empty value is passed to the associated CSI driver which will determine the default filesystem to apply. + type: string + nodePublishSecretRef: + description: nodePublishSecretRef is a reference to the secret object containing sensitive information to pass to the CSI driver to complete the CSI NodePublishVolume and NodeUnpublishVolume calls. This field is optional, and may be empty if no secret is required. If the secret object contains more than one secret, all secret references are passed. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + readOnly: + description: readOnly specifies a read-only configuration for the volume. Defaults to false (read/write). + type: boolean + volumeAttributes: + description: volumeAttributes stores driver-specific properties that are passed to the CSI driver. Consult your driver's documentation for supported values. + type: object + additionalProperties: + type: string + downwardAPI: + description: downwardAPI represents downward API about the pod that should populate this volume + type: object + properties: + defaultMode: + description: 'Optional: mode bits to use on created files by default. Must be a Optional: mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Defaults to 0644. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + items: + description: Items is a list of downward API volume file + type: array + items: + description: DownwardAPIVolumeFile represents information to create the file containing the pod field + type: object + required: + - path + properties: + fieldRef: + description: 'Required: Selects a field of the pod: only annotations, labels, name and namespace are supported.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + mode: + description: 'Optional: mode bits used to set permissions on this file, must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: 'Required: Path is the relative path name of the file to be created. Must not be absolute or contain the ''..'' path. Must be utf-8 encoded. The first item of the relative path must not start with ''..''' + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, requests.cpu and requests.memory) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + emptyDir: + description: 'emptyDir represents a temporary directory that shares a pod''s lifetime. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir' + type: object + properties: + medium: + description: 'medium represents what type of storage medium should back this directory. The default is "" which means to use the node''s default medium. Must be an empty string (default) or Memory. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir' + type: string + sizeLimit: + description: 'sizeLimit is the total amount of local storage required for this EmptyDir volume. The size limit is also applicable for memory medium. 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. The default is nil which means that the limit is undefined. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir' + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + ephemeral: + description: "ephemeral represents a volume that is handled by a cluster storage driver. The volume's lifecycle is tied to the pod that defines it - it will be created before the pod starts, and deleted when the pod is removed. \n Use this if: a) the volume is only needed while the pod runs, b) features of normal volumes like restoring from snapshot or capacity tracking are needed, c) the storage driver is specified through a storage class, and d) the storage driver supports dynamic volume provisioning through a PersistentVolumeClaim (see EphemeralVolumeSource for more information on the connection between this volume type and PersistentVolumeClaim). \n Use PersistentVolumeClaim or one of the vendor-specific APIs for volumes that persist for longer than the lifecycle of an individual pod. \n Use CSI for light-weight local ephemeral volumes if the CSI driver is meant to be used that way - see the documentation of the driver for more information. \n A pod can use both types of ephemeral volumes and persistent volumes at the same time." + type: object + properties: + volumeClaimTemplate: + description: "Will be used to create a stand-alone PVC to provision the volume. The pod in which this EphemeralVolumeSource is embedded will be the owner of the PVC, i.e. the PVC will be deleted together with the pod. The name of the PVC will be `-` where `` is the name from the `PodSpec.Volumes` array entry. Pod validation will reject the pod if the concatenated name is not valid for a PVC (for example, too long). \n An existing PVC with that name that is not owned by the pod will *not* be used for the pod to avoid using an unrelated volume by mistake. Starting the pod is then blocked until the unrelated PVC is removed. If such a pre-created PVC is meant to be used by the pod, the PVC has to updated with an owner reference to the pod once the pod exists. Normally this should not be necessary, but it may be useful when manually reconstructing a broken cluster. \n This field is read-only and no changes will be made by Kubernetes to the PVC after it has been created. \n Required, must not be nil." + type: object + required: + - spec + properties: + metadata: + description: May contain labels and annotations that will be copied into the PVC when creating it. No other fields are allowed and will be rejected during validation. + type: object + spec: + description: The specification for the PersistentVolumeClaim. The entire content is copied unchanged into the PVC that gets created from this template. The same fields as in a PersistentVolumeClaim are also valid here. + type: object + properties: + accessModes: + description: 'accessModes contains the desired access modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1' + type: array + items: + type: string + dataSource: + description: 'dataSource field can be used to specify either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot) * An existing PVC (PersistentVolumeClaim) If the provisioner or an external controller can support the specified data source, it will create a new volume based on the contents of the specified data source. When the AnyVolumeDataSource feature gate is enabled, dataSource contents will be copied to dataSourceRef, and dataSourceRef contents will be copied to dataSource when dataSourceRef.namespace is not specified. If the namespace is specified, then dataSourceRef will not be copied to dataSource.' + type: object + required: + - kind + - name + properties: + apiGroup: + description: APIGroup is the group for the resource being referenced. If APIGroup is not specified, the specified Kind must be in the core API group. For any other third-party types, APIGroup is required. + type: string + kind: + description: Kind is the type of resource being referenced + type: string + name: + description: Name is the name of resource being referenced + type: string + dataSourceRef: + description: 'dataSourceRef specifies the object from which to populate the volume with data, if a non-empty volume is desired. This may be any object from a non-empty API group (non core object) or a PersistentVolumeClaim object. When this field is specified, volume binding will only succeed if the type of the specified object matches some installed volume populator or dynamic provisioner. This field will replace the functionality of the dataSource field and as such if both fields are non-empty, they must have the same value. For backwards compatibility, when namespace isn''t specified in dataSourceRef, both fields (dataSource and dataSourceRef) will be set to the same value automatically if one of them is empty and the other is non-empty. When namespace is specified in dataSourceRef, dataSource isn''t set to the same value and must be empty. There are three important differences between dataSource and dataSourceRef: * While dataSource only allows two specific types of objects, dataSourceRef allows any non-core object, as well as PersistentVolumeClaim objects. * While dataSource ignores disallowed values (dropping them), dataSourceRef preserves all values, and generates an error if a disallowed value is specified. * While dataSource only allows local objects, dataSourceRef allows objects in any namespaces. (Beta) Using this field requires the AnyVolumeDataSource feature gate to be enabled. (Alpha) Using the namespace field of dataSourceRef requires the CrossNamespaceVolumeDataSource feature gate to be enabled.' + type: object + required: + - kind + - name + properties: + apiGroup: + description: APIGroup is the group for the resource being referenced. If APIGroup is not specified, the specified Kind must be in the core API group. For any other third-party types, APIGroup is required. + type: string + kind: + description: Kind is the type of resource being referenced + type: string + name: + description: Name is the name of resource being referenced + type: string + namespace: + description: Namespace is the namespace of resource being referenced Note that when a namespace is specified, a gateway.networking.k8s.io/ReferenceGrant object is required in the referent namespace to allow that namespace's owner to accept the reference. See the ReferenceGrant documentation for details. (Alpha) This field requires the CrossNamespaceVolumeDataSource feature gate to be enabled. + type: string + resources: + description: 'resources represents the minimum resources the volume should have. If RecoverVolumeExpansionFailure feature is enabled users are allowed to specify resource requirements that are lower than previous value but must still be higher than capacity recorded in the status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources' + type: object + properties: + claims: + description: "Claims lists the names of resources, defined in spec.resourceClaims, that are used by this container. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable. It can only be set for containers." + type: array + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + type: object + required: + - name + 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 + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + description: 'Limits describes the maximum amount of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + requests: + 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 + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + selector: + description: selector is a label query over volumes to consider for binding. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + storageClassName: + description: 'storageClassName is the name of the StorageClass required by the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1' + type: string + volumeMode: + description: volumeMode defines what type of volume is required by the claim. Value of Filesystem is implied when not included in claim spec. + type: string + volumeName: + description: volumeName is the binding reference to the PersistentVolume backing this claim. + type: string + fc: + description: fc represents a Fibre Channel resource that is attached to a kubelet's host machine and then exposed to the pod. + type: object + properties: + fsType: + description: 'fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + lun: + description: 'lun is Optional: FC target lun number' + type: integer + format: int32 + readOnly: + description: 'readOnly is Optional: Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts.' + type: boolean + targetWWNs: + description: 'targetWWNs is Optional: FC target worldwide names (WWNs)' + type: array + items: + type: string + wwids: + description: 'wwids Optional: FC volume world wide identifiers (wwids) Either wwids or combination of targetWWNs and lun must be set, but not both simultaneously.' + type: array + items: + type: string + flexVolume: + description: flexVolume represents a generic volume resource that is provisioned/attached using an exec based plugin. + type: object + required: + - driver + properties: + driver: + description: driver is the name of the driver to use for this volume. + type: string + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". The default filesystem depends on FlexVolume script. + type: string + options: + description: 'options is Optional: this field holds extra command options if any.' + type: object + additionalProperties: + type: string + readOnly: + description: 'readOnly is Optional: defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts.' + type: boolean + secretRef: + description: 'secretRef is Optional: secretRef is reference to the secret object containing sensitive information to pass to the plugin scripts. This may be empty if no secret object is specified. If the secret object contains more than one secret, all secrets are passed to the plugin scripts.' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + flocker: + description: flocker represents a Flocker volume attached to a kubelet's host machine. This depends on the Flocker control service being running + type: object + properties: + datasetName: + description: datasetName is Name of the dataset stored as metadata -> name on the dataset for Flocker should be considered as deprecated + type: string + datasetUUID: + description: datasetUUID is the UUID of the dataset. This is unique identifier of a Flocker dataset + type: string + gcePersistentDisk: + description: 'gcePersistentDisk represents a GCE Disk resource that is attached to a kubelet''s host machine and then exposed to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: object + required: + - pdName + properties: + fsType: + description: 'fsType is filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + partition: + description: 'partition is the partition in the volume that you want to mount. If omitted, the default is to mount by volume name. Examples: For volume /dev/sda1, you specify the partition as "1". Similarly, the volume partition for /dev/sda is "0" (or you can leave the property empty). More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: integer + format: int32 + pdName: + description: 'pdName is unique name of the PD resource in GCE. Used to identify the disk in GCE. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: string + readOnly: + description: 'readOnly here will force the ReadOnly setting in VolumeMounts. Defaults to false. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: boolean + gitRepo: + description: 'gitRepo represents a git repository at a particular revision. DEPRECATED: GitRepo is deprecated. To provision a container with a git repo, mount an EmptyDir into an InitContainer that clones the repo using git, then mount the EmptyDir into the Pod''s container.' + type: object + required: + - repository + properties: + directory: + description: directory is the target directory name. Must not contain or start with '..'. If '.' is supplied, the volume directory will be the git repository. Otherwise, if specified, the volume will contain the git repository in the subdirectory with the given name. + type: string + repository: + description: repository is the URL + type: string + revision: + description: revision is the commit hash for the specified revision. + type: string + glusterfs: + description: 'glusterfs represents a Glusterfs mount on the host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/glusterfs/README.md' + type: object + required: + - endpoints + - path + properties: + endpoints: + description: 'endpoints is the endpoint name that details Glusterfs topology. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod' + type: string + path: + description: 'path is the Glusterfs volume path. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod' + type: string + readOnly: + description: 'readOnly here will force the Glusterfs volume to be mounted with read-only permissions. Defaults to false. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod' + type: boolean + hostPath: + description: 'hostPath represents a pre-existing file or directory on the host machine that is directly exposed to the container. This is generally used for system agents or other privileged things that are allowed to see the host machine. Most containers will NOT need this. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath --- TODO(jonesdl) We need to restrict who can use host directory mounts and who can/can not mount host directories as read/write.' + type: object + required: + - path + properties: + path: + description: 'path of the directory on the host. If the path is a symlink, it will follow the link to the real path. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath' + type: string + type: + description: 'type for HostPath Volume Defaults to "" More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath' + type: string + iscsi: + description: 'iscsi represents an ISCSI Disk resource that is attached to a kubelet''s host machine and then exposed to the pod. More info: https://examples.k8s.io/volumes/iscsi/README.md' + type: object + required: + - iqn + - lun + - targetPortal + properties: + chapAuthDiscovery: + description: chapAuthDiscovery defines whether support iSCSI Discovery CHAP authentication + type: boolean + chapAuthSession: + description: chapAuthSession defines whether support iSCSI Session CHAP authentication + type: boolean + fsType: + description: 'fsType is the filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#iscsi TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + initiatorName: + description: initiatorName is the custom iSCSI Initiator Name. If initiatorName is specified with iscsiInterface simultaneously, new iSCSI interface : will be created for the connection. + type: string + iqn: + description: iqn is the target iSCSI Qualified Name. + type: string + iscsiInterface: + description: iscsiInterface is the interface Name that uses an iSCSI transport. Defaults to 'default' (tcp). + type: string + lun: + description: lun represents iSCSI Target Lun number. + type: integer + format: int32 + portals: + description: portals is the iSCSI Target Portal List. The portal is either an IP or ip_addr:port if the port is other than default (typically TCP ports 860 and 3260). + type: array + items: + type: string + readOnly: + description: readOnly here will force the ReadOnly setting in VolumeMounts. Defaults to false. + type: boolean + secretRef: + description: secretRef is the CHAP Secret for iSCSI target and initiator authentication + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + targetPortal: + description: targetPortal is iSCSI Target Portal. The Portal is either an IP or ip_addr:port if the port is other than default (typically TCP ports 860 and 3260). + type: string + name: + description: 'name of the volume. Must be a DNS_LABEL and unique within the pod. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + nfs: + description: 'nfs represents an NFS mount on the host that shares a pod''s lifetime More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: object + required: + - path + - server + properties: + path: + description: 'path that is exported by the NFS server. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: string + readOnly: + description: 'readOnly here will force the NFS export to be mounted with read-only permissions. Defaults to false. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: boolean + server: + description: 'server is the hostname or IP address of the NFS server. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: string + persistentVolumeClaim: + description: 'persistentVolumeClaimVolumeSource represents a reference to a PersistentVolumeClaim in the same namespace. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims' + type: object + required: + - claimName + properties: + claimName: + description: 'claimName is the name of a PersistentVolumeClaim in the same namespace as the pod using this volume. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims' + type: string + readOnly: + description: readOnly Will force the ReadOnly setting in VolumeMounts. Default false. + type: boolean + photonPersistentDisk: + description: photonPersistentDisk represents a PhotonController persistent disk attached and mounted on kubelets host machine + type: object + required: + - pdID + properties: + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + pdID: + description: pdID is the ID that identifies Photon Controller persistent disk + type: string + portworxVolume: + description: portworxVolume represents a portworx volume attached and mounted on kubelets host machine + type: object + required: + - volumeID + properties: + fsType: + description: fSType represents the filesystem type to mount Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs". Implicitly inferred to be "ext4" if unspecified. + type: string + readOnly: + description: readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + volumeID: + description: volumeID uniquely identifies a Portworx volume + type: string + projected: + description: projected items for all in one resources secrets, configmaps, and downward API + type: object + properties: + defaultMode: + description: defaultMode are the mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set. + type: integer + format: int32 + sources: + description: sources is the list of volume projections + type: array + items: + description: Projection that may be projected along with other supported volume types + type: object + properties: + configMap: + description: configMap information about the configMap data to project + type: object + properties: + items: + description: items if unspecified, each key-value pair in the Data field of the referenced ConfigMap will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the ConfigMap, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: optional specify whether the ConfigMap or its keys must be defined + type: boolean + downwardAPI: + description: downwardAPI information about the downwardAPI data to project + type: object + properties: + items: + description: Items is a list of DownwardAPIVolume file + type: array + items: + description: DownwardAPIVolumeFile represents information to create the file containing the pod field + type: object + required: + - path + properties: + fieldRef: + description: 'Required: Selects a field of the pod: only annotations, labels, name and namespace are supported.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + mode: + description: 'Optional: mode bits used to set permissions on this file, must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: 'Required: Path is the relative path name of the file to be created. Must not be absolute or contain the ''..'' path. Must be utf-8 encoded. The first item of the relative path must not start with ''..''' + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, requests.cpu and requests.memory) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + secret: + description: secret information about the secret data to project + type: object + properties: + items: + description: items if unspecified, each key-value pair in the Data field of the referenced Secret will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the Secret, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: optional field specify whether the Secret or its key must be defined + type: boolean + serviceAccountToken: + description: serviceAccountToken is information about the serviceAccountToken data to project + type: object + required: + - path + properties: + audience: + description: audience is the intended audience of the token. A recipient of a token must identify itself with an identifier specified in the audience of the token, and otherwise should reject the token. The audience defaults to the identifier of the apiserver. + type: string + expirationSeconds: + description: expirationSeconds is the requested duration of validity of the service account token. As the token approaches expiration, the kubelet volume plugin will proactively rotate the service account token. The kubelet will start trying to rotate the token if the token is older than 80 percent of its time to live or if the token is older than 24 hours.Defaults to 1 hour and must be at least 10 minutes. + type: integer + format: int64 + path: + description: path is the path relative to the mount point of the file to project the token into. + type: string + quobyte: + description: quobyte represents a Quobyte mount on the host that shares a pod's lifetime + type: object + required: + - registry + - volume + properties: + group: + description: group to map volume access to Default is no group + type: string + readOnly: + description: readOnly here will force the Quobyte volume to be mounted with read-only permissions. Defaults to false. + type: boolean + registry: + description: registry represents a single or multiple Quobyte Registry services specified as a string as host:port pair (multiple entries are separated with commas) which acts as the central registry for volumes + type: string + tenant: + description: tenant owning the given Quobyte volume in the Backend Used with dynamically provisioned Quobyte volumes, value is set by the plugin + type: string + user: + description: user to map volume access to Defaults to serivceaccount user + type: string + volume: + description: volume is a string that references an already created Quobyte volume by name. + type: string + rbd: + description: 'rbd represents a Rados Block Device mount on the host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/rbd/README.md' + type: object + required: + - image + - monitors + properties: + fsType: + description: 'fsType is the filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#rbd TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + image: + description: 'image is the rados image name. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + keyring: + description: 'keyring is the path to key ring for RBDUser. Default is /etc/ceph/keyring. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + monitors: + description: 'monitors is a collection of Ceph monitors. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: array + items: + type: string + pool: + description: 'pool is the rados pool name. Default is rbd. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + readOnly: + description: 'readOnly here will force the ReadOnly setting in VolumeMounts. Defaults to false. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: boolean + secretRef: + description: 'secretRef is name of the authentication secret for RBDUser. If provided overrides keyring. Default is nil. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + user: + description: 'user is the rados user name. Default is admin. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + scaleIO: + description: scaleIO represents a ScaleIO persistent volume attached and mounted on Kubernetes nodes. + type: object + required: + - gateway + - secretRef + - system + properties: + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Default is "xfs". + type: string + gateway: + description: gateway is the host address of the ScaleIO API Gateway. + type: string + protectionDomain: + description: protectionDomain is the name of the ScaleIO Protection Domain for the configured storage. + type: string + readOnly: + description: readOnly Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + secretRef: + description: secretRef references to the secret for ScaleIO user and other sensitive information. If this is not provided, Login operation will fail. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + sslEnabled: + description: sslEnabled Flag enable/disable SSL communication with Gateway, default false + type: boolean + storageMode: + description: storageMode indicates whether the storage for a volume should be ThickProvisioned or ThinProvisioned. Default is ThinProvisioned. + type: string + storagePool: + description: storagePool is the ScaleIO Storage Pool associated with the protection domain. + type: string + system: + description: system is the name of the storage system as configured in ScaleIO. + type: string + volumeName: + description: volumeName is the name of a volume already created in the ScaleIO system that is associated with this volume source. + type: string + secret: + description: 'secret represents a secret that should populate this volume. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret' + type: object + properties: + defaultMode: + description: 'defaultMode is Optional: mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Defaults to 0644. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + items: + description: items If unspecified, each key-value pair in the Data field of the referenced Secret will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the Secret, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + optional: + description: optional field specify whether the Secret or its keys must be defined + type: boolean + secretName: + description: 'secretName is the name of the secret in the pod''s namespace to use. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret' + type: string + storageos: + description: storageOS represents a StorageOS volume attached and mounted on Kubernetes nodes. + type: object + properties: + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + readOnly: + description: readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + secretRef: + description: secretRef specifies the secret to use for obtaining the StorageOS API credentials. If not specified, default values will be attempted. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + volumeName: + description: volumeName is the human-readable name of the StorageOS volume. Volume names are only unique within a namespace. + type: string + volumeNamespace: + description: volumeNamespace specifies the scope of the volume within StorageOS. If no namespace is specified then the Pod's namespace will be used. This allows the Kubernetes name scoping to be mirrored within StorageOS for tighter integration. Set VolumeName to any name to override the default behaviour. Set to "default" if you are not using namespaces within StorageOS. Namespaces that do not pre-exist within StorageOS will be created. + type: string + vsphereVolume: + description: vsphereVolume represents a vSphere volume attached and mounted on kubelets host machine + type: object + required: + - volumePath + properties: + fsType: + description: fsType is filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + storagePolicyID: + description: storagePolicyID is the storage Policy Based Management (SPBM) profile ID associated with the StoragePolicyName. + type: string + storagePolicyName: + description: storagePolicyName is the storage Policy Based Management (SPBM) profile name. + type: string + volumePath: + description: volumePath is the path that identifies vSphere volume vmdk + type: string + permissions: + type: array + items: + description: StrategyDeploymentPermissions describe the rbac rules and service account needed by the install strategy + type: object + required: + - rules + - serviceAccountName + properties: + rules: + type: array + items: + description: PolicyRule holds information that describes a policy rule, but does not contain information about who the rule applies to or which namespace the rule applies to. + type: object + required: + - verbs + properties: + apiGroups: + description: APIGroups is the name of the APIGroup that contains the resources. If multiple API groups are specified, any action requested against one of the enumerated resources in any API group will be allowed. "" represents the core API group and "*" represents all API groups. + type: array + items: + type: string + nonResourceURLs: + description: NonResourceURLs is a set of partial urls that a user should have access to. *s are allowed, but only as the full, final step in the path Since non-resource URLs are not namespaced, this field is only applicable for ClusterRoles referenced from a ClusterRoleBinding. Rules can either apply to API resources (such as "pods" or "secrets") or non-resource URL paths (such as "/api"), but not both. + type: array + items: + type: string + resourceNames: + description: ResourceNames is an optional white list of names that the rule applies to. An empty set means that everything is allowed. + type: array + items: + type: string + resources: + description: Resources is a list of resources this rule applies to. '*' represents all resources. + type: array + items: + type: string + verbs: + description: Verbs is a list of Verbs that apply to ALL the ResourceKinds contained in this rule. '*' represents all verbs. + type: array + items: + type: string + serviceAccountName: + type: string + strategy: + type: string + installModes: + description: InstallModes specify supported installation types + type: array + items: + description: InstallMode associates an InstallModeType with a flag representing if the CSV supports it + type: object + required: + - supported + - type + properties: + supported: + type: boolean + type: + description: InstallModeType is a supported type of install mode for CSV installation + type: string + keywords: + description: A list of keywords describing the operator. + type: array + items: + type: string + labels: + description: Map of string keys and values that can be used to organize and categorize (scope and select) objects. + type: object + additionalProperties: + type: string + links: + description: A list of links related to the operator. + type: array + items: + type: object + properties: + name: + type: string + url: + type: string + maintainers: + description: A list of organizational entities maintaining the operator. + type: array + items: + type: object + properties: + email: + type: string + name: + type: string + maturity: + type: string + minKubeVersion: + type: string + nativeAPIs: + type: array + items: + description: GroupVersionKind unambiguously identifies a kind. It doesn't anonymously include GroupVersion to avoid automatic coercion. It doesn't use a GroupVersion to avoid custom marshalling + type: object + required: + - group + - kind + - version + properties: + group: + type: string + kind: + type: string + version: + type: string + provider: + description: The publishing entity behind the operator. + type: object + properties: + name: + type: string + url: + type: string + relatedImages: + description: List any related images, or other container images that your Operator might require to perform their functions. This list should also include operand images as well. All image references should be specified by digest (SHA) and not by tag. This field is only used during catalog creation and plays no part in cluster runtime. + type: array + items: + type: object + required: + - image + - name + properties: + image: + type: string + name: + type: string + replaces: + description: The name of a CSV this one replaces. Should match the `metadata.Name` field of the old CSV. + type: string + selector: + description: Label selector for related resources. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + skips: + description: The name(s) of one or more CSV(s) that should be skipped in the upgrade graph. Should match the `metadata.Name` field of the CSV that should be skipped. This field is only used during catalog creation and plays no part in cluster runtime. + type: array + items: + type: string + version: + type: string + webhookdefinitions: + type: array + items: + description: WebhookDescription provides details to OLM about required webhooks + type: object + required: + - admissionReviewVersions + - generateName + - sideEffects + - type + properties: + admissionReviewVersions: + type: array + items: + type: string + containerPort: + type: integer + format: int32 + default: 443 + maximum: 65535 + minimum: 1 + conversionCRDs: + type: array + items: + type: string + deploymentName: + type: string + failurePolicy: + description: FailurePolicyType specifies a failure policy that defines how unrecognized errors from the admission endpoint are handled. + type: string + generateName: + type: string + matchPolicy: + description: MatchPolicyType specifies the type of match policy. + type: string + objectSelector: + description: A label selector is a label query over a set of resources. The result of matchLabels and matchExpressions are ANDed. An empty label selector matches all objects. A null label selector matches no objects. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + reinvocationPolicy: + description: ReinvocationPolicyType specifies what type of policy the admission hook uses. + type: string + rules: + type: array + items: + description: RuleWithOperations is a tuple of Operations and Resources. It is recommended to make sure that all the tuple expansions are valid. + type: object + properties: + apiGroups: + description: APIGroups is the API groups the resources belong to. '*' is all groups. If '*' is present, the length of the slice must be one. Required. + type: array + items: + type: string + x-kubernetes-list-type: atomic + apiVersions: + description: APIVersions is the API versions the resources belong to. '*' is all versions. If '*' is present, the length of the slice must be one. Required. + type: array + items: + type: string + x-kubernetes-list-type: atomic + operations: + description: Operations is the operations the admission hook cares about - CREATE, UPDATE, DELETE, CONNECT or * for all of those operations and any future admission operations that are added. If '*' is present, the length of the slice must be one. Required. + type: array + items: + description: OperationType specifies an operation for a request. + type: string + x-kubernetes-list-type: atomic + resources: + description: "Resources is a list of resources this rule applies to. \n For example: 'pods' means pods. 'pods/log' means the log subresource of pods. '*' means all resources, but not subresources. 'pods/*' means all subresources of pods. '*/scale' means all scale subresources. '*/*' means all resources and their subresources. \n If wildcard is present, the validation rule will ensure resources do not overlap with each other. \n Depending on the enclosing object, subresources might not be allowed. Required." + type: array + items: + type: string + x-kubernetes-list-type: atomic + scope: + description: scope specifies the scope of this rule. Valid values are "Cluster", "Namespaced", and "*" "Cluster" means that only cluster-scoped resources will match this rule. Namespace API objects are cluster-scoped. "Namespaced" means that only namespaced resources will match this rule. "*" means that there are no scope restrictions. Subresources match the scope of their parent resource. Default is "*". + type: string + sideEffects: + description: SideEffectClass specifies the types of side effects a webhook may have. + type: string + targetPort: + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + timeoutSeconds: + type: integer + format: int32 + type: + description: WebhookAdmissionType is the type of admission webhooks supported by OLM + type: string + enum: + - ValidatingAdmissionWebhook + - MutatingAdmissionWebhook + - ConversionWebhook + webhookPath: + type: string + status: + description: ClusterServiceVersionStatus represents information about the status of a CSV. Status may trail the actual state of a system. + type: object + properties: + certsLastUpdated: + description: Last time the owned APIService certs were updated + type: string + format: date-time + certsRotateAt: + description: Time the owned APIService certs will rotate next + type: string + format: date-time + cleanup: + description: CleanupStatus represents information about the status of cleanup while a CSV is pending deletion + type: object + properties: + pendingDeletion: + description: PendingDeletion is the list of custom resource objects that are pending deletion and blocked on finalizers. This indicates the progress of cleanup that is blocking CSV deletion or operator uninstall. + type: array + items: + description: ResourceList represents a list of resources which are of the same Group/Kind + type: object + required: + - group + - instances + - kind + properties: + group: + type: string + instances: + type: array + items: + type: object + required: + - name + properties: + name: + type: string + namespace: + description: Namespace can be empty for cluster-scoped resources + type: string + kind: + type: string + conditions: + description: List of conditions, a history of state transitions + type: array + items: + description: Conditions appear in the status as a record of state transitions on the ClusterServiceVersion + type: object + properties: + lastTransitionTime: + description: Last time the status transitioned from one status to another. + type: string + format: date-time + lastUpdateTime: + description: Last time we updated the status + type: string + format: date-time + message: + description: A human readable message indicating details about why the ClusterServiceVersion is in this condition. + type: string + phase: + description: Condition of the ClusterServiceVersion + type: string + reason: + description: A brief CamelCase message indicating details about why the ClusterServiceVersion is in this state. e.g. 'RequirementsNotMet' + type: string + lastTransitionTime: + description: Last time the status transitioned from one status to another. + type: string + format: date-time + lastUpdateTime: + description: Last time we updated the status + type: string + format: date-time + message: + description: A human readable message indicating details about why the ClusterServiceVersion is in this condition. + type: string + phase: + description: Current condition of the ClusterServiceVersion + type: string + reason: + description: A brief CamelCase message indicating details about why the ClusterServiceVersion is in this state. e.g. 'RequirementsNotMet' + type: string + requirementStatus: + description: The status of each requirement for this CSV + type: array + items: + type: object + required: + - group + - kind + - message + - name + - status + - version + properties: + dependents: + type: array + items: + description: DependentStatus is the status for a dependent requirement (to prevent infinite nesting) + type: object + required: + - group + - kind + - status + - version + properties: + group: + type: string + kind: + type: string + message: + type: string + status: + description: StatusReason is a camelcased reason for the status of a RequirementStatus or DependentStatus + type: string + uuid: + type: string + version: + type: string + group: + type: string + kind: + type: string + message: + type: string + name: + type: string + status: + description: StatusReason is a camelcased reason for the status of a RequirementStatus or DependentStatus + type: string + uuid: + type: string + version: + type: string + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: installplans.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: InstallPlan + listKind: InstallPlanList + plural: installplans + shortNames: + - ip + singular: installplan + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: The first CSV in the list of clusterServiceVersionNames + jsonPath: .spec.clusterServiceVersionNames[0] + name: CSV + type: string + - description: The approval mode + jsonPath: .spec.approval + name: Approval + type: string + - jsonPath: .spec.approved + name: Approved + type: boolean + name: v1alpha1 + schema: + openAPIV3Schema: + description: InstallPlan defines the installation of a set of operators. + type: object + required: + - metadata + - spec + 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: InstallPlanSpec defines a set of Application resources to be installed + type: object + required: + - approval + - approved + - clusterServiceVersionNames + properties: + approval: + description: Approval is the user approval policy for an InstallPlan. It must be one of "Automatic" or "Manual". + type: string + approved: + type: boolean + clusterServiceVersionNames: + type: array + items: + type: string + generation: + type: integer + source: + type: string + sourceNamespace: + type: string + status: + description: "InstallPlanStatus represents the information about the status of steps required to complete installation. \n Status may trail the actual state of a system." + type: object + required: + - catalogSources + - phase + properties: + attenuatedServiceAccountRef: + description: AttenuatedServiceAccountRef references the service account that is used to do scoped operator install. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + bundleLookups: + description: BundleLookups is the set of in-progress requests to pull and unpackage bundle content to the cluster. + type: array + items: + description: BundleLookup is a request to pull and unpackage the content of a bundle to the cluster. + type: object + required: + - catalogSourceRef + - identifier + - path + - replaces + properties: + catalogSourceRef: + description: CatalogSourceRef is a reference to the CatalogSource the bundle path was resolved from. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + conditions: + description: Conditions represents the overall state of a BundleLookup. + type: array + items: + type: object + required: + - status + - type + properties: + lastTransitionTime: + description: Last time the condition transitioned from one status to another. + type: string + format: date-time + lastUpdateTime: + description: Last time the condition was probed. + type: string + format: date-time + 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 condition. + type: string + identifier: + description: Identifier is the catalog-unique name of the operator (the name of the CSV for bundles that contain CSVs) + type: string + path: + description: Path refers to the location of a bundle to pull. It's typically an image reference. + type: string + properties: + description: The effective properties of the unpacked bundle. + type: string + replaces: + description: Replaces is the name of the bundle to replace with the one found at Path. + type: string + catalogSources: + type: array + items: + type: string + conditions: + type: array + items: + description: InstallPlanCondition represents the overall status of the execution of an InstallPlan. + type: object + properties: + lastTransitionTime: + type: string + format: date-time + lastUpdateTime: + type: string + format: date-time + message: + type: string + reason: + description: ConditionReason is a camelcased reason for the state transition. + type: string + status: + type: string + type: + description: InstallPlanConditionType describes the state of an InstallPlan at a certain point as a whole. + type: string + message: + description: Message is a human-readable message containing detailed information that may be important to understanding why the plan has its current status. + type: string + phase: + description: InstallPlanPhase is the current status of a InstallPlan as a whole. + type: string + plan: + type: array + items: + description: Step represents the status of an individual step in an InstallPlan. + type: object + required: + - resolving + - resource + - status + properties: + optional: + type: boolean + resolving: + type: string + resource: + description: StepResource represents the status of a resource to be tracked by an InstallPlan. + type: object + required: + - group + - kind + - name + - sourceName + - sourceNamespace + - version + properties: + group: + type: string + kind: + type: string + manifest: + type: string + name: + type: string + sourceName: + type: string + sourceNamespace: + type: string + version: + type: string + status: + description: StepStatus is the current status of a particular resource an in InstallPlan + type: string + startTime: + description: StartTime is the time when the controller began applying the resources listed in the plan to the cluster. + type: string + format: date-time + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: olmconfigs.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: OLMConfig + listKind: OLMConfigList + plural: olmconfigs + singular: olmconfig + scope: Cluster + versions: + - name: v1 + schema: + openAPIV3Schema: + description: OLMConfig is a resource responsible for configuring OLM. + type: object + required: + - metadata + 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: OLMConfigSpec is the spec for an OLMConfig resource. + type: object + properties: + features: + description: Features contains the list of configurable OLM features. + type: object + properties: + disableCopiedCSVs: + description: DisableCopiedCSVs is used to disable OLM's "Copied CSV" feature for operators installed at the cluster scope, where a cluster scoped operator is one that has been installed in an OperatorGroup that targets all namespaces. When reenabled, OLM will recreate the "Copied CSVs" for each cluster scoped operator. + type: boolean + packageServerSyncInterval: + description: PackageServerSyncInterval is used to define the sync interval for packagerserver pods. Packageserver pods periodically check the status of CatalogSources; this specifies the period using duration format (e.g. "60m"). For this parameter, only hours ("h"), minutes ("m"), and seconds ("s") may be specified. When not specified, the period defaults to the value specified within the packageserver. + type: string + pattern: ^([0-9]+(\.[0-9]+)?(s|m|h))+$ + status: + description: OLMConfigStatus is the status for an OLMConfig resource. + type: object + properties: + conditions: + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - lastTransitionTime + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: operatorconditions.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: OperatorCondition + listKind: OperatorConditionList + plural: operatorconditions + shortNames: + - condition + singular: operatorcondition + scope: Namespaced + versions: + - name: v1 + schema: + openAPIV3Schema: + description: OperatorCondition is a Custom Resource of type `OperatorCondition` which is used to convey information to OLM about the state of an operator. + type: object + required: + - metadata + 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: OperatorConditionSpec allows a cluster admin to convey information about the state of an operator to OLM, potentially overriding state reported by the operator. + type: object + properties: + deployments: + type: array + items: + type: string + overrides: + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + serviceAccounts: + type: array + items: + type: string + status: + description: OperatorConditionStatus allows an operator to convey information its state to OLM. The status may trail the actual state of a system. + type: object + properties: + conditions: + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - lastTransitionTime + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + served: true + storage: false + subresources: + status: {} + - name: v2 + schema: + openAPIV3Schema: + description: OperatorCondition is a Custom Resource of type `OperatorCondition` which is used to convey information to OLM about the state of an operator. + type: object + required: + - metadata + 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: OperatorConditionSpec allows an operator to report state to OLM and provides cluster admin with the ability to manually override state reported by the operator. + type: object + properties: + conditions: + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - lastTransitionTime + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + deployments: + type: array + items: + type: string + overrides: + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + serviceAccounts: + type: array + items: + type: string + status: + description: OperatorConditionStatus allows OLM to convey which conditions have been observed. + type: object + properties: + conditions: + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - lastTransitionTime + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: operatorgroups.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: OperatorGroup + listKind: OperatorGroupList + plural: operatorgroups + shortNames: + - og + singular: operatorgroup + scope: Namespaced + versions: + - name: v1 + schema: + openAPIV3Schema: + description: OperatorGroup is the unit of multitenancy for OLM managed operators. It constrains the installation of operators in its namespace to a specified set of target namespaces. + type: object + required: + - metadata + 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: OperatorGroupSpec is the spec for an OperatorGroup resource. + type: object + default: + upgradeStrategy: Default + properties: + selector: + description: Selector selects the OperatorGroup's target namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + serviceAccountName: + description: ServiceAccountName is the admin specified service account which will be used to deploy operator(s) in this operator group. + type: string + staticProvidedAPIs: + description: Static tells OLM not to update the OperatorGroup's providedAPIs annotation + type: boolean + targetNamespaces: + description: TargetNamespaces is an explicit set of namespaces to target. If it is set, Selector is ignored. + type: array + items: + type: string + x-kubernetes-list-type: set + upgradeStrategy: + description: "UpgradeStrategy defines the upgrade strategy for operators in the namespace. There are currently two supported upgrade strategies: \n Default: OLM will only allow clusterServiceVersions to move to the replacing phase from the succeeded phase. This effectively means that OLM will not allow operators to move to the next version if an installation or upgrade has failed. \n TechPreviewUnsafeFailForward: OLM will allow clusterServiceVersions to move to the replacing phase from the succeeded phase or from the failed phase. Additionally, OLM will generate new installPlans when a subscription references a failed installPlan and the catalog has been updated with a new upgrade for the existing set of operators. \n WARNING: The TechPreviewUnsafeFailForward upgrade strategy is unsafe and may result in unexpected behavior or unrecoverable data loss unless you have deep understanding of the set of operators being managed in the namespace." + type: string + default: Default + enum: + - Default + - TechPreviewUnsafeFailForward + status: + description: OperatorGroupStatus is the status for an OperatorGroupResource. + type: object + required: + - lastUpdated + properties: + conditions: + description: Conditions is an array of the OperatorGroup's conditions. + type: array + items: + description: "Condition contains details for one aspect of the current state of this API Resource. --- This struct is intended for direct use as an array at the field path .status.conditions. For example, \n type FooStatus struct{ // Represents the observations of a foo's current state. // Known .status.conditions.type are: \"Available\", \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge // +listType=map // +listMapKey=type Conditions []metav1.Condition `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\" protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }" + type: object + required: + - lastTransitionTime + - message + - reason + - status + - type + 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. + type: string + format: date-time + message: + description: message is a human readable message indicating details about the transition. This may be an empty string. + type: string + maxLength: 32768 + 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. + type: integer + format: int64 + minimum: 0 + 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. + type: string + maxLength: 1024 + minLength: 1 + pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$ + status: + description: status of the condition, one of True, False, Unknown. + type: string + enum: + - "True" + - "False" + - Unknown + type: + description: type of condition in CamelCase or in foo.example.com/CamelCase. --- Many .condition.type values are consistent across resources like Available, but because arbitrary conditions can be useful (see .node.status.conditions), the ability to deconflict is important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt) + type: string + 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])$ + lastUpdated: + description: LastUpdated is a timestamp of the last time the OperatorGroup's status was Updated. + type: string + format: date-time + namespaces: + description: Namespaces is the set of target namespaces for the OperatorGroup. + type: array + items: + type: string + x-kubernetes-list-type: set + serviceAccountRef: + description: ServiceAccountRef references the service account object specified. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + served: true + storage: true + subresources: + status: {} + - name: v1alpha2 + schema: + openAPIV3Schema: + description: OperatorGroup is the unit of multitenancy for OLM managed operators. It constrains the installation of operators in its namespace to a specified set of target namespaces. + type: object + required: + - metadata + 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: OperatorGroupSpec is the spec for an OperatorGroup resource. + type: object + properties: + selector: + description: Selector selects the OperatorGroup's target namespaces. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + serviceAccountName: + description: ServiceAccountName is the admin specified service account which will be used to deploy operator(s) in this operator group. + type: string + staticProvidedAPIs: + description: Static tells OLM not to update the OperatorGroup's providedAPIs annotation + type: boolean + targetNamespaces: + description: TargetNamespaces is an explicit set of namespaces to target. If it is set, Selector is ignored. + type: array + items: + type: string + status: + description: OperatorGroupStatus is the status for an OperatorGroupResource. + type: object + required: + - lastUpdated + properties: + lastUpdated: + description: LastUpdated is a timestamp of the last time the OperatorGroup's status was Updated. + type: string + format: date-time + namespaces: + description: Namespaces is the set of target namespaces for the OperatorGroup. + type: array + items: + type: string + serviceAccountRef: + description: ServiceAccountRef references the service account object specified. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + served: true + storage: false + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: operators.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: Operator + listKind: OperatorList + plural: operators + singular: operator + scope: Cluster + versions: + - name: v1 + schema: + openAPIV3Schema: + description: Operator represents a cluster operator. + type: object + 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: OperatorSpec defines the desired state of Operator + type: object + status: + description: OperatorStatus defines the observed state of an Operator and its components + type: object + properties: + components: + description: Components describes resources that compose the operator. + type: object + required: + - labelSelector + properties: + labelSelector: + description: LabelSelector is a label query over a set of resources used to select the operator's components + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + refs: + description: Refs are a set of references to the operator's component resources, selected with LabelSelector. + type: array + items: + description: RichReference is a reference to a resource, enriched with its status conditions. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + conditions: + description: Conditions represents the latest state of the component. + type: array + items: + description: Condition represent the latest available observations of an component's state. + type: object + required: + - status + - type + properties: + lastTransitionTime: + description: Last time the condition transitioned from one status to another. + type: string + format: date-time + lastUpdateTime: + description: Last time the condition was probed + type: string + format: date-time + 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 condition. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + served: true + storage: true + subresources: + status: {} +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.9.0 + creationTimestamp: null + name: subscriptions.operators.coreos.com +spec: + group: operators.coreos.com + names: + categories: + - olm + kind: Subscription + listKind: SubscriptionList + plural: subscriptions + shortNames: + - sub + - subs + singular: subscription + scope: Namespaced + versions: + - additionalPrinterColumns: + - description: The package subscribed to + jsonPath: .spec.name + name: Package + type: string + - description: The catalog source for the specified package + jsonPath: .spec.source + name: Source + type: string + - description: The channel of updates to subscribe to + jsonPath: .spec.channel + name: Channel + type: string + name: v1alpha1 + schema: + openAPIV3Schema: + description: Subscription keeps operators up to date by tracking changes to Catalogs. + type: object + required: + - metadata + - spec + 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: SubscriptionSpec defines an Application that can be installed + type: object + required: + - name + - source + - sourceNamespace + properties: + channel: + type: string + config: + description: SubscriptionConfig contains configuration specified for a subscription. + type: object + properties: + affinity: + description: If specified, overrides the pod's scheduling constraints. nil sub-attributes will *not* override the original values in the pod.spec for those sub-attributes. Use empty object ({}) to erase original sub-attribute values. + type: object + properties: + nodeAffinity: + description: Describes node affinity scheduling rules for the pod. + type: object + 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. + type: array + 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). + type: object + required: + - preference + - weight + properties: + preference: + description: A node selector term, associated with the corresponding weight. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + weight: + description: Weight associated with matching the corresponding nodeSelectorTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: object + required: + - nodeSelectorTerms + properties: + nodeSelectorTerms: + description: Required. A list of node selector terms. The terms are ORed. + type: array + 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. + type: object + properties: + matchExpressions: + description: A list of node selector requirements by node's labels. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchFields: + description: A list of node selector requirements by node's fields. + type: array + items: + description: A node selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + podAffinity: + description: Describes pod affinity scheduling rules (e.g. co-locate this pod in the same node, zone, etc. as some other pod(s)). + type: object + 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. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: array + 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 + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + 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)). + type: object + 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. + type: array + items: + description: The weights of all of the matched WeightedPodAffinityTerm fields are added per-node to find the most preferred node(s) + type: object + required: + - podAffinityTerm + - weight + properties: + podAffinityTerm: + description: Required. A pod affinity term, associated with the corresponding weight. + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + weight: + description: weight associated with matching the corresponding podAffinityTerm, in the range 1-100. + type: integer + format: int32 + 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. + type: array + 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 + type: object + required: + - topologyKey + properties: + labelSelector: + description: A label query over a set of resources, in this case pods. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + 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". + type: array + items: + type: string + 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 + annotations: + description: Annotations is an unstructured key value map stored with each Deployment, Pod, APIService in the Operator. Typically, annotations may be set by external tools to store and retrieve arbitrary metadata. Use this field to pre-define annotations that OLM should add to each of the Subscription's deployments, pods, and apiservices. + type: object + additionalProperties: + type: string + env: + description: Env is a list of environment variables to set in the container. Cannot be updated. + type: array + items: + description: EnvVar represents an environment variable present in a Container. + type: object + required: + - name + properties: + name: + description: Name of the environment variable. Must be a C_IDENTIFIER. + type: string + value: + description: 'Variable references $(VAR_NAME) are expanded using the previously defined environment variables in the container and any service environment variables. If a variable cannot be resolved, the reference in the input string will be unchanged. Double $$ are reduced to a single $, which allows for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)". Escaped references will never be expanded, regardless of whether the variable exists or not. Defaults to "".' + type: string + valueFrom: + description: Source for the environment variable's value. Cannot be used if value is not empty. + type: object + properties: + configMapKeyRef: + description: Selects a key of a ConfigMap. + type: object + required: + - key + properties: + key: + description: The key to select. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap or its key must be defined + type: boolean + fieldRef: + description: 'Selects a field of the pod: supports metadata.name, metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`, spec.nodeName, spec.serviceAccountName, status.hostIP, status.podIP, status.podIPs.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage, requests.cpu, requests.memory and requests.ephemeral-storage) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + secretKeyRef: + description: Selects a key of a secret in the pod's namespace + type: object + required: + - key + properties: + key: + description: The key of the secret to select from. Must be a valid secret key. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret or its key must be defined + type: boolean + envFrom: + description: EnvFrom is a list of sources to populate environment variables in the container. The keys defined within a source must be a C_IDENTIFIER. All invalid keys will be reported as an event when the container is starting. When a key exists in multiple sources, the value associated with the last source will take precedence. Values defined by an Env with a duplicate key will take precedence. Immutable. + type: array + items: + description: EnvFromSource represents the source of a set of ConfigMaps + type: object + properties: + configMapRef: + description: The ConfigMap to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the ConfigMap must be defined + type: boolean + prefix: + description: An optional identifier to prepend to each key in the ConfigMap. Must be a C_IDENTIFIER. + type: string + secretRef: + description: The Secret to select from + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: Specify whether the Secret must be defined + type: boolean + nodeSelector: + description: 'NodeSelector is a selector which must be true for the pod to fit on a node. Selector which must match a node''s labels for the pod to be scheduled on that node. More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/' + type: object + additionalProperties: + type: string + resources: + description: 'Resources represents compute resources required by this container. Immutable. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/' + type: object + properties: + claims: + description: "Claims lists the names of resources, defined in spec.resourceClaims, that are used by this container. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable. It can only be set for containers." + type: array + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + type: object + required: + - name + 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 + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + description: 'Limits describes the maximum amount of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + requests: + 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 + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + selector: + description: Selector is the label selector for pods to be configured. Existing ReplicaSets whose pods are selected by this will be the ones affected by this deployment. It must match the pod template's labels. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + tolerations: + description: Tolerations are the pod's tolerations. + type: array + items: + description: The pod this Toleration is attached to tolerates any taint that matches the triple using the matching operator . + type: object + 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. + type: integer + format: int64 + 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 + volumeMounts: + description: List of VolumeMounts to set in the container. + type: array + items: + description: VolumeMount describes a mounting of a Volume within a container. + type: object + required: + - mountPath + - name + properties: + mountPath: + description: Path within the container at which the volume should be mounted. Must not contain ':'. + type: string + mountPropagation: + description: mountPropagation determines how mounts are propagated from the host to container and the other way around. When not set, MountPropagationNone is used. This field is beta in 1.10. + type: string + name: + description: This must match the Name of a Volume. + type: string + readOnly: + description: Mounted read-only if true, read-write otherwise (false or unspecified). Defaults to false. + type: boolean + subPath: + description: Path within the volume from which the container's volume should be mounted. Defaults to "" (volume's root). + type: string + subPathExpr: + description: Expanded path within the volume from which the container's volume should be mounted. Behaves similarly to SubPath but environment variable references $(VAR_NAME) are expanded using the container's environment. Defaults to "" (volume's root). SubPathExpr and SubPath are mutually exclusive. + type: string + volumes: + description: List of Volumes to set in the podSpec. + type: array + items: + description: Volume represents a named volume in a pod that may be accessed by any container in the pod. + type: object + required: + - name + properties: + awsElasticBlockStore: + description: 'awsElasticBlockStore represents an AWS Disk resource that is attached to a kubelet''s host machine and then exposed to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore' + type: object + required: + - volumeID + properties: + fsType: + description: 'fsType is the filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + partition: + description: 'partition is the partition in the volume that you want to mount. If omitted, the default is to mount by volume name. Examples: For volume /dev/sda1, you specify the partition as "1". Similarly, the volume partition for /dev/sda is "0" (or you can leave the property empty).' + type: integer + format: int32 + readOnly: + description: 'readOnly value true will force the readOnly setting in VolumeMounts. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore' + type: boolean + volumeID: + description: 'volumeID is unique ID of the persistent disk resource in AWS (Amazon EBS volume). More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore' + type: string + azureDisk: + description: azureDisk represents an Azure Data Disk mount on the host and bind mount to the pod. + type: object + required: + - diskName + - diskURI + properties: + cachingMode: + description: 'cachingMode is the Host Caching mode: None, Read Only, Read Write.' + type: string + diskName: + description: diskName is the Name of the data disk in the blob storage + type: string + diskURI: + description: diskURI is the URI of data disk in the blob storage + type: string + fsType: + description: fsType is Filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + kind: + description: 'kind expected values are Shared: multiple blob disks per storage account Dedicated: single blob disk per storage account Managed: azure managed data disk (only in managed availability set). defaults to shared' + type: string + readOnly: + description: readOnly Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + azureFile: + description: azureFile represents an Azure File Service mount on the host and bind mount to the pod. + type: object + required: + - secretName + - shareName + properties: + readOnly: + description: readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + secretName: + description: secretName is the name of secret that contains Azure Storage Account Name and Key + type: string + shareName: + description: shareName is the azure share Name + type: string + cephfs: + description: cephFS represents a Ceph FS mount on the host that shares a pod's lifetime + type: object + required: + - monitors + properties: + monitors: + description: 'monitors is Required: Monitors is a collection of Ceph monitors More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: array + items: + type: string + path: + description: 'path is Optional: Used as the mounted root, rather than the full Ceph tree, default is /' + type: string + readOnly: + description: 'readOnly is Optional: Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: boolean + secretFile: + description: 'secretFile is Optional: SecretFile is the path to key ring for User, default is /etc/ceph/user.secret More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: string + secretRef: + description: 'secretRef is Optional: SecretRef is reference to the authentication secret for User, default is empty. More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + user: + description: 'user is optional: User is the rados user name, default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it' + type: string + cinder: + description: 'cinder represents a cinder volume attached and mounted on kubelets host machine. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: object + required: + - volumeID + properties: + fsType: + description: 'fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: string + readOnly: + description: 'readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: boolean + secretRef: + description: 'secretRef is optional: points to a secret object containing parameters used to connect to OpenStack.' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + volumeID: + description: 'volumeID used to identify the volume in cinder. More info: https://examples.k8s.io/mysql-cinder-pd/README.md' + type: string + configMap: + description: configMap represents a configMap that should populate this volume + type: object + properties: + defaultMode: + description: 'defaultMode is optional: mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Defaults to 0644. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + items: + description: items if unspecified, each key-value pair in the Data field of the referenced ConfigMap will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the ConfigMap, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: optional specify whether the ConfigMap or its keys must be defined + type: boolean + csi: + description: csi (Container Storage Interface) represents ephemeral storage that is handled by certain external CSI drivers (Beta feature). + type: object + required: + - driver + properties: + driver: + description: driver is the name of the CSI driver that handles this volume. Consult with your admin for the correct name as registered in the cluster. + type: string + fsType: + description: fsType to mount. Ex. "ext4", "xfs", "ntfs". If not provided, the empty value is passed to the associated CSI driver which will determine the default filesystem to apply. + type: string + nodePublishSecretRef: + description: nodePublishSecretRef is a reference to the secret object containing sensitive information to pass to the CSI driver to complete the CSI NodePublishVolume and NodeUnpublishVolume calls. This field is optional, and may be empty if no secret is required. If the secret object contains more than one secret, all secret references are passed. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + readOnly: + description: readOnly specifies a read-only configuration for the volume. Defaults to false (read/write). + type: boolean + volumeAttributes: + description: volumeAttributes stores driver-specific properties that are passed to the CSI driver. Consult your driver's documentation for supported values. + type: object + additionalProperties: + type: string + downwardAPI: + description: downwardAPI represents downward API about the pod that should populate this volume + type: object + properties: + defaultMode: + description: 'Optional: mode bits to use on created files by default. Must be a Optional: mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Defaults to 0644. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + items: + description: Items is a list of downward API volume file + type: array + items: + description: DownwardAPIVolumeFile represents information to create the file containing the pod field + type: object + required: + - path + properties: + fieldRef: + description: 'Required: Selects a field of the pod: only annotations, labels, name and namespace are supported.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + mode: + description: 'Optional: mode bits used to set permissions on this file, must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: 'Required: Path is the relative path name of the file to be created. Must not be absolute or contain the ''..'' path. Must be utf-8 encoded. The first item of the relative path must not start with ''..''' + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, requests.cpu and requests.memory) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + emptyDir: + description: 'emptyDir represents a temporary directory that shares a pod''s lifetime. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir' + type: object + properties: + medium: + description: 'medium represents what type of storage medium should back this directory. The default is "" which means to use the node''s default medium. Must be an empty string (default) or Memory. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir' + type: string + sizeLimit: + description: 'sizeLimit is the total amount of local storage required for this EmptyDir volume. The size limit is also applicable for memory medium. 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. The default is nil which means that the limit is undefined. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir' + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + ephemeral: + description: "ephemeral represents a volume that is handled by a cluster storage driver. The volume's lifecycle is tied to the pod that defines it - it will be created before the pod starts, and deleted when the pod is removed. \n Use this if: a) the volume is only needed while the pod runs, b) features of normal volumes like restoring from snapshot or capacity tracking are needed, c) the storage driver is specified through a storage class, and d) the storage driver supports dynamic volume provisioning through a PersistentVolumeClaim (see EphemeralVolumeSource for more information on the connection between this volume type and PersistentVolumeClaim). \n Use PersistentVolumeClaim or one of the vendor-specific APIs for volumes that persist for longer than the lifecycle of an individual pod. \n Use CSI for light-weight local ephemeral volumes if the CSI driver is meant to be used that way - see the documentation of the driver for more information. \n A pod can use both types of ephemeral volumes and persistent volumes at the same time." + type: object + properties: + volumeClaimTemplate: + description: "Will be used to create a stand-alone PVC to provision the volume. The pod in which this EphemeralVolumeSource is embedded will be the owner of the PVC, i.e. the PVC will be deleted together with the pod. The name of the PVC will be `-` where `` is the name from the `PodSpec.Volumes` array entry. Pod validation will reject the pod if the concatenated name is not valid for a PVC (for example, too long). \n An existing PVC with that name that is not owned by the pod will *not* be used for the pod to avoid using an unrelated volume by mistake. Starting the pod is then blocked until the unrelated PVC is removed. If such a pre-created PVC is meant to be used by the pod, the PVC has to updated with an owner reference to the pod once the pod exists. Normally this should not be necessary, but it may be useful when manually reconstructing a broken cluster. \n This field is read-only and no changes will be made by Kubernetes to the PVC after it has been created. \n Required, must not be nil." + type: object + required: + - spec + properties: + metadata: + description: May contain labels and annotations that will be copied into the PVC when creating it. No other fields are allowed and will be rejected during validation. + type: object + spec: + description: The specification for the PersistentVolumeClaim. The entire content is copied unchanged into the PVC that gets created from this template. The same fields as in a PersistentVolumeClaim are also valid here. + type: object + properties: + accessModes: + description: 'accessModes contains the desired access modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1' + type: array + items: + type: string + dataSource: + description: 'dataSource field can be used to specify either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot) * An existing PVC (PersistentVolumeClaim) If the provisioner or an external controller can support the specified data source, it will create a new volume based on the contents of the specified data source. When the AnyVolumeDataSource feature gate is enabled, dataSource contents will be copied to dataSourceRef, and dataSourceRef contents will be copied to dataSource when dataSourceRef.namespace is not specified. If the namespace is specified, then dataSourceRef will not be copied to dataSource.' + type: object + required: + - kind + - name + properties: + apiGroup: + description: APIGroup is the group for the resource being referenced. If APIGroup is not specified, the specified Kind must be in the core API group. For any other third-party types, APIGroup is required. + type: string + kind: + description: Kind is the type of resource being referenced + type: string + name: + description: Name is the name of resource being referenced + type: string + dataSourceRef: + description: 'dataSourceRef specifies the object from which to populate the volume with data, if a non-empty volume is desired. This may be any object from a non-empty API group (non core object) or a PersistentVolumeClaim object. When this field is specified, volume binding will only succeed if the type of the specified object matches some installed volume populator or dynamic provisioner. This field will replace the functionality of the dataSource field and as such if both fields are non-empty, they must have the same value. For backwards compatibility, when namespace isn''t specified in dataSourceRef, both fields (dataSource and dataSourceRef) will be set to the same value automatically if one of them is empty and the other is non-empty. When namespace is specified in dataSourceRef, dataSource isn''t set to the same value and must be empty. There are three important differences between dataSource and dataSourceRef: * While dataSource only allows two specific types of objects, dataSourceRef allows any non-core object, as well as PersistentVolumeClaim objects. * While dataSource ignores disallowed values (dropping them), dataSourceRef preserves all values, and generates an error if a disallowed value is specified. * While dataSource only allows local objects, dataSourceRef allows objects in any namespaces. (Beta) Using this field requires the AnyVolumeDataSource feature gate to be enabled. (Alpha) Using the namespace field of dataSourceRef requires the CrossNamespaceVolumeDataSource feature gate to be enabled.' + type: object + required: + - kind + - name + properties: + apiGroup: + description: APIGroup is the group for the resource being referenced. If APIGroup is not specified, the specified Kind must be in the core API group. For any other third-party types, APIGroup is required. + type: string + kind: + description: Kind is the type of resource being referenced + type: string + name: + description: Name is the name of resource being referenced + type: string + namespace: + description: Namespace is the namespace of resource being referenced Note that when a namespace is specified, a gateway.networking.k8s.io/ReferenceGrant object is required in the referent namespace to allow that namespace's owner to accept the reference. See the ReferenceGrant documentation for details. (Alpha) This field requires the CrossNamespaceVolumeDataSource feature gate to be enabled. + type: string + resources: + description: 'resources represents the minimum resources the volume should have. If RecoverVolumeExpansionFailure feature is enabled users are allowed to specify resource requirements that are lower than previous value but must still be higher than capacity recorded in the status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources' + type: object + properties: + claims: + description: "Claims lists the names of resources, defined in spec.resourceClaims, that are used by this container. \n This is an alpha field and requires enabling the DynamicResourceAllocation feature gate. \n This field is immutable. It can only be set for containers." + type: array + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + type: object + required: + - name + 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 + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + description: 'Limits describes the maximum amount of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + requests: + 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 + additionalProperties: + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + selector: + description: selector is a label query over volumes to consider for binding. + type: object + properties: + matchExpressions: + description: matchExpressions is a list of label selector requirements. The requirements are ANDed. + type: array + items: + description: A label selector requirement is a selector that contains values, a key, and an operator that relates the key and values. + type: object + required: + - key + - operator + 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. + type: array + items: + type: string + matchLabels: + 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 + additionalProperties: + type: string + storageClassName: + description: 'storageClassName is the name of the StorageClass required by the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1' + type: string + volumeMode: + description: volumeMode defines what type of volume is required by the claim. Value of Filesystem is implied when not included in claim spec. + type: string + volumeName: + description: volumeName is the binding reference to the PersistentVolume backing this claim. + type: string + fc: + description: fc represents a Fibre Channel resource that is attached to a kubelet's host machine and then exposed to the pod. + type: object + properties: + fsType: + description: 'fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + lun: + description: 'lun is Optional: FC target lun number' + type: integer + format: int32 + readOnly: + description: 'readOnly is Optional: Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts.' + type: boolean + targetWWNs: + description: 'targetWWNs is Optional: FC target worldwide names (WWNs)' + type: array + items: + type: string + wwids: + description: 'wwids Optional: FC volume world wide identifiers (wwids) Either wwids or combination of targetWWNs and lun must be set, but not both simultaneously.' + type: array + items: + type: string + flexVolume: + description: flexVolume represents a generic volume resource that is provisioned/attached using an exec based plugin. + type: object + required: + - driver + properties: + driver: + description: driver is the name of the driver to use for this volume. + type: string + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". The default filesystem depends on FlexVolume script. + type: string + options: + description: 'options is Optional: this field holds extra command options if any.' + type: object + additionalProperties: + type: string + readOnly: + description: 'readOnly is Optional: defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts.' + type: boolean + secretRef: + description: 'secretRef is Optional: secretRef is reference to the secret object containing sensitive information to pass to the plugin scripts. This may be empty if no secret object is specified. If the secret object contains more than one secret, all secrets are passed to the plugin scripts.' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + flocker: + description: flocker represents a Flocker volume attached to a kubelet's host machine. This depends on the Flocker control service being running + type: object + properties: + datasetName: + description: datasetName is Name of the dataset stored as metadata -> name on the dataset for Flocker should be considered as deprecated + type: string + datasetUUID: + description: datasetUUID is the UUID of the dataset. This is unique identifier of a Flocker dataset + type: string + gcePersistentDisk: + description: 'gcePersistentDisk represents a GCE Disk resource that is attached to a kubelet''s host machine and then exposed to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: object + required: + - pdName + properties: + fsType: + description: 'fsType is filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + partition: + description: 'partition is the partition in the volume that you want to mount. If omitted, the default is to mount by volume name. Examples: For volume /dev/sda1, you specify the partition as "1". Similarly, the volume partition for /dev/sda is "0" (or you can leave the property empty). More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: integer + format: int32 + pdName: + description: 'pdName is unique name of the PD resource in GCE. Used to identify the disk in GCE. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: string + readOnly: + description: 'readOnly here will force the ReadOnly setting in VolumeMounts. Defaults to false. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk' + type: boolean + gitRepo: + description: 'gitRepo represents a git repository at a particular revision. DEPRECATED: GitRepo is deprecated. To provision a container with a git repo, mount an EmptyDir into an InitContainer that clones the repo using git, then mount the EmptyDir into the Pod''s container.' + type: object + required: + - repository + properties: + directory: + description: directory is the target directory name. Must not contain or start with '..'. If '.' is supplied, the volume directory will be the git repository. Otherwise, if specified, the volume will contain the git repository in the subdirectory with the given name. + type: string + repository: + description: repository is the URL + type: string + revision: + description: revision is the commit hash for the specified revision. + type: string + glusterfs: + description: 'glusterfs represents a Glusterfs mount on the host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/glusterfs/README.md' + type: object + required: + - endpoints + - path + properties: + endpoints: + description: 'endpoints is the endpoint name that details Glusterfs topology. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod' + type: string + path: + description: 'path is the Glusterfs volume path. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod' + type: string + readOnly: + description: 'readOnly here will force the Glusterfs volume to be mounted with read-only permissions. Defaults to false. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod' + type: boolean + hostPath: + description: 'hostPath represents a pre-existing file or directory on the host machine that is directly exposed to the container. This is generally used for system agents or other privileged things that are allowed to see the host machine. Most containers will NOT need this. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath --- TODO(jonesdl) We need to restrict who can use host directory mounts and who can/can not mount host directories as read/write.' + type: object + required: + - path + properties: + path: + description: 'path of the directory on the host. If the path is a symlink, it will follow the link to the real path. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath' + type: string + type: + description: 'type for HostPath Volume Defaults to "" More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath' + type: string + iscsi: + description: 'iscsi represents an ISCSI Disk resource that is attached to a kubelet''s host machine and then exposed to the pod. More info: https://examples.k8s.io/volumes/iscsi/README.md' + type: object + required: + - iqn + - lun + - targetPortal + properties: + chapAuthDiscovery: + description: chapAuthDiscovery defines whether support iSCSI Discovery CHAP authentication + type: boolean + chapAuthSession: + description: chapAuthSession defines whether support iSCSI Session CHAP authentication + type: boolean + fsType: + description: 'fsType is the filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#iscsi TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + initiatorName: + description: initiatorName is the custom iSCSI Initiator Name. If initiatorName is specified with iscsiInterface simultaneously, new iSCSI interface : will be created for the connection. + type: string + iqn: + description: iqn is the target iSCSI Qualified Name. + type: string + iscsiInterface: + description: iscsiInterface is the interface Name that uses an iSCSI transport. Defaults to 'default' (tcp). + type: string + lun: + description: lun represents iSCSI Target Lun number. + type: integer + format: int32 + portals: + description: portals is the iSCSI Target Portal List. The portal is either an IP or ip_addr:port if the port is other than default (typically TCP ports 860 and 3260). + type: array + items: + type: string + readOnly: + description: readOnly here will force the ReadOnly setting in VolumeMounts. Defaults to false. + type: boolean + secretRef: + description: secretRef is the CHAP Secret for iSCSI target and initiator authentication + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + targetPortal: + description: targetPortal is iSCSI Target Portal. The Portal is either an IP or ip_addr:port if the port is other than default (typically TCP ports 860 and 3260). + type: string + name: + description: 'name of the volume. Must be a DNS_LABEL and unique within the pod. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + nfs: + description: 'nfs represents an NFS mount on the host that shares a pod''s lifetime More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: object + required: + - path + - server + properties: + path: + description: 'path that is exported by the NFS server. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: string + readOnly: + description: 'readOnly here will force the NFS export to be mounted with read-only permissions. Defaults to false. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: boolean + server: + description: 'server is the hostname or IP address of the NFS server. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs' + type: string + persistentVolumeClaim: + description: 'persistentVolumeClaimVolumeSource represents a reference to a PersistentVolumeClaim in the same namespace. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims' + type: object + required: + - claimName + properties: + claimName: + description: 'claimName is the name of a PersistentVolumeClaim in the same namespace as the pod using this volume. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims' + type: string + readOnly: + description: readOnly Will force the ReadOnly setting in VolumeMounts. Default false. + type: boolean + photonPersistentDisk: + description: photonPersistentDisk represents a PhotonController persistent disk attached and mounted on kubelets host machine + type: object + required: + - pdID + properties: + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + pdID: + description: pdID is the ID that identifies Photon Controller persistent disk + type: string + portworxVolume: + description: portworxVolume represents a portworx volume attached and mounted on kubelets host machine + type: object + required: + - volumeID + properties: + fsType: + description: fSType represents the filesystem type to mount Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs". Implicitly inferred to be "ext4" if unspecified. + type: string + readOnly: + description: readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + volumeID: + description: volumeID uniquely identifies a Portworx volume + type: string + projected: + description: projected items for all in one resources secrets, configmaps, and downward API + type: object + properties: + defaultMode: + description: defaultMode are the mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set. + type: integer + format: int32 + sources: + description: sources is the list of volume projections + type: array + items: + description: Projection that may be projected along with other supported volume types + type: object + properties: + configMap: + description: configMap information about the configMap data to project + type: object + properties: + items: + description: items if unspecified, each key-value pair in the Data field of the referenced ConfigMap will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the ConfigMap, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: optional specify whether the ConfigMap or its keys must be defined + type: boolean + downwardAPI: + description: downwardAPI information about the downwardAPI data to project + type: object + properties: + items: + description: Items is a list of DownwardAPIVolume file + type: array + items: + description: DownwardAPIVolumeFile represents information to create the file containing the pod field + type: object + required: + - path + properties: + fieldRef: + description: 'Required: Selects a field of the pod: only annotations, labels, name and namespace are supported.' + type: object + required: + - fieldPath + properties: + apiVersion: + description: Version of the schema the FieldPath is written in terms of, defaults to "v1". + type: string + fieldPath: + description: Path of the field to select in the specified API version. + type: string + mode: + description: 'Optional: mode bits used to set permissions on this file, must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: 'Required: Path is the relative path name of the file to be created. Must not be absolute or contain the ''..'' path. Must be utf-8 encoded. The first item of the relative path must not start with ''..''' + type: string + resourceFieldRef: + description: 'Selects a resource of the container: only resources limits and requests (limits.cpu, limits.memory, requests.cpu and requests.memory) are currently supported.' + type: object + required: + - resource + properties: + containerName: + description: 'Container name: required for volumes, optional for env vars' + type: string + divisor: + description: Specifies the output format of the exposed resources, defaults to "1" + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + anyOf: + - type: integer + - type: string + x-kubernetes-int-or-string: true + resource: + description: 'Required: resource to select' + type: string + secret: + description: secret information about the secret data to project + type: object + properties: + items: + description: items if unspecified, each key-value pair in the Data field of the referenced Secret will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the Secret, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + optional: + description: optional field specify whether the Secret or its key must be defined + type: boolean + serviceAccountToken: + description: serviceAccountToken is information about the serviceAccountToken data to project + type: object + required: + - path + properties: + audience: + description: audience is the intended audience of the token. A recipient of a token must identify itself with an identifier specified in the audience of the token, and otherwise should reject the token. The audience defaults to the identifier of the apiserver. + type: string + expirationSeconds: + description: expirationSeconds is the requested duration of validity of the service account token. As the token approaches expiration, the kubelet volume plugin will proactively rotate the service account token. The kubelet will start trying to rotate the token if the token is older than 80 percent of its time to live or if the token is older than 24 hours.Defaults to 1 hour and must be at least 10 minutes. + type: integer + format: int64 + path: + description: path is the path relative to the mount point of the file to project the token into. + type: string + quobyte: + description: quobyte represents a Quobyte mount on the host that shares a pod's lifetime + type: object + required: + - registry + - volume + properties: + group: + description: group to map volume access to Default is no group + type: string + readOnly: + description: readOnly here will force the Quobyte volume to be mounted with read-only permissions. Defaults to false. + type: boolean + registry: + description: registry represents a single or multiple Quobyte Registry services specified as a string as host:port pair (multiple entries are separated with commas) which acts as the central registry for volumes + type: string + tenant: + description: tenant owning the given Quobyte volume in the Backend Used with dynamically provisioned Quobyte volumes, value is set by the plugin + type: string + user: + description: user to map volume access to Defaults to serivceaccount user + type: string + volume: + description: volume is a string that references an already created Quobyte volume by name. + type: string + rbd: + description: 'rbd represents a Rados Block Device mount on the host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/rbd/README.md' + type: object + required: + - image + - monitors + properties: + fsType: + description: 'fsType is the filesystem type of the volume that you want to mount. Tip: Ensure that the filesystem type is supported by the host operating system. Examples: "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#rbd TODO: how do we prevent errors in the filesystem from compromising the machine' + type: string + image: + description: 'image is the rados image name. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + keyring: + description: 'keyring is the path to key ring for RBDUser. Default is /etc/ceph/keyring. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + monitors: + description: 'monitors is a collection of Ceph monitors. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: array + items: + type: string + pool: + description: 'pool is the rados pool name. Default is rbd. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + readOnly: + description: 'readOnly here will force the ReadOnly setting in VolumeMounts. Defaults to false. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: boolean + secretRef: + description: 'secretRef is name of the authentication secret for RBDUser. If provided overrides keyring. Default is nil. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + user: + description: 'user is the rados user name. Default is admin. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it' + type: string + scaleIO: + description: scaleIO represents a ScaleIO persistent volume attached and mounted on Kubernetes nodes. + type: object + required: + - gateway + - secretRef + - system + properties: + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Default is "xfs". + type: string + gateway: + description: gateway is the host address of the ScaleIO API Gateway. + type: string + protectionDomain: + description: protectionDomain is the name of the ScaleIO Protection Domain for the configured storage. + type: string + readOnly: + description: readOnly Defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + secretRef: + description: secretRef references to the secret for ScaleIO user and other sensitive information. If this is not provided, Login operation will fail. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + sslEnabled: + description: sslEnabled Flag enable/disable SSL communication with Gateway, default false + type: boolean + storageMode: + description: storageMode indicates whether the storage for a volume should be ThickProvisioned or ThinProvisioned. Default is ThinProvisioned. + type: string + storagePool: + description: storagePool is the ScaleIO Storage Pool associated with the protection domain. + type: string + system: + description: system is the name of the storage system as configured in ScaleIO. + type: string + volumeName: + description: volumeName is the name of a volume already created in the ScaleIO system that is associated with this volume source. + type: string + secret: + description: 'secret represents a secret that should populate this volume. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret' + type: object + properties: + defaultMode: + description: 'defaultMode is Optional: mode bits used to set permissions on created files by default. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. Defaults to 0644. Directories within the path are not affected by this setting. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + items: + description: items If unspecified, each key-value pair in the Data field of the referenced Secret will be projected into the volume as a file whose name is the key and content is the value. If specified, the listed keys will be projected into the specified paths, and unlisted keys will not be present. If a key is specified which is not present in the Secret, the volume setup will error unless it is marked optional. Paths must be relative and may not contain the '..' path or start with '..'. + type: array + items: + description: Maps a string key to a path within a volume. + type: object + required: + - key + - path + properties: + key: + description: key is the key to project. + type: string + mode: + description: 'mode is Optional: mode bits used to set permissions on this file. Must be an octal value between 0000 and 0777 or a decimal value between 0 and 511. YAML accepts both octal and decimal values, JSON requires decimal values for mode bits. If not specified, the volume defaultMode will be used. This might be in conflict with other options that affect the file mode, like fsGroup, and the result can be other mode bits set.' + type: integer + format: int32 + path: + description: path is the relative path of the file to map the key to. May not be an absolute path. May not contain the path element '..'. May not start with the string '..'. + type: string + optional: + description: optional field specify whether the Secret or its keys must be defined + type: boolean + secretName: + description: 'secretName is the name of the secret in the pod''s namespace to use. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret' + type: string + storageos: + description: storageOS represents a StorageOS volume attached and mounted on Kubernetes nodes. + type: object + properties: + fsType: + description: fsType is the filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + readOnly: + description: readOnly defaults to false (read/write). ReadOnly here will force the ReadOnly setting in VolumeMounts. + type: boolean + secretRef: + description: secretRef specifies the secret to use for obtaining the StorageOS API credentials. If not specified, default values will be attempted. + type: object + properties: + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names TODO: Add other useful fields. apiVersion, kind, uid?' + type: string + volumeName: + description: volumeName is the human-readable name of the StorageOS volume. Volume names are only unique within a namespace. + type: string + volumeNamespace: + description: volumeNamespace specifies the scope of the volume within StorageOS. If no namespace is specified then the Pod's namespace will be used. This allows the Kubernetes name scoping to be mirrored within StorageOS for tighter integration. Set VolumeName to any name to override the default behaviour. Set to "default" if you are not using namespaces within StorageOS. Namespaces that do not pre-exist within StorageOS will be created. + type: string + vsphereVolume: + description: vsphereVolume represents a vSphere volume attached and mounted on kubelets host machine + type: object + required: + - volumePath + properties: + fsType: + description: fsType is filesystem type to mount. Must be a filesystem type supported by the host operating system. Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified. + type: string + storagePolicyID: + description: storagePolicyID is the storage Policy Based Management (SPBM) profile ID associated with the StoragePolicyName. + type: string + storagePolicyName: + description: storagePolicyName is the storage Policy Based Management (SPBM) profile name. + type: string + volumePath: + description: volumePath is the path that identifies vSphere volume vmdk + type: string + installPlanApproval: + description: Approval is the user approval policy for an InstallPlan. It must be one of "Automatic" or "Manual". + type: string + name: + type: string + source: + type: string + sourceNamespace: + type: string + startingCSV: + type: string + status: + type: object + required: + - lastUpdated + properties: + catalogHealth: + description: CatalogHealth contains the Subscription's view of its relevant CatalogSources' status. It is used to determine SubscriptionStatusConditions related to CatalogSources. + type: array + items: + description: SubscriptionCatalogHealth describes the health of a CatalogSource the Subscription knows about. + type: object + required: + - catalogSourceRef + - healthy + - lastUpdated + properties: + catalogSourceRef: + description: CatalogSourceRef is a reference to a CatalogSource. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + healthy: + description: Healthy is true if the CatalogSource is healthy; false otherwise. + type: boolean + lastUpdated: + description: LastUpdated represents the last time that the CatalogSourceHealth changed + type: string + format: date-time + conditions: + description: Conditions is a list of the latest available observations about a Subscription's current state. + type: array + items: + description: SubscriptionCondition represents the latest available observations of a Subscription's state. + type: object + required: + - status + - type + properties: + lastHeartbeatTime: + description: LastHeartbeatTime is the last time we got an update on a given condition + type: string + format: date-time + lastTransitionTime: + description: LastTransitionTime is the last time the condition transit from one status to another + type: string + format: date-time + message: + description: Message is a human-readable message indicating details about last transition. + type: string + reason: + description: Reason is a one-word CamelCase reason for the condition's last transition. + type: string + status: + description: Status is the status of the condition, one of True, False, Unknown. + type: string + type: + description: Type is the type of Subscription condition. + type: string + currentCSV: + description: CurrentCSV is the CSV the Subscription is progressing to. + type: string + installPlanGeneration: + description: InstallPlanGeneration is the current generation of the installplan + type: integer + installPlanRef: + description: InstallPlanRef is a reference to the latest InstallPlan that contains the Subscription's current CSV. + type: object + properties: + apiVersion: + description: API version of the referent. + type: string + fieldPath: + description: 'If referring to a piece of an object instead of an entire object, this string should contain a valid JSON/Go field access statement, such as desiredState.manifest.containers[2]. For example, if the object reference is to a container within a pod, this would take on a value like: "spec.containers{name}" (where "name" refers to the name of the container that triggered the event) or if no container name is specified "spec.containers[2]" (container with index 2 in this pod). This syntax is chosen only to have some well-defined way of referencing a part of an object. TODO: this design is not final and this field is subject to change in the future.' + type: string + kind: + description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds' + type: string + name: + description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names' + type: string + namespace: + description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/' + type: string + resourceVersion: + description: 'Specific resourceVersion to which this reference is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency' + type: string + uid: + description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids' + type: string + installedCSV: + description: InstalledCSV is the CSV currently installed by the Subscription. + type: string + installplan: + description: 'Install is a reference to the latest InstallPlan generated for the Subscription. DEPRECATED: InstallPlanRef' + type: object + required: + - apiVersion + - kind + - name + - uuid + properties: + apiVersion: + type: string + kind: + type: string + name: + type: string + uuid: + description: UID is a type that holds unique ID values, including UUIDs. Because we don't ONLY use UUIDs, this is an alias to string. Being a type captures intent and helps make sure that UIDs and names do not get conflated. + type: string + lastUpdated: + description: LastUpdated represents the last time that the Subscription status was updated. + type: string + format: date-time + reason: + description: Reason is the reason the Subscription was transitioned to its current state. + type: string + state: + description: State represents the current state of the Subscription + type: string + served: true + storage: true + subresources: + status: {} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/vizier_crd.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/vizier_crd.yaml new file mode 100644 index 0000000000..b25d7b5924 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/crds/vizier_crd.yaml @@ -0,0 +1,347 @@ + +--- +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.4.1 + creationTimestamp: null + name: viziers.px.dev +spec: + group: px.dev + names: + kind: Vizier + listKind: VizierList + plural: viziers + singular: vizier + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: Vizier is the Schema for the viziers 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: VizierSpec defines the desired state of Vizier + properties: + autopilot: + description: Autopilot should be set if running Pixie on GKE Autopilot. + type: boolean + clockConverter: + description: ClockConverter specifies which routine to use for converting + timestamps to a synced reference time. + enum: + - default + - grpc + type: string + cloudAddr: + description: CloudAddr is the address of the cloud instance that the + Vizier should be pointing to. + type: string + clusterName: + description: ClusterName is a name for the Vizier instance, usually + specifying which cluster the Vizier is deployed to. If not specified, + a random name will be generated. + type: string + customDeployKeySecret: + description: CustomDeployKeySecret is the name of the secret where + the deploy key is stored. + type: string + dataAccess: + description: DataAccess defines the level of data that may be accesssed + when executing a script on the cluster. If none specified, assumes + full data access. + enum: + - Full + - Restricted + type: string + dataCollectorParams: + description: DataCollectorParams specifies the set of params for configuring + the dataCollector. If no params are specified, defaults are used. + properties: + customPEMFlags: + additionalProperties: + type: string + description: This contains custom flags that should be passed + to the PEM via environment variables. + type: object + datastreamBufferSize: + description: DatastreamBufferSize is the data buffer size per + connection. Default size is 1 Mbyte. For high-throughput applications, + try increasing this number if experiencing data loss. + format: int32 + type: integer + datastreamBufferSpikeSize: + description: DatastreamBufferSpikeSize is the maximum temporary + size of a data stream buffer before processing. + format: int32 + type: integer + type: object + deployKey: + description: DeployKey is the deploy key associated with the Vizier + instance. This is used to link the Vizier to a specific user/org. + This is required unless specifying a CustomDeployKeySecret. + type: string + devCloudNamespace: + description: 'DevCloudNamespace should be specified only for dev versions + of Pixie cloud which have no ingress to help redirect traffic to + the correct service. The DevCloudNamespace is the namespace that + the dev Pixie cloud is running on, for example: "plc-dev".' + type: string + disableAutoUpdate: + description: DisableAutoUpdate specifies whether auto update should + be enabled for the Vizier instance. + type: boolean + leadershipElectionParams: + description: LeadershipElectionParams specifies configurable values + for the K8s leaderships elections which Vizier uses manage pod leadership. + properties: + electionPeriodMs: + description: ElectionPeriodMs defines how frequently Vizier attempts + to run a K8s leader election, in milliseconds. The period also + determines how long Vizier waits for a leader election response + back from the K8s API. If the K8s API is slow to respond, consider + increasing this number. + format: int64 + type: integer + type: object + patches: + additionalProperties: + type: string + description: Patches defines patches that should be applied to Vizier + resources. The key of the patch should be the name of the resource + that is patched. The value of the patch is the patch, encoded as + a string which follow the "strategic merge patch" rules for K8s. + type: object + pemMemoryLimit: + description: PemMemoryLimit is a memory limit applied specifically + to PEM pods. + type: string + pemMemoryRequest: + description: PemMemoryRequest is a memory request applied specifically + to PEM pods. It will automatically use the value of pemMemoryLimit + if not specified. + type: string + pod: + description: Pod defines the policy for creating Vizier pods. + properties: + annotations: + additionalProperties: + type: string + description: Annotations specifies the annotations to attach to + pods the operator creates. + type: object + labels: + additionalProperties: + type: string + description: Labels specifies the labels to attach to pods the + operator creates. + type: object + nodeSelector: + additionalProperties: + type: string + description: 'NodeSelector is a selector which must be true for + the pod to fit on a node. Selector which must match a node''s + labels for the pod to be scheduled on that node. More info: + https://kubernetes.io/docs/concepts/configuration/assign-pod-node/ + This field cannot be updated once the cluster is created.' + type: object + resources: + description: Resources is the resource requirements for a container. + This field cannot be updated once the cluster is created. + properties: + claims: + description: "Claims lists the names of resources, defined + in spec.resourceClaims, that are used by this container. + \n This is an alpha field and requires enabling the DynamicResourceAllocation + feature gate. \n This field is immutable." + items: + description: ResourceClaim references one entry in PodSpec.ResourceClaims. + properties: + name: + description: Name must match the name of one entry in + pod.spec.resourceClaims of the Pod where this field + is used. It makes that resource available inside a + container. + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + limits: + additionalProperties: + anyOf: + - type: integer + - type: string + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + description: 'Limits describes the maximum amount of compute + resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/' + type: object + 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 + securityContext: + description: The securityContext which should be set on non-privileged + pods. All pods which require privileged permissions will still + require a privileged securityContext. + properties: + enabled: + description: Whether a securityContext should be set on the + pod. In cases where no PSPs are applied to the cluster, + this is not necessary. + type: boolean + fsGroup: + description: A special supplemental group that applies to + all containers in a pod. + format: int64 + type: integer + runAsGroup: + description: The GID to run the entrypoint of the container + process. + format: int64 + type: integer + runAsUser: + description: The UID to run the entrypoint of the container + process. + format: int64 + type: integer + type: object + tolerations: + description: 'Tolerations allows scheduling pods on nodes with + matching taints. More info: https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/: + This field cannot be updated once the cluster is created.' + items: + description: The pod this Toleration is attached to tolerates + any taint that matches the triple using + the matching operator . + properties: + effect: + description: Effect indicates the taint effect to match. + Empty means match all taint effects. When specified, allowed + values are NoSchedule, PreferNoSchedule and NoExecute. + type: string + key: + description: Key is the taint key that the toleration applies + to. Empty means match all taint keys. If the key is empty, + operator must be Exists; this combination means to match + all values and all keys. + type: string + operator: + description: Operator represents a key's relationship to + the value. Valid operators are Exists and Equal. Defaults + to Equal. Exists is equivalent to wildcard for value, + so that a pod can tolerate all taints of a particular + category. + type: string + tolerationSeconds: + description: TolerationSeconds represents the period of + time the toleration (which must be of effect NoExecute, + otherwise this field is ignored) tolerates the taint. + By default, it is not set, which means tolerate the taint + forever (do not evict). Zero and negative values will + be treated as 0 (evict immediately) by the system. + format: int64 + type: integer + value: + description: Value is the taint value the toleration matches + to. If the operator is Exists, the value should be empty, + otherwise just a regular string. + type: string + type: object + type: array + type: object + registry: + description: 'Registry specifies the image registry to use rather + than Pixie''s default registry (gcr.io). We expect any forward slashes + in Pixie''s image paths are replaced with a "-". For example: "gcr.io/pixie-oss/pixie-dev/vizier/metadata_server_image:latest" + should be pushed to "$registry/gcr.io-pixie-oss-pixie-dev-vizier-metadata_server_image:latest".' + type: string + useEtcdOperator: + description: UseEtcdOperator specifies whether the metadata service + should use etcd for storage. + type: boolean + version: + description: Version is the desired version of the Vizier instance. + type: string + type: object + status: + description: VizierStatus defines the observed state of Vizier + properties: + checksum: + description: A checksum of the last reconciled Vizier spec. If this + checksum does not match the checksum of the current vizier spec, + reconciliation should be performed. + format: byte + type: string + lastReconciliationPhaseTime: + description: LastReconciliationPhaseTime is the last time that the + ReconciliationPhase changed. + format: date-time + type: string + message: + description: Message is a human-readable message with details about + why the Vizier is in this condition. + type: string + operatorVersion: + description: OperatorVersion is the actual version of the Operator + instance. + type: string + reconciliationPhase: + description: ReconciliationPhase describes the state the Reconciler + is in for this Vizier. See the documentation above the ReconciliationPhase + type for more information. + type: string + sentryDSN: + description: SentryDSN is key for Viziers that is used to send errors + and stacktraces to Sentry. + type: string + version: + description: Version is the actual version of the Vizier instance. + type: string + vizierPhase: + description: VizierPhase is a high-level summary of where the Vizier + is in its lifecycle. + type: string + vizierReason: + description: VizierReason is a short, machine understandable string + that gives the reason for the transition into the Vizier's current + status. + type: string + type: object + type: object + served: true + storage: true + subresources: + status: {} +status: + acceptedNames: + kind: "" + plural: "" + conditions: [] + storedVersions: [] diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/00_olm.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/00_olm.yaml new file mode 100644 index 0000000000..fe058140f1 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/00_olm.yaml @@ -0,0 +1,232 @@ +{{- $olmCRDFound := false }} +{{- $nsLookup := len (lookup "v1" "Namespace" "" "") }} +{{- range $index, $crdLookup := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" "").items -}}{{ if eq $crdLookup.metadata.name "operators.operators.coreos.com"}}{{ $olmCRDFound = true }}{{ end }}{{end}} +{{ if and (not $olmCRDFound) (not (eq $nsLookup 0))}}{{ fail "CRDs missing! Please deploy CRDs from https://github.com/pixie-io/pixie/tree/main/k8s/operator/helm/crds to continue with deploy." }}{{end}} +{{- $lookupLen := 0 -}}{{- $opLookup := (lookup "operators.coreos.com/v1" "OperatorGroup" "" "").items -}}{{if $opLookup }}{{ $lookupLen = len $opLookup }}{{ end }} +{{ if (or (eq (.Values.deployOLM | toString) "true") (and (not (eq (.Values.deployOLM | toString) "false")) (eq $lookupLen 0))) }} +{{ if not (eq .Values.olmNamespace .Release.Namespace) }} +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ .Values.olmNamespace }} +{{ end }} +--- +kind: ServiceAccount +apiVersion: v1 +metadata: + name: olm-operator-serviceaccount + namespace: {{ .Values.olmNamespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: system:controller:operator-lifecycle-manager +rules: +- apiGroups: ["*"] + resources: ["*"] + verbs: ["*"] +- nonResourceURLs: ["*"] + verbs: ["*"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: olm-operator-cluster-binding-olm +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: system:controller:operator-lifecycle-manager +subjects: +- kind: ServiceAccount + name: olm-operator-serviceaccount + namespace: {{ .Values.olmNamespace }} +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: olm-operator + namespace: {{ .Values.olmNamespace }} + labels: + app: olm-operator +spec: + strategy: + type: RollingUpdate + replicas: 1 + selector: + matchLabels: + app: olm-operator + template: + metadata: + labels: + app: olm-operator + spec: + serviceAccountName: olm-operator-serviceaccount + containers: + - name: olm-operator + command: + - /bin/olm + args: + - --namespace + - $(OPERATOR_NAMESPACE) + - --writeStatusName + - "" + image: {{ if .Values.registry }}{{ .Values.registry }}/quay.io-operator-framework-{{ else }}quay.io/operator-framework/{{ end }}olm@sha256:1b6002156f568d722c29138575733591037c24b4bfabc67946f268ce4752c3e6 + ports: + - containerPort: 8080 + - containerPort: 8081 + name: metrics + protocol: TCP + livenessProbe: + httpGet: + path: /healthz + port: 8080 + readinessProbe: + httpGet: + path: /healthz + port: 8080 + terminationMessagePolicy: FallbackToLogsOnError + env: + - name: OPERATOR_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: OPERATOR_NAME + value: olm-operator + resources: + requests: + cpu: 10m + memory: 160Mi + nodeSelector: + kubernetes.io/os: linux + tolerations: + - key: "kubernetes.io/arch" + operator: "Equal" + value: "amd64" + effect: "NoSchedule" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "amd64" + effect: "NoExecute" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "arm64" + effect: "NoSchedule" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "arm64" + effect: "NoExecute" +--- +apiVersion: apps/v1 +kind: Deployment +metadata: + name: catalog-operator + namespace: {{ .Values.olmNamespace }} + labels: + app: catalog-operator +spec: + strategy: + type: RollingUpdate + replicas: 1 + selector: + matchLabels: + app: catalog-operator + template: + metadata: + labels: + app: catalog-operator + spec: + serviceAccountName: olm-operator-serviceaccount + containers: + - name: catalog-operator + command: + - /bin/catalog + args: + - '--namespace' + - {{ .Values.olmNamespace }} + - --configmapServerImage={{ if .Values.registry }}{{ .Values.registry }}/quay.io-operator-framework-{{ else }}quay.io/operator-framework/{{ end }}configmap-operator-registry:latest + - --util-image + - {{ if .Values.registry }}{{ .Values.registry }}/quay.io-operator-framework-{{ else }}quay.io/operator-framework/{{ end }}olm@sha256:1b6002156f568d722c29138575733591037c24b4bfabc67946f268ce4752c3e6 + - --opmImage + - {{ if .Values.registry }}{{ .Values.registry }}/quay.io-operator-framework-{{ else }}quay.io/operator-framework/{{ end }}opm@sha256:d999588bd4e9509ec9e75e49adfb6582d256e9421e454c7fb5e9fe57e7b1aada + image: {{ if .Values.registry }}{{ .Values.registry }}/quay.io-operator-framework-{{ else }}quay.io/operator-framework/{{ end }}olm@sha256:1b6002156f568d722c29138575733591037c24b4bfabc67946f268ce4752c3e6 + imagePullPolicy: IfNotPresent + ports: + - containerPort: 8080 + - containerPort: 8081 + name: metrics + protocol: TCP + livenessProbe: + httpGet: + path: /healthz + port: 8080 + readinessProbe: + httpGet: + path: /healthz + port: 8080 + terminationMessagePolicy: FallbackToLogsOnError + env: + resources: + requests: + cpu: 10m + memory: 80Mi + nodeSelector: + kubernetes.io/os: linux + tolerations: + - key: "kubernetes.io/arch" + operator: "Equal" + value: "amd64" + effect: "NoSchedule" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "amd64" + effect: "NoExecute" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "arm64" + effect: "NoSchedule" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "arm64" + effect: "NoExecute" +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: aggregate-olm-edit + labels: + rbac.authorization.k8s.io/aggregate-to-admin: "true" + rbac.authorization.k8s.io/aggregate-to-edit: "true" +rules: +- apiGroups: ["operators.coreos.com"] + resources: ["subscriptions"] + verbs: ["create", "update", "patch", "delete"] +- apiGroups: ["operators.coreos.com"] + resources: ["clusterserviceversions", "catalogsources", "installplans", "subscriptions"] + verbs: ["delete"] +--- +kind: ClusterRole +apiVersion: rbac.authorization.k8s.io/v1 +metadata: + name: aggregate-olm-view + labels: + rbac.authorization.k8s.io/aggregate-to-admin: "true" + rbac.authorization.k8s.io/aggregate-to-edit: "true" + rbac.authorization.k8s.io/aggregate-to-view: "true" +rules: +- apiGroups: ["operators.coreos.com"] + resources: ["clusterserviceversions", "catalogsources", "installplans", "subscriptions", "operatorgroups"] + verbs: ["get", "list", "watch"] +- apiGroups: ["packages.operators.coreos.com"] + resources: ["packagemanifests", "packagemanifests/icon"] + verbs: ["get", "list", "watch"] +--- +apiVersion: operators.coreos.com/v1 +kind: OperatorGroup +metadata: + name: olm-operators + namespace: {{ .Values.olmNamespace }} +spec: + targetNamespaces: + - {{ .Values.olmNamespace }} +{{- end}} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/01_px_olm.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/01_px_olm.yaml new file mode 100644 index 0000000000..2c2921958e --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/01_px_olm.yaml @@ -0,0 +1,13 @@ +{{ if not (eq .Values.olmOperatorNamespace .Release.Namespace) }} +--- +apiVersion: v1 +kind: Namespace +metadata: + name: {{ .Values.olmOperatorNamespace }} +{{ end }} +--- +apiVersion: operators.coreos.com/v1 +kind: OperatorGroup +metadata: + name: global-operators + namespace: {{ .Values.olmOperatorNamespace }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/02_catalog.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/02_catalog.yaml new file mode 100644 index 0000000000..e7f68804a0 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/02_catalog.yaml @@ -0,0 +1,37 @@ +apiVersion: operators.coreos.com/v1alpha1 +kind: CatalogSource +metadata: + name: pixie-operator-index + namespace: {{ .Values.olmOperatorNamespace }} + {{- if .Values.olmCatalogSource.annotations }} + annotations: {{ .Values.olmCatalogSource.annotations | toYaml | nindent 4 }} + {{- end }} + {{- if .Values.olmCatalogSource.labels }} + labels: {{ .Values.olmCatalogSource.labels | toYaml | nindent 4 }} + {{- end }} +spec: + sourceType: grpc + image: {{ if .Values.registry }}{{ .Values.registry }}/gcr.io-pixie-oss-pixie-prod-operator-bundle_index:0.0.1{{ else }}gcr.io/pixie-oss/pixie-prod/operator/bundle_index:0.0.1{{ end }} + displayName: Pixie Vizier Operator + publisher: px.dev + updateStrategy: + registryPoll: + interval: 10m + grpcPodConfig: + tolerations: + - key: "kubernetes.io/arch" + operator: "Equal" + value: "amd64" + effect: "NoSchedule" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "amd64" + effect: "NoExecute" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "arm64" + effect: "NoSchedule" + - key: "kubernetes.io/arch" + operator: "Equal" + value: "arm64" + effect: "NoExecute" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/03_subscription.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/03_subscription.yaml new file mode 100644 index 0000000000..78223cc9e3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/03_subscription.yaml @@ -0,0 +1,11 @@ +apiVersion: operators.coreos.com/v1alpha1 +kind: Subscription +metadata: + name: pixie-operator-subscription + namespace: {{ .Values.olmOperatorNamespace }} +spec: + channel: {{ .Values.olmBundleChannel }} + name: pixie-operator + source: pixie-operator-index + sourceNamespace: {{ .Values.olmOperatorNamespace }} + installPlanApproval: Automatic diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/04_vizier.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/04_vizier.yaml new file mode 100644 index 0000000000..7c8ca65ad2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/04_vizier.yaml @@ -0,0 +1,100 @@ +apiVersion: px.dev/v1alpha1 +kind: Vizier +metadata: + name: {{ .Values.name }} + namespace: {{ .Release.Namespace }} +spec: + {{- if .Values.version }} + version: {{ .Values.version }} + {{- end }} + {{- if .Values.deployKey }} + deployKey: {{ .Values.deployKey }} + {{- end }} + {{- if .Values.customDeployKeySecret }} + customDeployKeySecret: {{ .Values.customDeployKeySecret }} + {{- end }} + cloudAddr: {{ .Values.cloudAddr }} + disableAutoUpdate: {{ .Values.disableAutoUpdate }} + useEtcdOperator: {{ .Values.useEtcdOperator }} + {{- if (.Values.global).cluster }} + clusterName: {{ .Values.global.cluster }} + {{- else if .Values.clusterName }} + clusterName: {{ .Values.clusterName }} + {{- end }} + {{- if .Values.devCloudNamespace }} + devCloudNamespace: {{ .Values.devCloudNamespace }} + {{- end }} + {{- if .Values.pemMemoryLimit }} + pemMemoryLimit: {{ .Values.pemMemoryLimit }} + {{- end }} + {{- if .Values.pemMemoryRequest }} + pemMemoryRequest: {{ .Values.pemMemoryRequest }} + {{- end }} + {{- if .Values.dataAccess }} + dataAccess: {{ .Values.dataAccess }} + {{- end }} + {{- if .Values.patches }} + patches: {{ .Values.patches | toYaml | nindent 4 }} + {{- end }} + {{- if ((.Values.global).images).registry }} + registry: {{ .Values.global.images.registry }} + {{- else if .Values.registry }} + registry: {{ .Values.registry }} + {{- end}} + {{- if .Values.autopilot }} + autopilot: {{ .Values.autopilot }} + {{- end}} + {{- if .Values.dataCollectorParams }} + dataCollectorParams: + {{- if .Values.dataCollectorParams.datastreamBufferSize }} + datastreamBufferSize: {{ .Values.dataCollectorParams.datastreamBufferSize }} + {{- end }} + {{- if .Values.dataCollectorParams.datastreamBufferSpikeSize }} + datastreamBufferSpikeSize: {{ .Values.dataCollectorParams.datastreamBufferSpikeSize }} + {{- end }} + {{- if .Values.dataCollectorParams.customPEMFlags }} + customPEMFlags: + {{- range $key, $value := .Values.dataCollectorParams.customPEMFlags}} + {{$key}}: "{{$value}}" + {{- end}} + {{- end }} + {{- end}} + {{- if .Values.leadershipElectionParams }} + leadershipElectionParams: + {{- if .Values.leadershipElectionParams.electionPeriodMs }} + electionPeriodMs: {{ .Values.leadershipElectionParams.electionPeriodMs }} + {{- end }} + {{- end }} + {{- if or .Values.pod.securityContext (or .Values.pod.nodeSelector (or .Values.pod.tolerations (or .Values.pod.annotations (or .Values.pod.labels .Values.pod.resources)))) }} + pod: + {{- if .Values.pod.annotations }} + annotations: {{ .Values.pod.annotations | toYaml | nindent 6 }} + {{- end }} + {{- if .Values.pod.labels }} + labels: {{ .Values.pod.labels | toYaml | nindent 6 }} + {{- end }} + {{- if .Values.pod.resources }} + resources: {{ .Values.pod.resources | toYaml | nindent 6 }} + {{- end }} + {{- if .Values.pod.nodeSelector }} + nodeSelector: {{ .Values.pod.nodeSelector | toYaml | nindent 6 }} + {{- end }} + {{- if .Values.pod.tolerations }} + tolerations: {{ .Values.pod.tolerations | toYaml | nindent 4 }} + {{- end }} + {{- if .Values.pod.securityContext }} + securityContext: + enabled: {{ .Values.pod.securityContext.enabled }} + {{- if .Values.pod.securityContext.enabled }} + {{- if .Values.pod.securityContext.fsGroup }} + fsGroup: {{ .Values.pod.securityContext.fsGroup }} + {{- end }} + {{- if .Values.pod.securityContext.runAsUser }} + runAsUser: {{ .Values.pod.securityContext.runAsUser }} + {{- end }} + {{- if .Values.pod.securityContext.runAsGroup }} + runAsGroup: {{ .Values.pod.securityContext.runAsGroup }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter.yaml new file mode 100644 index 0000000000..b1cde0c927 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter.yaml @@ -0,0 +1,25 @@ +apiVersion: batch/v1 +kind: Job +metadata: + annotations: + helm.sh/hook: pre-delete + helm.sh/hook-delete-policy: hook-succeeded + name: vizier-deleter + namespace: '{{ .Release.Namespace }}' +spec: + template: + metadata: + name: vizier-deleter + spec: + containers: + - env: + - name: PL_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + - name: PL_VIZIER_NAME + value: '{{ .Values.name }}' + image: gcr.io/pixie-oss/pixie-prod/operator-vizier_deleter:0.1.6 + name: delete-job + restartPolicy: Never + serviceAccountName: pl-deleter-service-account diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter_role.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter_role.yaml new file mode 100644 index 0000000000..73e5ec7e48 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/templates/deleter_role.yaml @@ -0,0 +1,77 @@ +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: pl-deleter-service-account +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: pl-deleter-cluster-binding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: pl-deleter-role +subjects: +- kind: ServiceAccount + name: pl-deleter-service-account + namespace: "{{ .Release.Namespace }}" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: pl-deleter-cluster-role +rules: +# Allow actions on Kubernetes objects +- apiGroups: + - rbac.authorization.k8s.io + - etcd.database.coreos.com + - nats.io + resources: + - clusterroles + - clusterrolebindings + - persistentvolumes + - etcdclusters + - natsclusters + verbs: ["*"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: pl-deleter-role +rules: +- apiGroups: + - "" + - apps + - rbac.authorization.k8s.io + - extensions + - batch + - policy + resources: + - configmaps + - secrets + - pods + - services + - deployments + - daemonsets + - persistentvolumes + - roles + - rolebindings + - serviceaccounts + - statefulsets + - cronjobs + - jobs + verbs: ["*"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: pl-deleter-binding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: pl-deleter-role +subjects: +- kind: ServiceAccount + name: pl-deleter-service-account + namespace: "{{ .Release.Namespace }}" diff --git a/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/values.yaml b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/values.yaml new file mode 100644 index 0000000000..a3ffe7c9d3 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/charts/pixie-operator-chart/values.yaml @@ -0,0 +1,75 @@ +## OLM configuration +# OLM is used for deploying and ensuring the operator is up-to-date. +# deployOLM indicates whether OLM should be deployed. This should only be +# disabled if an instance of OLM is already configured on the cluster. +# Should be string "true" if true, but "false" otherwise. If empty, defaults +# to whether OLM is present in the cluster. +deployOLM: "" +# The namespace that olm should run in. If olm has already been deployed +# to the cluster, this should be the namespace that olm is already running in. +olmNamespace: "olm" +# The namespace which olm operators should run in. If olm has already +# been deployed to the cluster, this should be the namespace that the olm operators +# are running in. +olmOperatorNamespace: "px-operator" +# The bundle channel which OLM should listen to for the Vizier operator bundles. +# Should be "stable" for production-versions of the operator, and "test" for release candidates. +olmBundleChannel: "stable" +# Optional annotations and labels for CatalogSource. +olmCatalogSource: + # Optional custom annotations to add to deployed pods managed by CatalogSource object. + annotations: {} + # Optional custom labels to add to deployed pods managed by CatalogSource object. + labels: {} +## Vizier configuration +# The name of the Vizier instance deployed to the cluster. +name: "pixie" +# The name of the cluster that the Vizier is monitoring. If empty, +# a random name will be generated. +clusterName: "" +# The version of the Vizier instance deployed to the cluster. If empty, +# the operator will automatically deploy the latest version. +version: "" +# The deploy key is used to link the deployed Vizier to a specific user/project. +# This is required if not specifying a customDeployKeySecret, and can be generated through the UI or CLI. +deployKey: "" +# The deploy key may be read from a custom secret in the Pixie namespace. This secret should be formatted where the +# key of the deploy key is "deploy-key". +customDeployKeySecret: "" +# Whether auto-update should be disabled. +disableAutoUpdate: false +# Whether the metadata service should use etcd for in-memory storage. Recommended +# only for clusters which do not have persistent volumes configured. +useEtcdOperator: false +# The address of the Pixie cloud instance that the Vizier should be connected to. +# This should only be updated when using a self-hosted version of Pixie Cloud. +cloudAddr: "withpixie.ai:443" +# DevCloudNamespace should be specified only for self-hosted versions of Pixie cloud which have no ingress to help +# redirect traffic to the correct service. The DevCloudNamespace is the namespace that the dev Pixie cloud is +# running on, for example: "plc-dev". +devCloudNamespace: "" +# A memory limit applied specifically to PEM pods. If none is specified, a default limit of 2Gi is set. +pemMemoryLimit: "" +# A memory request applied specifically to PEM pods. If none is specified, it will default to pemMemoryLimit. +pemMemoryRequest: "" +# DataAccess defines the level of data that may be accesssed when executing a script on the cluster. +dataAccess: "Full" +pod: + # Optional custom annotations to add to deployed pods. + annotations: {} + # Optional custom labels to add to deployed pods. + labels: {} + resources: {} + # limits: + # cpu: 500m + # memory: 7Gi + # requests: + # cpu: 100m + # memory: 5Gi + nodeSelector: {} + tolerations: [] +# A set of custom patches to apply to the deployed Vizier resources. +# The key should be the name of the resource to apply the patch to, and the value is the patch to apply. +# Currently, only a JSON format is accepted, such as: +# `{"spec": {"template": {"spec": { "tolerations": [{"key": "test", "operator": "Exists", "effect": "NoExecute" }]}}}}` +patches: {} diff --git a/charts/new-relic/nri-bundle/5.0.94/ci/test-values.yaml b/charts/new-relic/nri-bundle/5.0.94/ci/test-values.yaml new file mode 100644 index 0000000000..7ba6c8c329 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/ci/test-values.yaml @@ -0,0 +1,21 @@ +global: + licenseKey: 1234567890abcdef1234567890abcdef12345678 + cluster: test-cluster + +infrastructure: + enabled: true + +prometheus: + enabled: true + +webhook: + enabled: true + +ksm: + enabled: true + +kubeEvents: + enabled: true + +logging: + enabled: true diff --git a/charts/new-relic/nri-bundle/5.0.94/questions.yaml b/charts/new-relic/nri-bundle/5.0.94/questions.yaml new file mode 100644 index 0000000000..de3fa9fea2 --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/questions.yaml @@ -0,0 +1,113 @@ +questions: +- variable: infrastructure.enabled + default: true + required: false + type: boolean + label: Enable Infrastructure + group: "Select Components" +- variable: prometheus.enabled + default: false + required: false + type: boolean + label: Enable Prometheus + group: "Select Components" +- variable: ksm.enabled + default: false + required: false + type: boolean + label: Enable KSM + group: "Select Components" + description: "This is mandatory if `Enable Infrastructure` is set to `true` and the user does not provide its own instance of KSM version >=1.8 and <=2.0" +- variable: webhook.enabled + default: true + required: false + type: boolean + label: Enable webhook + group: "Select Components" +- variable: kubeEvents.enabled + default: false + required: false + type: boolean + label: Enable Kube Events + group: "Select Components" +- variable: logging.enabled + default: false + required: false + type: boolean + label: Enable Logging + group: "Select Components" +- variable: newrelic-pixie.enabled + default: false + required: false + type: boolean + label: Enable New Relic Pixie Integration + group: "Select Components" + show_subquestion_if: true + subquestions: + - variable: newrelic-pixie.apiKey + default: "" + required: false + type: string + label: New Relic Pixie API Key + group: "Select Components" + description: "Required if deploying Pixie." +- variable: pixie-chart.enabled + default: false + required: false + type: boolean + label: Enable Pixie Chart + group: "Select Components" + show_subquestion_if: true + subquestions: + - variable: pixie-chart.deployKey + default: "" + required: false + type: string + label: Pixie Deploy Key + group: "Select Components" + description: "Required if deploying Pixie." + - variable: pixie-chart.clusterName + default: "" + required: false + type: string + label: Kubernetes Cluster Name for Pixie + group: "Select Components" + description: "Required if deploying Pixie." +- variable: newrelic-infra-operator.enabled + default: false + required: false + type: boolean + label: Enable New Relic Infra Operator + group: "Select Components" +- variable: metrics-adapter.enabled + default: false + required: false + type: boolean + label: Enable Metrics Adapter + group: "Select Components" +- variable: global.licenseKey + default: "xxxx" + required: true + type: string + label: New Relic License Key + group: "Global Settings" +- variable: global.cluster + default: "xxxx" + required: true + type: string + label: Name of Kubernetes Cluster for New Relic + group: "Global Settings" +- variable: global.lowDataMode + default: false + required: false + type: boolean + label: Enable Low Data Mode + description: "Reduces amount of data ingest by New Relic." + group: "Global Settings" +- variable: global.privileged + default: false + required: false + type: boolean + label: Enable Privileged Mode + description: "Allows for access to underlying node from container." + group: "Global Settings" diff --git a/charts/new-relic/nri-bundle/5.0.94/values.yaml b/charts/new-relic/nri-bundle/5.0.94/values.yaml new file mode 100644 index 0000000000..47c58df8ec --- /dev/null +++ b/charts/new-relic/nri-bundle/5.0.94/values.yaml @@ -0,0 +1,169 @@ +newrelic-infrastructure: + # newrelic-infrastructure.enabled -- Install the [`newrelic-infrastructure` chart](https://github.com/newrelic/nri-kubernetes/tree/main/charts/newrelic-infrastructure) + enabled: true + +nri-prometheus: + # nri-prometheus.enabled -- Install the [`nri-prometheus` chart](https://github.com/newrelic/nri-prometheus/tree/main/charts/nri-prometheus) + enabled: false + +nri-metadata-injection: + # nri-metadata-injection.enabled -- Install the [`nri-metadata-injection` chart](https://github.com/newrelic/k8s-metadata-injection/tree/main/charts/nri-metadata-injection) + enabled: true + +kube-state-metrics: + # kube-state-metrics.enabled -- Install the [`kube-state-metrics` chart](https://github.com/prometheus-community/helm-charts/tree/main/charts/kube-state-metrics) from the stable helm charts repository. + # This is mandatory if `infrastructure.enabled` is set to `true` and the user does not provide its own instance of KSM version >=1.8 and <=2.0. Note, kube-state-metrics v2+ disables labels/annotations + # metrics by default. You can enable the target labels/annotations metrics to be monitored by using the metricLabelsAllowlist/metricAnnotationsAllowList options described [here](https://github.com/prometheus-community/helm-charts/blob/159cd8e4fb89b8b107dcc100287504bb91bf30e0/charts/kube-state-metrics/values.yaml#L274) in + # your Kubernetes clusters. + enabled: false + +nri-kube-events: + # nri-kube-events.enabled -- Install the [`nri-kube-events` chart](https://github.com/newrelic/nri-kube-events/tree/main/charts/nri-kube-events) + enabled: false + +newrelic-logging: + # newrelic-logging.enabled -- Install the [`newrelic-logging` chart](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-logging) + enabled: false + +newrelic-pixie: + # newrelic-pixie.enabled -- Install the [`newrelic-pixie`](https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-pixie) + enabled: false + +k8s-agents-operator: + # k8s-agents-operator.enabled -- Install the [`k8s-agents-operator` chart](https://github.com/newrelic/k8s-agents-operator/tree/main/charts/k8s-agents-operator) + enabled: false + +pixie-chart: + # pixie-chart.enabled -- Install the [`pixie-chart` chart](https://docs.pixielabs.ai/installing-pixie/install-schemes/helm/#3.-deploy) + enabled: false + +newrelic-infra-operator: + # newrelic-infra-operator.enabled -- Install the [`newrelic-infra-operator` chart](https://github.com/newrelic/newrelic-infra-operator/tree/main/charts/newrelic-infra-operator) (Beta) + enabled: false + +newrelic-prometheus-agent: + # newrelic-prometheus-agent.enabled -- Install the [`newrelic-prometheus-agent` chart](https://github.com/newrelic/newrelic-prometheus-configurator/tree/main/charts/newrelic-prometheus-agent) + enabled: false + +newrelic-k8s-metrics-adapter: + # newrelic-k8s-metrics-adapter.enabled -- Install the [`newrelic-k8s-metrics-adapter.` chart](https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/main/charts/newrelic-k8s-metrics-adapter) (Beta) + enabled: false + + +# -- change the behaviour globally to all the supported helm charts. +# See [user's guide of the common library](https://github.com/newrelic/helm-charts/blob/master/library/common-library/README.md) for further information. +# @default -- See [`values.yaml`](values.yaml) +global: + # -- The cluster name for the Kubernetes cluster. + cluster: "" + + # -- The license key for your New Relic Account. This will be preferred configuration option if both `licenseKey` and `customSecret` are specified. + licenseKey: "" + # -- The license key for your New Relic Account. This will be preferred configuration option if both `insightsKey` and `customSecret` are specified. + insightsKey: "" + # -- Name of the Secret object where the license key is stored + customSecretName: "" + # -- Key in the Secret object where the license key is stored + customSecretLicenseKey: "" + + # -- Additional labels for chart objects + labels: {} + # -- Additional labels for chart pods + podLabels: {} + + images: + # -- Changes the registry where to get the images. Useful when there is an internal image cache/proxy + registry: "" + # -- Set secrets to be able to fetch images + pullSecrets: [] + + serviceAccount: + # -- Add these annotations to the service account we create + annotations: {} + # -- Configures if the service account should be created or not + create: + # -- Change the name of the service account. This is honored if you disable on this chart the creation of the service account so you can use your own + name: + + # -- (bool) Sets pod's hostNetwork + # @default -- false + hostNetwork: + # -- Sets pod's dnsConfig + dnsConfig: {} + + # -- Sets pod's priorityClassName + priorityClassName: "" + # -- Sets security context (at pod level) + podSecurityContext: {} + # -- Sets security context (at container level) + containerSecurityContext: {} + + # -- Sets pod/node affinities + affinity: {} + # -- Sets pod's node selector + nodeSelector: {} + # -- Sets pod's tolerations to node taints + tolerations: [] + + # -- Adds extra attributes to the cluster and all the metrics emitted to the backend + customAttributes: {} + + # -- (bool) Reduces number of metrics sent in order to reduce costs + # @default -- false + lowDataMode: + + # -- (bool) In each integration it has different behavior. See [Further information](#values-managed-globally-3) but all aims to send less metrics to the backend to try to save costs | + # @default -- false + privileged: + + # -- (bool) Must be set to `true` when deploying in an EKS Fargate environment + # @default -- false + fargate: + + # -- Configures the integration to send all HTTP/HTTPS request through the proxy in that URL. The URL should have a standard format like `https://user:password@hostname:port` + proxy: "" + + # -- (bool) Send the metrics to the staging backend. Requires a valid staging license key + # @default -- false + nrStaging: + fedramp: + # fedramp.enabled -- (bool) Enables FedRAMP + # @default -- false + enabled: + + # -- (bool) Sets the debug logs to this integration or all integrations if it is set globally + # @default -- false + verboseLog: + + +# To add values to the subcharts. Follow Helm's guide: https://helm.sh/docs/chart_template_guide/subcharts_and_globals + +# If you wish to monitor services running on Kubernetes you can provide integrations +# configuration under `integrations_config` that it will passed down to the `newrelic-infrastructure` chart. +# +# You just need to create a new entry where the "name" is the filename of the configuration file and the data is the content of +# the integration configuration. The name must end in ".yaml" as this will be the +# filename generated and the Infrastructure agent only looks for YAML files. +# +# The data part is the actual integration configuration as described in the spec here: +# https://docs.newrelic.com/docs/integrations/integrations-sdk/file-specifications/integration-configuration-file-specifications-agent-v180 +# +# In the following example you can see how to monitor a Redis integration with autodiscovery +# +# +# newrelic-infrastructure: +# integrations: +# nri-redis-sampleapp: +# discovery: +# command: +# exec: /var/db/newrelic-infra/nri-discovery-kubernetes --tls --port 10250 +# match: +# label.app: sampleapp +# integrations: +# - name: nri-redis +# env: +# # using the discovered IP as the hostname address +# HOSTNAME: ${discovery.ip} +# PORT: 6379 +# labels: +# env: test diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/.helmignore b/charts/stackstate/stackstate-k8s-agent/1.0.98/.helmignore new file mode 100644 index 0000000000..15a5c12775 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/.helmignore @@ -0,0 +1,26 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ +linter_values.yaml +ci/ +installation/ +logo.svg diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.lock b/charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.lock new file mode 100644 index 0000000000..a8df83f136 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.lock @@ -0,0 +1,6 @@ +dependencies: +- name: http-header-injector + repository: https://helm.stackstate.io + version: 0.0.11 +digest: sha256:ae5ad7c3176f89b71aabef7cd75f99394750f4fffb9905b86fb45c345595c24c +generated: "2024-05-30T13:30:45.346757+02:00" diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.yaml new file mode 100644 index 0000000000..b5973dd895 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/Chart.yaml @@ -0,0 +1,25 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: StackState Agent + catalog.cattle.io/kube-version: '>=1.19.0-0' + catalog.cattle.io/release-name: stackstate-k8s-agent +apiVersion: v2 +appVersion: 3.0.0 +dependencies: +- alias: httpHeaderInjectorWebhook + name: http-header-injector + repository: https://helm.stackstate.io + version: 0.0.11 +description: Helm chart for the StackState Agent. +home: https://github.com/StackVista/stackstate-agent +icon: file://assets/icons/stackstate-k8s-agent.svg +keywords: +- monitoring +- observability +- stackstate +kubeVersion: '>=1.19.0-0' +maintainers: +- email: ops@stackstate.com + name: Stackstate +name: stackstate-k8s-agent +version: 1.0.98 diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/README.md b/charts/stackstate/stackstate-k8s-agent/1.0.98/README.md new file mode 100644 index 0000000000..a8d6620350 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/README.md @@ -0,0 +1,263 @@ +# stackstate-k8s-agent + +Helm chart for the StackState Agent. + +Current chart version is `1.0.98` + +**Homepage:** + +## Requirements + +| Repository | Name | Version | +|------------|------|---------| +| https://helm.stackstate.io | httpHeaderInjectorWebhook(http-header-injector) | 0.0.11 | + +## Required Values + +In order to successfully install this chart, you **must** provide the following variables: + +* `stackstate.apiKey` +* `stackstate.cluster.name` +* `stackstate.url` + +The parameter `stackstate.cluster.name` is entered when installing the Cluster Agent StackPack. + +Install them on the command line on Helm with the following command: + +```shell +helm install \ +--set-string 'stackstate.apiKey'='' \ +--set-string 'stackstate.cluster.name'='' \ +--set-string 'stackstate.url'='' \ +stackstate/stackstate-k8s-agent +``` + +## Recommended Values + +It is also recommended that you set a value for `stackstate.cluster.authToken`. If it is not provided, a value will be generated for you, but the value will change each time an upgrade is performed. + +The command for **also** installing with a set token would be: + +```shell +helm install \ +--set-string 'stackstate.apiKey'='' \ +--set-string 'stackstate.cluster.name'='' \ +--set-string 'stackstate.cluster.authToken'='' \ +--set-string 'stackstate.url'='' \ +stackstate/stackstate-k8s-agent +``` + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| all.hardening.enabled | bool | `false` | An indication of whether the containers will be evaluated for hardening at runtime | +| all.image.registry | string | `"quay.io"` | The image registry to use. | +| checksAgent.affinity | object | `{}` | Affinity settings for pod assignment. | +| checksAgent.apm.enabled | bool | `true` | Enable / disable the agent APM module. | +| checksAgent.checksTagCardinality | string | `"orchestrator"` | | +| checksAgent.config | object | `{"override":[]}` | | +| checksAgent.config.override | list | `[]` | A list of objects containing three keys `name`, `path` and `data`, specifying filenames at specific paths which need to be (potentially) overridden using a mounted configmap | +| checksAgent.enabled | bool | `true` | Enable / disable runnning cluster checks in a separately deployed pod | +| checksAgent.image.pullPolicy | string | `"IfNotPresent"` | Default container image pull policy. | +| checksAgent.image.repository | string | `"stackstate/stackstate-k8s-agent"` | Base container image repository. | +| checksAgent.image.tag | string | `"c4caacef"` | Default container image tag. | +| checksAgent.livenessProbe.enabled | bool | `true` | Enable use of livenessProbe check. | +| checksAgent.livenessProbe.failureThreshold | int | `3` | `failureThreshold` for the liveness probe. | +| checksAgent.livenessProbe.initialDelaySeconds | int | `15` | `initialDelaySeconds` for the liveness probe. | +| checksAgent.livenessProbe.periodSeconds | int | `15` | `periodSeconds` for the liveness probe. | +| checksAgent.livenessProbe.successThreshold | int | `1` | `successThreshold` for the liveness probe. | +| checksAgent.livenessProbe.timeoutSeconds | int | `5` | `timeoutSeconds` for the liveness probe. | +| checksAgent.logLevel | string | `"INFO"` | Logging level for clusterchecks agent processes. | +| checksAgent.networkTracing.enabled | bool | `true` | Enable / disable the agent network tracing module. | +| checksAgent.nodeSelector | object | `{}` | Node labels for pod assignment. | +| checksAgent.priorityClassName | string | `""` | Priority class for clusterchecks agent pods. | +| checksAgent.processAgent.enabled | bool | `true` | Enable / disable the agent process agent module. | +| checksAgent.readinessProbe.enabled | bool | `true` | Enable use of readinessProbe check. | +| checksAgent.readinessProbe.failureThreshold | int | `3` | `failureThreshold` for the readiness probe. | +| checksAgent.readinessProbe.initialDelaySeconds | int | `15` | `initialDelaySeconds` for the readiness probe. | +| checksAgent.readinessProbe.periodSeconds | int | `15` | `periodSeconds` for the readiness probe. | +| checksAgent.readinessProbe.successThreshold | int | `1` | `successThreshold` for the readiness probe. | +| checksAgent.readinessProbe.timeoutSeconds | int | `5` | `timeoutSeconds` for the readiness probe. | +| checksAgent.replicas | int | `1` | Number of clusterchecks agent pods to schedule | +| checksAgent.resources.limits.cpu | string | `"400m"` | CPU resource limits. | +| checksAgent.resources.limits.memory | string | `"600Mi"` | Memory resource limits. | +| checksAgent.resources.requests.cpu | string | `"20m"` | CPU resource requests. | +| checksAgent.resources.requests.memory | string | `"512Mi"` | Memory resource requests. | +| checksAgent.scc.enabled | bool | `false` | Enable / disable the installation of the SecurityContextConfiguration needed for installation on OpenShift | +| checksAgent.serviceaccount.annotations | object | `{}` | Annotations for the service account for the cluster checks pods | +| checksAgent.skipSslValidation | bool | `false` | Set to true if self signed certificates are used. | +| checksAgent.strategy | object | `{"type":"RollingUpdate"}` | The strategy for the Deployment object. | +| checksAgent.tolerations | list | `[]` | Toleration labels for pod assignment. | +| clusterAgent.affinity | object | `{}` | Affinity settings for pod assignment. | +| clusterAgent.collection.kubeStateMetrics.annotationsAsTags | object | `{}` | Extra annotations to collect from resources and to turn into StackState tag. | +| clusterAgent.collection.kubeStateMetrics.clusterCheck | bool | `false` | For large clusters where the Kubernetes State Metrics Check Core needs to be distributed on dedicated workers. | +| clusterAgent.collection.kubeStateMetrics.enabled | bool | `true` | Enable / disable the cluster agent kube-state-metrics collection. | +| clusterAgent.collection.kubeStateMetrics.labelsAsTags | object | `{}` | Extra labels to collect from resources and to turn into StackState tag. # It has the following structure: # labelsAsTags: # : # can be pod, deployment, node, etc. # : # where is the kubernetes label and is the StackState tag # : # : # : # # Warning: the label must match the transformation done by kube-state-metrics, # for example tags.stackstate/version becomes tags_stackstate_version. | +| clusterAgent.collection.kubernetesEvents | bool | `true` | Enable / disable the cluster agent events collection. | +| clusterAgent.collection.kubernetesMetrics | bool | `true` | Enable / disable the cluster agent metrics collection. | +| clusterAgent.collection.kubernetesResources.configmaps | bool | `true` | Enable / disable collection of ConfigMaps. | +| clusterAgent.collection.kubernetesResources.cronjobs | bool | `true` | Enable / disable collection of CronJobs. | +| clusterAgent.collection.kubernetesResources.daemonsets | bool | `true` | Enable / disable collection of DaemonSets. | +| clusterAgent.collection.kubernetesResources.deployments | bool | `true` | Enable / disable collection of Deployments. | +| clusterAgent.collection.kubernetesResources.endpoints | bool | `true` | Enable / disable collection of Endpoints. If endpoints are disabled then StackState won't be able to connect a Service to Pods that serving it | +| clusterAgent.collection.kubernetesResources.horizontalpodautoscalers | bool | `true` | Enable / disable collection of HorizontalPodAutoscalers. | +| clusterAgent.collection.kubernetesResources.ingresses | bool | `true` | Enable / disable collection of Ingresses. | +| clusterAgent.collection.kubernetesResources.jobs | bool | `true` | Enable / disable collection of Jobs. | +| clusterAgent.collection.kubernetesResources.limitranges | bool | `true` | Enable / disable collection of LimitRanges. | +| clusterAgent.collection.kubernetesResources.namespaces | bool | `true` | Enable / disable collection of Namespaces. | +| clusterAgent.collection.kubernetesResources.persistentvolumeclaims | bool | `true` | Enable / disable collection of PersistentVolumeClaims. Disabling these will not let StackState connect PersistentVolumes to pods they are attached to | +| clusterAgent.collection.kubernetesResources.persistentvolumes | bool | `true` | Enable / disable collection of PersistentVolumes. | +| clusterAgent.collection.kubernetesResources.poddisruptionbudgets | bool | `true` | Enable / disable collection of PodDisruptionBudgets. | +| clusterAgent.collection.kubernetesResources.replicasets | bool | `true` | Enable / disable collection of ReplicaSets. | +| clusterAgent.collection.kubernetesResources.replicationcontrollers | bool | `true` | Enable / disable collection of ReplicationControllers. | +| clusterAgent.collection.kubernetesResources.resourcequotas | bool | `true` | Enable / disable collection of ResourceQuotas. | +| clusterAgent.collection.kubernetesResources.secrets | bool | `true` | Enable / disable collection of Secrets. | +| clusterAgent.collection.kubernetesResources.statefulsets | bool | `true` | Enable / disable collection of StatefulSets. | +| clusterAgent.collection.kubernetesResources.storageclasses | bool | `true` | Enable / disable collection of StorageClasses. | +| clusterAgent.collection.kubernetesResources.volumeattachments | bool | `true` | Enable / disable collection of Volume Attachments. Used to bind Nodes to Persistent Volumes. | +| clusterAgent.collection.kubernetesTimeout | int | `10` | Default timeout (in seconds) when obtaining information from the Kubernetes API. | +| clusterAgent.collection.kubernetesTopology | bool | `true` | Enable / disable the cluster agent topology collection. | +| clusterAgent.config | object | `{"configMap":{"maxDataSize":null},"events":{"categories":{}},"override":[],"topology":{"collectionInterval":90}}` | | +| clusterAgent.config.configMap.maxDataSize | string | `nil` | Maximum amount of characters for the data property of a ConfigMap collected by the kubernetes topology check | +| clusterAgent.config.events.categories | object | `{}` | Custom mapping from Kubernetes event reason to StackState event category. Categories allowed: Alerts, Activities, Changes, Others | +| clusterAgent.config.override | list | `[]` | A list of objects containing three keys `name`, `path` and `data`, specifying filenames at specific paths which need to be (potentially) overridden using a mounted configmap | +| clusterAgent.config.topology.collectionInterval | int | `90` | Interval for running topology collection, in seconds | +| clusterAgent.enabled | bool | `true` | Enable / disable the cluster agent. | +| clusterAgent.image.pullPolicy | string | `"IfNotPresent"` | Default container image pull policy. | +| clusterAgent.image.repository | string | `"stackstate/stackstate-k8s-cluster-agent"` | Base container image repository. | +| clusterAgent.image.tag | string | `"c4caacef"` | Default container image tag. | +| clusterAgent.livenessProbe.enabled | bool | `true` | Enable use of livenessProbe check. | +| clusterAgent.livenessProbe.failureThreshold | int | `3` | `failureThreshold` for the liveness probe. | +| clusterAgent.livenessProbe.initialDelaySeconds | int | `15` | `initialDelaySeconds` for the liveness probe. | +| clusterAgent.livenessProbe.periodSeconds | int | `15` | `periodSeconds` for the liveness probe. | +| clusterAgent.livenessProbe.successThreshold | int | `1` | `successThreshold` for the liveness probe. | +| clusterAgent.livenessProbe.timeoutSeconds | int | `5` | `timeoutSeconds` for the liveness probe. | +| clusterAgent.logLevel | string | `"INFO"` | Logging level for stackstate-k8s-agent processes. | +| clusterAgent.nodeSelector | object | `{}` | Node labels for pod assignment. | +| clusterAgent.priorityClassName | string | `""` | Priority class for stackstate-k8s-agent pods. | +| clusterAgent.readinessProbe.enabled | bool | `true` | Enable use of readinessProbe check. | +| clusterAgent.readinessProbe.failureThreshold | int | `3` | `failureThreshold` for the readiness probe. | +| clusterAgent.readinessProbe.initialDelaySeconds | int | `15` | `initialDelaySeconds` for the readiness probe. | +| clusterAgent.readinessProbe.periodSeconds | int | `15` | `periodSeconds` for the readiness probe. | +| clusterAgent.readinessProbe.successThreshold | int | `1` | `successThreshold` for the readiness probe. | +| clusterAgent.readinessProbe.timeoutSeconds | int | `5` | `timeoutSeconds` for the readiness probe. | +| clusterAgent.replicaCount | int | `1` | Number of replicas of the cluster agent to deploy. | +| clusterAgent.resources.limits.cpu | string | `"400m"` | CPU resource limits. | +| clusterAgent.resources.limits.memory | string | `"800Mi"` | Memory resource limits. | +| clusterAgent.resources.requests.cpu | string | `"70m"` | CPU resource requests. | +| clusterAgent.resources.requests.memory | string | `"512Mi"` | Memory resource requests. | +| clusterAgent.service.port | int | `5005` | Change the Cluster Agent service port | +| clusterAgent.service.targetPort | int | `5005` | Change the Cluster Agent service targetPort | +| clusterAgent.serviceaccount.annotations | object | `{}` | Annotations for the service account for the cluster agent pods | +| clusterAgent.skipSslValidation | bool | `false` | If true, ignores the server certificate being signed by an unknown authority. | +| clusterAgent.strategy | object | `{"type":"RollingUpdate"}` | The strategy for the Deployment object. | +| clusterAgent.tolerations | list | `[]` | Toleration labels for pod assignment. | +| fullnameOverride | string | `""` | Override the fullname of the chart. | +| global.extraAnnotations | object | `{}` | Extra annotations added ta all resources created by the helm chart | +| global.extraEnv.open | object | `{}` | Extra open environment variables to inject into pods. | +| global.extraEnv.secret | object | `{}` | Extra secret environment variables to inject into pods via a `Secret` object. | +| global.extraLabels | object | `{}` | Extra labels added ta all resources created by the helm chart | +| global.imagePullCredentials | object | `{}` | Globally define credentials for pulling images. | +| global.imagePullSecrets | list | `[]` | Secrets / credentials needed for container image registry. | +| global.proxy.url | string | `""` | Proxy for all traffic to stackstate | +| global.skipSslValidation | bool | `false` | Enable tls validation from client | +| httpHeaderInjectorWebhook.enabled | bool | `false` | Enable the webhook for injection http header injection sidecar proxy | +| logsAgent.affinity | object | `{}` | Affinity settings for pod assignment. | +| logsAgent.enabled | bool | `true` | Enable / disable k8s pod log collection | +| logsAgent.image.pullPolicy | string | `"IfNotPresent"` | Default container image pull policy. | +| logsAgent.image.repository | string | `"stackstate/promtail"` | Base container image repository. | +| logsAgent.image.tag | string | `"2.9.8-5b179aee"` | Default container image tag. | +| logsAgent.nodeSelector | object | `{}` | Node labels for pod assignment. | +| logsAgent.priorityClassName | string | `""` | Priority class for logsAgent pods. | +| logsAgent.resources.limits.cpu | string | `"1300m"` | CPU resource limits. | +| logsAgent.resources.limits.memory | string | `"192Mi"` | Memory resource limits. | +| logsAgent.resources.requests.cpu | string | `"20m"` | CPU resource requests. | +| logsAgent.resources.requests.memory | string | `"100Mi"` | Memory resource requests. | +| logsAgent.serviceaccount.annotations | object | `{}` | Annotations for the service account for the daemonset pods | +| logsAgent.skipSslValidation | bool | `false` | If true, ignores the server certificate being signed by an unknown authority. | +| logsAgent.tolerations | list | `[]` | Toleration labels for pod assignment. | +| logsAgent.updateStrategy | object | `{"rollingUpdate":{"maxUnavailable":100},"type":"RollingUpdate"}` | The update strategy for the DaemonSet object. | +| nameOverride | string | `""` | Override the name of the chart. | +| nodeAgent.affinity | object | `{}` | Affinity settings for pod assignment. | +| nodeAgent.apm.enabled | bool | `true` | Enable / disable the nodeAgent APM module. | +| nodeAgent.autoScalingEnabled | bool | `false` | Enable / disable autoscaling for the node agent pods. | +| nodeAgent.checksTagCardinality | string | `"orchestrator"` | low, orchestrator or high. Orchestrator level adds pod_name, high adds display_container_name | +| nodeAgent.config | object | `{"override":[]}` | | +| nodeAgent.config.override | list | `[]` | A list of objects containing three keys `name`, `path` and `data`, specifying filenames at specific paths which need to be (potentially) overridden using a mounted configmap | +| nodeAgent.containerRuntime.customSocketPath | string | `""` | If the container socket path does not match the default for CRI-O, Containerd or Docker, supply a custom socket path. | +| nodeAgent.containerRuntime.hostProc | string | `"/proc"` | | +| nodeAgent.containers.agent.env | object | `{}` | Additional environment variables for the agent container | +| nodeAgent.containers.agent.image.pullPolicy | string | `"IfNotPresent"` | Default container image pull policy. | +| nodeAgent.containers.agent.image.repository | string | `"stackstate/stackstate-k8s-agent"` | Base container image repository. | +| nodeAgent.containers.agent.image.tag | string | `"c4caacef"` | Default container image tag. | +| nodeAgent.containers.agent.livenessProbe.enabled | bool | `true` | Enable use of livenessProbe check. | +| nodeAgent.containers.agent.livenessProbe.failureThreshold | int | `3` | `failureThreshold` for the liveness probe. | +| nodeAgent.containers.agent.livenessProbe.initialDelaySeconds | int | `15` | `initialDelaySeconds` for the liveness probe. | +| nodeAgent.containers.agent.livenessProbe.periodSeconds | int | `15` | `periodSeconds` for the liveness probe. | +| nodeAgent.containers.agent.livenessProbe.successThreshold | int | `1` | `successThreshold` for the liveness probe. | +| nodeAgent.containers.agent.livenessProbe.timeoutSeconds | int | `5` | `timeoutSeconds` for the liveness probe. | +| nodeAgent.containers.agent.logLevel | string | `nil` | Set logging verbosity, valid log levels are: trace, debug, info, warn, error, critical, and off # If not set, fall back to the value of agent.logLevel. | +| nodeAgent.containers.agent.processAgent.enabled | bool | `false` | Enable / disable the agent process agent module. - deprecated | +| nodeAgent.containers.agent.readinessProbe.enabled | bool | `true` | Enable use of readinessProbe check. | +| nodeAgent.containers.agent.readinessProbe.failureThreshold | int | `3` | `failureThreshold` for the readiness probe. | +| nodeAgent.containers.agent.readinessProbe.initialDelaySeconds | int | `15` | `initialDelaySeconds` for the readiness probe. | +| nodeAgent.containers.agent.readinessProbe.periodSeconds | int | `15` | `periodSeconds` for the readiness probe. | +| nodeAgent.containers.agent.readinessProbe.successThreshold | int | `1` | `successThreshold` for the readiness probe. | +| nodeAgent.containers.agent.readinessProbe.timeoutSeconds | int | `5` | `timeoutSeconds` for the readiness probe. | +| nodeAgent.containers.agent.resources.limits.cpu | string | `"270m"` | CPU resource limits. | +| nodeAgent.containers.agent.resources.limits.memory | string | `"420Mi"` | Memory resource limits. | +| nodeAgent.containers.agent.resources.requests.cpu | string | `"20m"` | CPU resource requests. | +| nodeAgent.containers.agent.resources.requests.memory | string | `"180Mi"` | Memory resource requests. | +| nodeAgent.containers.processAgent.enabled | bool | `true` | Enable / disable the process agent container. | +| nodeAgent.containers.processAgent.env | object | `{}` | Additional environment variables for the process-agent container | +| nodeAgent.containers.processAgent.image.pullPolicy | string | `"IfNotPresent"` | Process-agent container image pull policy. | +| nodeAgent.containers.processAgent.image.registry | string | `nil` | | +| nodeAgent.containers.processAgent.image.repository | string | `"stackstate/stackstate-k8s-process-agent"` | Process-agent container image repository. | +| nodeAgent.containers.processAgent.image.tag | string | `"cae7a4fa"` | Default process-agent container image tag. | +| nodeAgent.containers.processAgent.logLevel | string | `nil` | Set logging verbosity, valid log levels are: trace, debug, info, warn, error, critical, and off # If not set, fall back to the value of agent.logLevel. | +| nodeAgent.containers.processAgent.procVolumeReadOnly | bool | `true` | Configure whether /host/proc is read only for the process agent container | +| nodeAgent.containers.processAgent.resources.limits.cpu | string | `"125m"` | CPU resource limits. | +| nodeAgent.containers.processAgent.resources.limits.memory | string | `"400Mi"` | Memory resource limits. | +| nodeAgent.containers.processAgent.resources.requests.cpu | string | `"25m"` | CPU resource requests. | +| nodeAgent.containers.processAgent.resources.requests.memory | string | `"128Mi"` | Memory resource requests. | +| nodeAgent.httpTracing.enabled | bool | `true` | | +| nodeAgent.logLevel | string | `"INFO"` | Logging level for agent processes. | +| nodeAgent.networkTracing.enabled | bool | `true` | Enable / disable the nodeAgent network tracing module. | +| nodeAgent.nodeSelector | object | `{}` | Node labels for pod assignment. | +| nodeAgent.priorityClassName | string | `""` | Priority class for nodeAgent pods. | +| nodeAgent.protocolInspection.enabled | bool | `true` | Enable / disable the nodeAgent protocol inspection. | +| nodeAgent.scaling.autoscalerLimits.agent.maximum.cpu | string | `"200m"` | Maximum CPU resource limits for main agent. | +| nodeAgent.scaling.autoscalerLimits.agent.maximum.memory | string | `"450Mi"` | Maximum memory resource limits for main agent. | +| nodeAgent.scaling.autoscalerLimits.agent.minimum.cpu | string | `"20m"` | Minimum CPU resource limits for main agent. | +| nodeAgent.scaling.autoscalerLimits.agent.minimum.memory | string | `"180Mi"` | Minimum memory resource limits for main agent. | +| nodeAgent.scaling.autoscalerLimits.processAgent.maximum.cpu | string | `"200m"` | Maximum CPU resource limits for process agent. | +| nodeAgent.scaling.autoscalerLimits.processAgent.maximum.memory | string | `"500Mi"` | Maximum memory resource limits for process agent. | +| nodeAgent.scaling.autoscalerLimits.processAgent.minimum.cpu | string | `"25m"` | Minimum CPU resource limits for process agent. | +| nodeAgent.scaling.autoscalerLimits.processAgent.minimum.memory | string | `"100Mi"` | Minimum memory resource limits for process agent. | +| nodeAgent.scc.enabled | bool | `false` | Enable / disable the installation of the SecurityContextConfiguration needed for installation on OpenShift. | +| nodeAgent.service | object | `{"annotations":{},"loadBalancerSourceRanges":["10.0.0.0/8"],"type":"ClusterIP"}` | The Kubernetes service for the agent | +| nodeAgent.service.annotations | object | `{}` | Annotations for the service | +| nodeAgent.service.loadBalancerSourceRanges | list | `["10.0.0.0/8"]` | The IP4 CIDR allowed to reach LoadBalancer for the service. For LoadBalancer type of service only. | +| nodeAgent.service.type | string | `"ClusterIP"` | Type of Kubernetes service: ClusterIP, LoadBalancer, NodePort | +| nodeAgent.serviceaccount.annotations | object | `{}` | Annotations for the service account for the agent daemonset pods | +| nodeAgent.skipKubeletTLSVerify | bool | `false` | Set to true if you want to skip kubelet tls verification. | +| nodeAgent.skipSslValidation | bool | `false` | Set to true if self signed certificates are used. | +| nodeAgent.tolerations | list | `[]` | Toleration labels for pod assignment. | +| nodeAgent.updateStrategy | object | `{"rollingUpdate":{"maxUnavailable":100},"type":"RollingUpdate"}` | The update strategy for the DaemonSet object. | +| openShiftLogging.installSecret | bool | `false` | Install a secret for logging on openshift | +| processAgent.checkIntervals.connections | int | `30` | Override the default value of the connections check interval in seconds. | +| processAgent.checkIntervals.container | int | `28` | Override the default value of the container check interval in seconds. | +| processAgent.checkIntervals.process | int | `32` | Override the default value of the process check interval in seconds. | +| processAgent.softMemoryLimit.goMemLimit | string | `"340MiB"` | Soft-limit for golang heap allocation, for sanity, must be around 85% of nodeAgent.containers.processAgent.resources.limits.cpu. | +| processAgent.softMemoryLimit.httpObservationsBufferSize | int | `40000` | Sets a maximum for the number of http observations to keep in memory between check runs, to use 40k requires around ~400Mib of memory. | +| processAgent.softMemoryLimit.httpStatsBufferSize | int | `40000` | Sets a maximum for the number of http stats to keep in memory between check runs, to use 40k requires around ~400Mib of memory. | +| stackstate.apiKey | string | `nil` | **PROVIDE YOUR API KEY HERE** API key to be used by the StackState agent. | +| stackstate.cluster.authToken | string | `""` | Provide a token to enable secure communication between the agent and the cluster agent. | +| stackstate.cluster.name | string | `nil` | **PROVIDE KUBERNETES CLUSTER NAME HERE** Name of the Kubernetes cluster where the agent will be installed. | +| stackstate.customApiKeySecretKey | string | `"sts-api-key"` | Key in the secret containing the receiver API key. | +| stackstate.customClusterAuthTokenSecretKey | string | `"sts-cluster-auth-token"` | Key in the secret containing the cluster auth token. | +| stackstate.customSecretName | string | `""` | Name of the secret containing the receiver API key. | +| stackstate.manageOwnSecrets | bool | `false` | Set to true if you don't want this helm chart to create secrets for you. | +| stackstate.url | string | `nil` | **PROVIDE STACKSTATE URL HERE** URL of the StackState installation to receive data from the agent. | +| targetSystem | string | `"linux"` | Target OS for this deployment (possible values: linux) | diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/README.md.gotmpl b/charts/stackstate/stackstate-k8s-agent/1.0.98/README.md.gotmpl new file mode 100644 index 0000000000..7909e6f0d7 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/README.md.gotmpl @@ -0,0 +1,45 @@ +{{ template "chart.header" . }} +{{ template "chart.description" . }} + +Current chart version is `{{ template "chart.version" . }}` + +{{ template "chart.homepageLine" . }} + +{{ template "chart.requirementsSection" . }} + +## Required Values + +In order to successfully install this chart, you **must** provide the following variables: + +* `stackstate.apiKey` +* `stackstate.cluster.name` +* `stackstate.url` + +The parameter `stackstate.cluster.name` is entered when installing the Cluster Agent StackPack. + +Install them on the command line on Helm with the following command: + +```shell +helm install \ +--set-string 'stackstate.apiKey'='' \ +--set-string 'stackstate.cluster.name'='' \ +--set-string 'stackstate.url'='' \ +stackstate/stackstate-k8s-agent +``` + +## Recommended Values + +It is also recommended that you set a value for `stackstate.cluster.authToken`. If it is not provided, a value will be generated for you, but the value will change each time an upgrade is performed. + +The command for **also** installing with a set token would be: + +```shell +helm install \ +--set-string 'stackstate.apiKey'='' \ +--set-string 'stackstate.cluster.name'='' \ +--set-string 'stackstate.cluster.authToken'='' \ +--set-string 'stackstate.url'='' \ +stackstate/stackstate-k8s-agent +``` + +{{ template "chart.valuesSection" . }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/Releasing.md b/charts/stackstate/stackstate-k8s-agent/1.0.98/Releasing.md new file mode 100644 index 0000000000..bab6c2b944 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/Releasing.md @@ -0,0 +1,15 @@ +To make a new release of this helm chart, follow the following steps: + + +- Create a branch from master +- Set the latest tags for the docker images, based on the dev settings (while we do not promote to prod, the moment we promote to prod we should take those tags) from https://gitlab.com/stackvista/devops/agent-promoter/-/blob/master/config.yml. Set the value to the folowing keys: + * stackstate-k8s-cluster-agent: + * [clusterAgent.image.tag] + * stackstate-k8s-agent: + * [nodeAgent.containers.agent.image.tag] + * [checksAgent.image.tag] + * stackstate-k8s-process-agent: + * [nodeAgent.containers.processAgent.image.tag] +- Bump the version of the chart +- Merge the mr and hit the public release button on the ci pipeline +- Manually smoke-test (deploy) the newly released stackstate/stackstate-k8s-agent chart to make sure it runs diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/app-readme.md b/charts/stackstate/stackstate-k8s-agent/1.0.98/app-readme.md new file mode 100644 index 0000000000..8025fe1d36 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/app-readme.md @@ -0,0 +1,5 @@ +## Introduction + +StackState is a modern Application Troubleshooting and Observability solution designed for the rapid evolving engineering landscape. With specific enhancements for Kubernetes environments it empowers engineers, allowing them to remediate application issues independently in production. + +The StackState Agent auto-discovers your entire environment in minutes, assimilating topology, logs, metrics, and events and sends this of to the StackState server. By using StackState you're able to tracke all activity in your environment in real-time and over time. StackState provides instant understanding of the business impact of an issue, offering end-to-end chain observability and ensuring that you can quickly correlate any product or environmental changes to the overall health of your cloud-native implementation. diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/.helmignore b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/.helmignore new file mode 100644 index 0000000000..69790771c9 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/.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 +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ +linter_values.yaml +ci/ +installation/ diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Chart.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Chart.yaml new file mode 100644 index 0000000000..ff28ae8dab --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Chart.yaml @@ -0,0 +1,15 @@ +apiVersion: v2 +appVersion: 0.0.1 +description: 'Helm chart for deploying the http-header-injector sidecar, which automatically + injects x-request-id into http traffic going through the cluster for pods which + have the annotation `http-header-injector.stackstate.io/inject: enabled` is set. ' +home: https://github.com/StackVista/http-header-injector +icon: https://www.stackstate.com/wp-content/uploads/2019/02/152x152-favicon.png +keywords: +- monitoring +- stackstate +maintainers: +- email: ops@stackstate.com + name: Stackstate Lupulus Team +name: http-header-injector +version: 0.0.11 diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/README.md b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/README.md new file mode 100644 index 0000000000..840ff52404 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/README.md @@ -0,0 +1,56 @@ +# http-header-injector + +![Version: 0.0.11](https://img.shields.io/badge/Version-0.0.11-informational?style=flat-square) ![AppVersion: 0.0.1](https://img.shields.io/badge/AppVersion-0.0.1-informational?style=flat-square) + +Helm chart for deploying the http-header-injector sidecar, which automatically injects x-request-id into http traffic +going through the cluster for pods which have the annotation `http-header-injector.stackstate.io/inject: enabled` is set. + +**Homepage:** + +## Maintainers + +| Name | Email | Url | +| ---- | ------ | --- | +| Stackstate Lupulus Team | | | + +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| certificatePrehook | object | `{"image":{"pullPolicy":"IfNotPresent","registry":null,"repository":"stackstate/container-tools","tag":"1.4.0"},"resources":{"limits":{"cpu":"100m","memory":"200Mi"},"requests":{"cpu":"100m","memory":"200Mi"}}}` | Helm prehook to setup/remove a certificate for the sidecarInjector mutationwebhook | +| certificatePrehook.image.pullPolicy | string | `"IfNotPresent"` | Policy when pulling an image | +| certificatePrehook.image.registry | string | `nil` | Registry for the docker image. | +| certificatePrehook.image.tag | string | `"1.4.0"` | The tag for the docker image | +| debug | bool | `false` | Enable debugging. This will leave leave artifacts around like the prehook jobs for further inspection | +| enabled | bool | `true` | Enable/disable the mutationwebhook | +| global.extraAnnotations | object | `{}` | Extra annotations added ta all resources created by the helm chart | +| global.extraLabels | object | `{}` | Extra labels added ta all resources created by the helm chart | +| global.imagePullCredentials | object | `{}` | Globally define credentials for pulling images. | +| global.imagePullSecrets | list | `[]` | Globally add image pull secrets that are used. | +| global.imageRegistry | string | `nil` | Globally override the image registry that is used. Can be overridden by specific containers. Defaults to quay.io | +| images.pullSecretName | string | `nil` | | +| proxy | object | `{"image":{"pullPolicy":"IfNotPresent","registry":null,"repository":"stackstate/http-header-injector-proxy","tag":"sha-5ff79451"},"resources":{"limits":{"memory":"40Mi"},"requests":{"memory":"25Mi"}}}` | Proxy being injected into pods for rewriting http headers | +| proxy.image.pullPolicy | string | `"IfNotPresent"` | Policy when pulling an image | +| proxy.image.registry | string | `nil` | Registry for the docker image. | +| proxy.image.tag | string | `"sha-5ff79451"` | The tag for the docker image | +| proxy.resources.limits.memory | string | `"40Mi"` | Memory resource limits. | +| proxy.resources.requests.memory | string | `"25Mi"` | Memory resource requests. | +| proxyInit | object | `{"image":{"pullPolicy":"IfNotPresent","registry":null,"repository":"stackstate/http-header-injector-proxy-init","tag":"sha-5ff79451"}}` | InitContainer within pod which redirects traffic to the proxy container. | +| proxyInit.image.pullPolicy | string | `"IfNotPresent"` | Policy when pulling an image | +| proxyInit.image.registry | string | `nil` | Registry for the docker image | +| proxyInit.image.tag | string | `"sha-5ff79451"` | The tag for the docker image | +| sidecarInjector | object | `{"image":{"pullPolicy":"IfNotPresent","registry":null,"repository":"stackstate/generic-sidecar-injector","tag":"sha-9c852245"}}` | Service for injecting the proxy sidecar into pods | +| sidecarInjector.image.pullPolicy | string | `"IfNotPresent"` | Policy when pulling an image | +| sidecarInjector.image.registry | string | `nil` | Registry for the docker image. | +| sidecarInjector.image.tag | string | `"sha-9c852245"` | The tag for the docker image | +| webhook | object | `{"failurePolicy":"Ignore","tls":{"certManager":{"issuer":"","issuerKind":"ClusterIssuer","issuerNamespace":""},"mode":"generated","provided":{"caBundle":"","crt":"","key":""},"secret":{"name":""}}}` | MutationWebhook that will be installed to inject a sidecar into pods | +| webhook.failurePolicy | string | `"Ignore"` | How should the webhook fail? Best is to use Ignore, because there is a brief moment at initialization when the hook s there but the service not. Also, putting this to fail can cause the control plane be unresponsive. | +| webhook.tls.certManager.issuer | string | `""` | The issuer that is used for the webhook. Only used if you set webhook.tls.mode to "cert-manager". | +| webhook.tls.certManager.issuerKind | string | `"ClusterIssuer"` | The issuer kind that is used for the webhook, valid values are "Issuer" or "ClusterIssuer". Only used if you set webhook.tls.mode to "cert-manager". | +| webhook.tls.certManager.issuerNamespace | string | `""` | The namespace the cert-manager issuer is located in. If left empty defaults to the release's namespace that is used for the webhook. Only used if you set webhook.tls.mode to "cert-manager". | +| webhook.tls.mode | string | `"generated"` | The mode for the webhook. Can be "provided", "generated", "secret" or "cert-manager". If you want to use cert-manager, you need to install it first. NOTE: If you choose "generated", additional privileges are required to create the certificate and webhook at runtime. | +| webhook.tls.provided.caBundle | string | `""` | The caBundle that is used for the webhook. This is the certificate that is used to sign the webhook. Only used if you set webhook.tls.mode to "provided". | +| webhook.tls.provided.crt | string | `""` | The certificate that is used for the webhook. Only used if you set webhook.tls.mode to "provided". | +| webhook.tls.provided.key | string | `""` | The key that is used for the webhook. Only used if you set webhook.tls.mode to "provided". | +| webhook.tls.secret.name | string | `""` | The name of the secret containing the pre-provisioned certificate data that is used for the webhook. Only used if you set webhook.tls.mode to "secret". | + diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Readme.md.gotpl b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Readme.md.gotpl new file mode 100644 index 0000000000..225032aa2a --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/Readme.md.gotpl @@ -0,0 +1,26 @@ +{{ template "chart.header" . }} +{{ template "chart.description" . }} + +Current chart version is `{{ template "chart.version" . }}` + +{{ template "chart.homepageLine" . }} + +{{ template "chart.requirementsSection" . }} + +## Required Values + +No values have to be included to install this chart. After installing this chart, it becomes possible to annotate pods with +the `http-header-injector.stackstate.io/inject: enabled` annotation to make sure the sidecar provided by this chart is +activated on a pod. + +## Recommended Values + +{{ template "chart.valuesSection" . -}} + +## Install + +Install from the command line on Helm with the following command: + +```shell +helm install stackstate/http-header-injector +``` diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/_defines.tpl b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/_defines.tpl new file mode 100644 index 0000000000..ee6b7320ef --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/_defines.tpl @@ -0,0 +1,131 @@ +{{- define "http-header-injector.app.name" -}} +{{ .Release.Name }}-http-header-injector +{{- end -}} + +{{- define "http-header-injector.webhook-service.name" -}} +{{ .Release.Name }}-http-header-injector +{{- end -}} + +{{- define "http-header-injector.webhook-service.fqname" -}} +{{ .Release.Name }}-http-header-injector.{{ .Release.Namespace }}.svc +{{- end -}} + +{{- define "http-header-injector.cert-secret.name" -}} +{{- if eq .Values.webhook.tls.mode "secret" -}} +{{ .Values.webhook.tls.secret.name }} +{{- else -}} +{{ .Release.Name }}-http-injector-cert +{{- end -}} +{{- end -}} + +{{- define "http-header-injector.cert-clusterrole.name" -}} +{{ .Release.Name }}-http-injector-cert-cluster-role +{{- end -}} + +{{- define "http-header-injector.cert-serviceaccount.name" -}} +{{ .Release.Name }}-http-injector-cert-sa +{{- end -}} + +{{- define "http-header-injector.cert-config.name" -}} +{{ .Release.Name }}-cert-config +{{- end -}} + +{{- define "http-header-injector.mutatingwebhookconfiguration.name" -}} +{{ .Release.Name }}-http-header-injector-webhook.stackstate.io +{{- end -}} + +{{- define "http-header-injector.webhook-config.name" -}} +{{ .Release.Name }}-http-header-injector-config +{{- end -}} + +{{- define "http-header-injector.mutating-webhook.name" -}} +{{ .Release.Name }}-http-header-injector-webhook +{{- end -}} + +{{- define "http-header-injector.pull-secret.name" -}} +{{ include "http-header-injector.app.name" . }}-pull-secret +{{- end -}} + +{{/* If the issuer is located in a different namespace, it is possible to set that, else default to the release namespace */}} +{{- define "cert-manager.certificate.namespace" -}} +{{ .Values.webhook.tls.certManager.issuerNamespace | default .Release.Namespace }} +{{- end -}} + +{{- define "http-header-injector.image.registry.global" -}} + {{- if .Values.global }} + {{- .Values.global.imageRegistry | default "quay.io" -}} + {{- else -}} + quay.io + {{- end -}} +{{- end -}} + +{{- define "http-header-injector.image.registry" -}} + {{- if ((.ContainerConfig).image).registry -}} + {{- tpl .ContainerConfig.image.registry . -}} + {{- else -}} + {{- include "http-header-injector.image.registry.global" . }} + {{- end -}} +{{- end -}} + +{{- define "http-header-injector.image.pullSecrets" -}} + {{- $pullSecrets := list }} + {{- $pullSecrets = append $pullSecrets (include "http-header-injector.pull-secret.name" .) }} + {{- range .Values.global.imagePullSecrets -}} + {{- $pullSecrets = append $pullSecrets . -}} + {{- end -}} + {{- if (not (empty $pullSecrets)) -}} +imagePullSecrets: + {{- range $pullSecrets | uniq }} + - name: {{ . }} + {{- end }} + {{- end -}} +{{- end -}} + +{{- define "http-header-injector.cert-setup.container.main" }} +{{- $containerConfig := dict "ContainerConfig" .Values.certificatePrehook -}} +name: webhook-cert-setup +image: "{{ include "http-header-injector.image.registry" (merge $containerConfig .) }}/{{ .Values.certificatePrehook.image.repository }}:{{ .Values.certificatePrehook.image.tag }}" +imagePullPolicy: {{ .Values.certificatePrehook.image.pullPolicy }} +{{- with .Values.certificatePrehook.resources }} +resources: + {{- toYaml . | nindent 2 }} +{{- end }} +volumeMounts: + - name: "{{ include "http-header-injector.cert-config.name" . }}" + mountPath: /scripts + readOnly: true +command: ["/scripts/generate-cert.sh"] +{{- end }} + +{{- define "http-header-injector.cert-delete.container.main" }} +{{- $containerConfig := dict "ContainerConfig" .Values.certificatePrehook -}} +name: webhook-cert-delete +image: "{{ include "http-header-injector.image.registry" (merge $containerConfig .) }}/{{ .Values.certificatePrehook.image.repository }}:{{ .Values.certificatePrehook.image.tag }}" +imagePullPolicy: {{ .Values.certificatePrehook.image.pullPolicy }} +{{- with .Values.certificatePrehook.resources }} +resources: + {{- toYaml . | nindent 2 }} +{{- end }} +volumeMounts: + - name: "{{ include "http-header-injector.cert-config.name" . }}" + mountPath: /scripts +command: [ "/scripts/delete-cert.sh" ] +{{- end }} + +{{/* +Returns a YAML with extra annotations. +*/}} +{{- define "http-header-injector.global.extraAnnotations" -}} +{{- with .Values.global.extraAnnotations }} +{{- toYaml . }} +{{- end }} +{{- end -}} + +{{/* +Returns a YAML with extra labels. +*/}} +{{- define "http-header-injector.global.extraLabels" -}} +{{- with .Values.global.extraLabels }} +{{- toYaml . }} +{{- end }} +{{- end -}} \ No newline at end of file diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrolbinding.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrolbinding.yaml new file mode 100644 index 0000000000..fc0c012585 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrolbinding.yaml @@ -0,0 +1,24 @@ +{{- if eq .Values.webhook.tls.mode "generated" }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: "{{ include "http-header-injector.cert-serviceaccount.name" . }}" + labels: + app.kubernetes.io/component: http-header-injector-cert-hook + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-delete,post-upgrade + "helm.sh/hook-weight": "-3" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: "{{ include "http-header-injector.cert-clusterrole.name" . }}" +subjects: + - kind: ServiceAccount + name: "{{ include "http-header-injector.cert-serviceaccount.name" . }}" + namespace: {{ .Release.Namespace }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrole.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrole.yaml new file mode 100644 index 0000000000..afab838b3c --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-clusterrole.yaml @@ -0,0 +1,26 @@ +{{- if eq .Values.webhook.tls.mode "generated" }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: "{{ include "http-header-injector.cert-clusterrole.name" . }}" + labels: + app.kubernetes.io/component: http-header-injector-cert-hook + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-delete,post-upgrade + "helm.sh/hook-weight": "-4" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +rules: + - apiGroups: [ "admissionregistration.k8s.io" ] + resources: [ "mutatingwebhookconfigurations" ] + verbs: [ "get", "create", "patch","update","delete" ] + - apiGroups: [ "" ] + resources: [ "secrets" ] + verbs: [ "create", "get", "patch","update","delete" ] + - apiGroups: [ "apps" ] + resources: [ "deployments" ] + verbs: [ "get" ] +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-config.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-config.yaml new file mode 100644 index 0000000000..a22bdf4fb9 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-config.yaml @@ -0,0 +1,158 @@ +{{- if eq .Values.webhook.tls.mode "generated" }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: "{{ include "http-header-injector.cert-config.name" . }}" + labels: + app.kubernetes.io/component: http-header-injector-cert-hook + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-delete,post-upgrade + "helm.sh/hook-weight": "-3" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +data: + generate-cert.sh: | + #!/bin/bash + + # We are going for a self-signed certificate here. We would like to use k8s CertificateSigningRequest, however, + # currently there are no out of the box signers that can sign a 'server auth' certificate, which is required for mutation webhooks. + set -ex + + SCRIPTDIR="${BASH_SOURCE%/*}" + + DIR=`mktemp -d` + + cd "$DIR" + + {{ if .Values.enabled }} + echo "Chart enabled, creating secret and webhook" + + openssl genrsa -out ca.key 2048 + + openssl req -x509 -new -nodes -key ca.key -subj "/CN={{ include "http-header-injector.webhook-service.fqname" . }}" -days 10000 -out ca.crt + + openssl genrsa -out tls.key 2048 + + openssl req -new -key tls.key -out tls.csr -config "$SCRIPTDIR/csr.conf" + + openssl x509 -req -in tls.csr -CA ca.crt -CAkey ca.key \ + -CAcreateserial -out tls.crt -days 10000 \ + -extensions v3_ext -extfile "$SCRIPTDIR/csr.conf" -sha256 + + # Create or update the secret + echo "Applying secret" + kubectl create secret tls "{{ include "http-header-injector.cert-secret.name" . }}" \ + -n "{{ .Release.Namespace }}" \ + --cert=./tls.crt \ + --key=./tls.key \ + --dry-run=client \ + -o yaml | kubectl apply -f - + + echo "Applying mutationwebhook" + caBundle=`base64 -w 0 ca.crt` + cat "$SCRIPTDIR/mutatingwebhookconfiguration.yaml" | sed "s/\\\$CA_BUNDLE/$caBundle/g" | kubectl apply -f - + {{ else }} + echo "Chart disabled, not creating secret and webhook" + {{ end }} + delete-cert.sh: | + #!/bin/bash + + set -x + + DIR="${BASH_SOURCE%/*}" + if [[ ! -d "$DIR" ]]; then DIR="$PWD"; fi + if [[ "$DIR" = "." ]]; then DIR="$PWD"; fi + + cd "$DIR" + + # Using detection of deployment hee to also make this work in post-delete. + if kubectl get deployments "{{ include "http-header-injector.app.name" . }}" -n "{{ .Release.Namespace }}"; then + echo "Chart enabled, not removing secret and mutationwebhook" + exit 0 + else + echo "Chart disabled, removing secret and mutationwebhook" + fi + + # Create or update the secret + echo "Deleting secret" + kubectl delete secret "{{ include "http-header-injector.cert-secret.name" . }}" -n "{{ .Release.Namespace }}" + + echo "Applying mutationwebhook" + kubectl delete MutatingWebhookConfiguration "{{ include "http-header-injector.mutating-webhook.name" . }}" -n "{{ .Release.Namespace }}" + + exit 0 + + csr.conf: | + [ req ] + default_bits = 2048 + prompt = no + default_md = sha256 + req_extensions = req_ext + distinguished_name = dn + + [ dn ] + C = NL + ST = Utrecht + L = Hilversum + O = StackState + OU = Dev + CN = {{ include "http-header-injector.webhook-service.fqname" . }} + + [ req_ext ] + subjectAltName = @alt_names + + [ alt_names ] + DNS.1 = {{ include "http-header-injector.webhook-service.fqname" . }} + + [ v3_ext ] + authorityKeyIdentifier=keyid,issuer:always + basicConstraints=CA:FALSE + keyUsage=keyEncipherment,dataEncipherment + extendedKeyUsage=serverAuth + subjectAltName=@alt_names + + mutatingwebhookconfiguration.yaml: | + apiVersion: admissionregistration.k8s.io/v1 + kind: MutatingWebhookConfiguration + metadata: + name: "{{ include "http-header-injector.mutating-webhook.name" . }}" + namespace: "{{ .Release.Namespace }}" + labels: +{{ include "http-header-injector.global.extraLabels" . | indent 8 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 8 }} + webhooks: + - clientConfig: + caBundle: "$CA_BUNDLE" + service: + name: "{{ include "http-header-injector.webhook-service.name" . }}" + path: /mutate + namespace: {{ .Release.Namespace }} + port: 8443 + # Putting failure on ignore, not doing so can crash the entire control plane if something goes wrong with the service. + failurePolicy: "{{ .Values.webhook.failurePolicy }}" + name: "{{ include "http-header-injector.mutatingwebhookconfiguration.name" . }}" + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: + - kube-system + - cert-manager + - {{ .Release.Namespace }} + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + resources: + - pods + sideEffects: None + admissionReviewVersions: + - v1 +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-delete.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-delete.yaml new file mode 100644 index 0000000000..6f72ce2478 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-delete.yaml @@ -0,0 +1,39 @@ +{{- if eq .Values.webhook.tls.mode "generated" }} +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ .Release.Name }}-header-injector-cert-delete + labels: + app.kubernetes.io/component: http-header-injector-cert-hook-delete + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: + "helm.sh/hook": post-delete,post-upgrade + "helm.sh/hook-weight": "-2" + "helm.sh/hook-delete-policy": before-hook-creation{{- if not .Values.debug -}},hook-succeeded{{- end }} +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +spec: + template: + metadata: + labels: + app.kubernetes.io/component: http-header-injector-delete + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 8 }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/cert-hook-config.yaml") . | sha256sum }} +{{ include "http-header-injector.global.extraAnnotations" . | indent 8 }} + spec: + serviceAccountName: "{{ include "http-header-injector.cert-serviceaccount.name" . }}" + {{- include "http-header-injector.image.pullSecrets" . | nindent 6 }} + volumes: + - name: "{{ include "http-header-injector.cert-config.name" . }}" + configMap: + name: "{{ include "http-header-injector.cert-config.name" . }}" + defaultMode: 0777 + containers: + - {{ include "http-header-injector.cert-delete.container.main" . | nindent 8 }} + restartPolicy: Never + backoffLimit: 0 +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-setup.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-setup.yaml new file mode 100644 index 0000000000..cc1c89631f --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-job-setup.yaml @@ -0,0 +1,39 @@ +{{- if eq .Values.webhook.tls.mode "generated" }} +apiVersion: batch/v1 +kind: Job +metadata: + name: {{ .Release.Name }}-header-injector-cert-setup + labels: + app.kubernetes.io/component: http-header-injector-cert-hook-setup + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade + "helm.sh/hook-weight": "-2" + "helm.sh/hook-delete-policy": before-hook-creation{{- if not .Values.debug -}},hook-succeeded{{- end }} +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +spec: + template: + metadata: + labels: + app.kubernetes.io/component: http-header-injector-setup + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 8 }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/cert-hook-config.yaml") . | sha256sum }} +{{ include "http-header-injector.global.extraAnnotations" . | indent 8 }} + spec: + serviceAccountName: "{{ include "http-header-injector.cert-serviceaccount.name" . }}" + {{- include "http-header-injector.image.pullSecrets" . | nindent 6 }} + volumes: + - name: "{{ include "http-header-injector.cert-config.name" . }}" + configMap: + name: "{{ include "http-header-injector.cert-config.name" . }}" + defaultMode: 0777 + containers: + - {{ include "http-header-injector.cert-setup.container.main" . | nindent 8 }} + restartPolicy: Never + backoffLimit: 0 +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-serviceaccount.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-serviceaccount.yaml new file mode 100644 index 0000000000..29b26df95b --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/cert-hook-serviceaccount.yaml @@ -0,0 +1,18 @@ +{{- if eq .Values.webhook.tls.mode "generated" }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: "{{ include "http-header-injector.cert-serviceaccount.name" . }}" + namespace: {{ .Release.Namespace }} + annotations: + "helm.sh/hook": pre-install,pre-upgrade,post-delete,post-upgrade + "helm.sh/hook-weight": "-4" + "helm.sh/hook-delete-policy": before-hook-creation,hook-succeeded +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} + labels: + app.kubernetes.io/component: http-header-injector-cert-hook + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} + app: "{{ include "http-header-injector.app.name" . }}" +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/pull-secret.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/pull-secret.yaml new file mode 100644 index 0000000000..80b4ee404b --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/pull-secret.yaml @@ -0,0 +1,32 @@ +{{- $defaultRegistry := .Values.global.imageRegistry }} +{{- $top := . }} +{{- $registryAuthMap := dict }} + +{{- range $registry, $credentials := .Values.global.imagePullCredentials }} + {{- $registryAuthDocument := dict -}} + {{- $_ := set $registryAuthDocument "username" $credentials.username }} + {{- $_ := set $registryAuthDocument "password" $credentials.password }} + {{- $authMessage := printf "%s:%s" $registryAuthDocument.username $registryAuthDocument.password | b64enc }} + {{- $_ := set $registryAuthDocument "auth" $authMessage }} + {{- if eq $registry "default" }} + {{- $registryAuthMap := set $registryAuthMap (include "http-header-injector.image.registry.global" $top) $registryAuthDocument }} + {{ else }} + {{- $registryAuthMap := set $registryAuthMap $registry $registryAuthDocument }} + {{- end }} +{{- end }} +{{- $dockerAuthsDocuments := dict "auths" $registryAuthMap }} + +apiVersion: v1 +kind: Secret +metadata: + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} + name: {{ include "http-header-injector.pull-secret.name" . }} +data: + .dockerconfigjson: {{ $dockerAuthsDocuments | toJson | b64enc | quote }} +type: kubernetes.io/dockerconfigjson \ No newline at end of file diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-cert-secret.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-cert-secret.yaml new file mode 100644 index 0000000000..f571ca86bb --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-cert-secret.yaml @@ -0,0 +1,18 @@ +{{- if eq .Values.webhook.tls.mode "provided" }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "http-header-injector.cert-secret.name" . }} + namespace: {{ .Release.Namespace }} + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +type: kubernetes.io/tls +data: + tls.crt: {{ .Values.webhook.tls.provided.crt | b64enc }} + tls.key: {{ .Values.webhook.tls.provided.key | b64enc }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-certificate.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-certificate.yaml new file mode 100644 index 0000000000..a68c7c5f62 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-certificate.yaml @@ -0,0 +1,23 @@ +{{- if eq .Values.webhook.tls.mode "cert-manager" }} +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + name: {{ include "http-header-injector.webhook-service.name" . }} + namespace: {{ include "cert-manager.certificate.namespace" . }} + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +spec: + secretName: {{ include "http-header-injector.cert-secret.name" . }} + issuerRef: + name: {{ .Values.webhook.tls.certManager.issuer }} + kind: {{ .Values.webhook.tls.certManager.issuerKind }} + dnsNames: + - "{{ include "http-header-injector.webhook-service.name" . }}" + - "{{ include "http-header-injector.webhook-service.name" . }}.{{ .Release.Namespace }}" + - "{{ include "http-header-injector.webhook-service.name" . }}.{{ .Release.Namespace }}.svc" +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-config.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-config.yaml new file mode 100644 index 0000000000..20b38ce963 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-config.yaml @@ -0,0 +1,128 @@ +{{- if .Values.enabled -}} +{{- $proxyContainerConfig := dict "ContainerConfig" .Values.proxy -}} +{{- $proxyInitContainerConfig := dict "ContainerConfig" .Values.proxyInit -}} +apiVersion: v1 +kind: ConfigMap +metadata: + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} + name: {{ .Release.Name }}-http-header-injector-config +data: + sidecarconfig.yaml: | + initContainers: + - name: http-header-proxy-init + image: "{{ include "http-header-injector.image.registry" (merge $proxyInitContainerConfig .) }}/{{ .Values.proxyInit.image.repository }}:{{ .Values.proxyInit.image.tag }}" + imagePullPolicy: {{ .Values.proxyInit.image.pullPolicy }} + command: ["/init-iptables.sh"] + env: + - name: CHART_VERSION + value: "{{ .Chart.Version }}" + - name: PROXY_PORT + value: {% if index .Annotations "config.http-header-injector.stackstate.io/proxy-port" %}"{% index .Annotations "config.http-header-injector.stackstate.io/proxy-port" %}"{% else %}"7060"{% end %} + - name: PROXY_UID + value: {% if index .Annotations "config.http-header-injector.stackstate.io/proxy-uid" %}"{% index .Annotations "config.http-header-injector.stackstate.io/proxy-uid" %}"{% else %}"2103"{% end %} + - name: POD_HOST_NETWORK + value: {% .Spec.HostNetwork %} + {% if eq (index .Annotations "linkerd.io/inject") "enabled" %} + - name: LINKERD + value: true + # Reference: https://linkerd.io/2.13/reference/proxy-configuration/ + - name: LINKERD_PROXY_UID + value: {% if index .Annotations "config.linkerd.io/proxy-uid" %}"{% index .Annotations "config.linkerd.io/proxy-uid" %}"{% else %}"2102"{% end %} + # Due to https://github.com/linkerd/linkerd2/issues/10981 this is now not realy possible, still bringing in the code for future reference + - name: LINKERD_ADMIN_PORT + value: {% if index .Annotations "config.linkerd.io/admin-port" %}"{% index .Annotations "config.linkerd.io/admin-port" %}"{% else %}"4191"{% end %} + {% end %} + securityContext: + allowPrivilegeEscalation: false + capabilities: + add: + - NET_ADMIN + - NET_RAW + privileged: false + readOnlyRootFilesystem: true + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + # This is required for iptables to be able to run + - mountPath: /run + name: http-header-proxy-init-xtables-lock + + containers: + - name: http-header-proxy + image: "{{ include "http-header-injector.image.registry" (merge $proxyContainerConfig .) }}/{{ .Values.proxy.image.repository }}:{{ .Values.proxy.image.tag }}" + imagePullPolicy: {{ .Values.proxy.image.pullPolicy }} + env: + - name: CHART_VERSION + value: "{{ .Chart.Version }}" + - name: PORT + value: {% if index .Annotations "config.http-header-injector.stackstate.io/proxy-port" %}"{% index .Annotations "config.http-header-injector.stackstate.io/proxy-port" %}"{% else %}"7060"{% end %} + - name: DEBUG + value: {% if index .Annotations "config.http-header-injector.stackstate.io/debug" %}"{% index .Annotations "config.http-header-injector.stackstate.io/debug" %}"{% else %}"disabled"{% end %} + securityContext: + runAsUser: {% if index .Annotations "config.http-header-injector.stackstate.io/proxy-uid" %}{% index .Annotations "config.http-header-injector.stackstate.io/proxy-uid" %}{% else %}2103{% end %} + seccompProfile: + type: RuntimeDefault + {{- with .Values.proxy.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + - name: http-header-inject-debug + image: "{{ include "http-header-injector.image.registry" (merge $proxyContainerConfig .) }}/{{ .Values.proxyInit.image.repository }}:{{ .Values.proxyInit.image.tag }}" + imagePullPolicy: {{ .Values.proxyInit.image.pullPolicy }} + command: ["/bin/sh", "-c", "while echo \"Running\"; do sleep 1; done"] + securityContext: + allowPrivilegeEscalation: false + capabilities: + add: + - NET_ADMIN + - NET_RAW + privileged: false + readOnlyRootFilesystem: true + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + # This is required for iptables to be able to run + - mountPath: /run + name: http-header-proxy-init-xtables-lock + + volumes: + - emptyDir: {} + name: http-header-proxy-init-xtables-lock + + mutationconfig.yaml: | + mutationConfigs: + - name: "http-header-injector" + annotationNamespace: "http-header-injector.stackstate.io" + annotationTrigger: "inject" + annotationConfig: + volumeMounts: [] + initContainersBeforePodInitContainers: [ "http-header-proxy-init" ] + initContainers: [ "http-header-proxy-init" ] + containers: [ "http-header-proxy" ] + volumes: [ "http-header-proxy-init-xtables-lock" ] + volumeMounts: [ ] + # Namespaces are ignored by the mutatingwebhook + ignoreNamespaces: [ ] + - name: "http-header-injector-debug" + annotationNamespace: "http-header-injector-debug.stackstate.io" + annotationTrigger: "inject" + annotationConfig: + volumeMounts: [] + initContainersBeforePodInitContainers: [ ] + initContainers: [ ] + containers: [ "http-header-inject-debug" ] + volumes: [ "http-header-proxy-init-xtables-lock" ] + volumeMounts: [ ] + # Namespaces are ignored by the mutatingwebhook + ignoreNamespaces: [ ] + {{- end -}} \ No newline at end of file diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-deployment.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-deployment.yaml new file mode 100644 index 0000000000..8af6ff51a2 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-deployment.yaml @@ -0,0 +1,61 @@ +{{- if .Values.enabled -}} +{{- $containerConfig := dict "ContainerConfig" .Values.sidecarInjector -}} +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} + app: "{{ include "http-header-injector.app.name" . }}" +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} + name: "{{ include "http-header-injector.app.name" . }}" +spec: + replicas: 1 + selector: + matchLabels: + app: "{{ include "http-header-injector.app.name" . }}" + template: + metadata: + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} + app: "{{ include "http-header-injector.app.name" . }}" +{{ include "http-header-injector.global.extraLabels" . | indent 8 }} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/webhook-config.yaml") . | sha256sum }} + # This is here to make sure the generic injector gets restarted and picks up a new secret that may have been generated upon upgrade. + revision: "{{ .Release.Revision }}" +{{ include "http-header-injector.global.extraAnnotations" . | indent 8 }} + name: "{{ include "http-header-injector.app.name" . }}" + spec: + {{- include "http-header-injector.image.pullSecrets" . | nindent 6 }} + volumes: + - name: "{{ include "http-header-injector.webhook-config.name" . }}" + configMap: + name: "{{ include "http-header-injector.webhook-config.name" . }}" + - name: "{{ include "http-header-injector.cert-secret.name" . }}" + secret: + secretName: "{{ include "http-header-injector.cert-secret.name" . }}" + containers: + - image: "{{ include "http-header-injector.image.registry" (merge $containerConfig .) }}/{{ .Values.sidecarInjector.image.repository }}:{{ .Values.sidecarInjector.image.tag }}" + imagePullPolicy: {{ .Values.sidecarInjector.image.pullPolicy }} + name: http-header-injector + volumeMounts: + - name: "{{ include "http-header-injector.webhook-config.name" . }}" + mountPath: /etc/webhook/config + readOnly: true + - name: "{{ include "http-header-injector.cert-secret.name" . }}" + mountPath: /etc/webhook/certs + readOnly: true + command: [ "/sidecarinjector" ] + args: + - --port=8443 + - --sidecar-config-file=/etc/webhook/config/sidecarconfig.yaml + - --mutation-config-file=/etc/webhook/config/mutationconfig.yaml + - --cert-file-path=/etc/webhook/certs/tls.crt + - --key-file-path=/etc/webhook/certs/tls.key +{{- end -}} \ No newline at end of file diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-mutatingwebhookconfiguration.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-mutatingwebhookconfiguration.yaml new file mode 100644 index 0000000000..de0acc1dff --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-mutatingwebhookconfiguration.yaml @@ -0,0 +1,54 @@ +{{- if not (eq .Values.webhook.tls.mode "generated") }} +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: "{{ include "http-header-injector.mutating-webhook.name" . }}" + namespace: "{{ .Release.Namespace }}" + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: + {{- if eq .Values.webhook.tls.mode "cert-manager" }} + cert-manager.io/inject-ca-from: {{ include "cert-manager.certificate.namespace" . }}/{{ include "http-header-injector.webhook-service.name" . }} + {{- else if eq .Values.webhook.tls.mode "secret" }} + cert-manager.io/inject-ca-from-secret: {{ .Release.Namespace }}/{{ .Values.webhook.tls.secret.name | required "'webhook.tls.secret.name' is required when webhook.tls.mode is 'secret'" }} + {{- end }} +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} +webhooks: + - clientConfig: + {{- if eq .Values.webhook.tls.mode "provided" }} + caBundle: "{{ .Values.webhook.tls.provided.caBundle | b64enc }}" + {{- else if or (eq .Values.webhook.tls.mode "cert-manager") (eq .Values.webhook.tls.mode "secret") }} + caBundle: "" + {{- end }} + service: + name: "{{ include "http-header-injector.webhook-service.name" . }}" + path: /mutate + namespace: {{ .Release.Namespace }} + port: 8443 + # Putting failure on ignore, not doing so can crash the entire control plane if something goes wrong with the service. + failurePolicy: "{{ .Values.webhook.failurePolicy }}" + name: "{{ include "http-header-injector.mutatingwebhookconfiguration.name" . }}" + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: NotIn + values: + - kube-system + - cert-manager + - {{ .Release.Namespace }} + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + resources: + - pods + sideEffects: None + admissionReviewVersions: + - v1 +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-service.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-service.yaml new file mode 100644 index 0000000000..55abdb022c --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/templates/webhook-service.yaml @@ -0,0 +1,20 @@ +{{- if .Values.enabled -}} +apiVersion: v1 +kind: Service +metadata: + labels: + app.kubernetes.io/component: http-header-injector + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "http-header-injector.app.name" . }} +{{ include "http-header-injector.global.extraLabels" . | indent 4 }} + annotations: +{{ include "http-header-injector.global.extraAnnotations" . | indent 4 }} + name: "{{ include "http-header-injector.webhook-service.name" . }}" +spec: + ports: + - port: 8443 + protocol: TCP + targetPort: 8443 + selector: + app: "{{ include "http-header-injector.app.name" . }}" +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/values.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/values.yaml new file mode 100644 index 0000000000..a1b4be2fcc --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/charts/http-header-injector/values.yaml @@ -0,0 +1,110 @@ +# enabled -- Enable/disable the mutationwebhook +enabled: true + +# debug -- Enable debugging. This will leave leave artifacts around like the prehook jobs for further inspection +debug: false + +global: + # global.imageRegistry -- Globally override the image registry that is used. Can be overridden by specific containers. Defaults to quay.io + imageRegistry: null + # global.imagePullSecrets -- Globally add image pull secrets that are used. + imagePullSecrets: [] + # global.imagePullCredentials -- Globally define credentials for pulling images. + imagePullCredentials: {} + + # global.extraLabels -- Extra labels added ta all resources created by the helm chart + extraLabels: {} + # global.extraAnnotations -- Extra annotations added ta all resources created by the helm chart + extraAnnotations: {} + +images: + pullSecretName: + +# proxy -- Proxy being injected into pods for rewriting http headers +proxy: + image: + # proxy.image.registry -- Registry for the docker image. + registry: + # proxy.image.repository - Repository for the docker image + repository: "stackstate/http-header-injector-proxy" + # proxy.image.pullPolicy -- Policy when pulling an image + pullPolicy: IfNotPresent + # proxy.image.tag -- The tag for the docker image + tag: sha-5ff79451 + + # proxy.resource -- Resources for the proxy container + resources: + requests: + # proxy.resources.requests.memory -- Memory resource requests. + memory: "25Mi" + limits: + # proxy.resources.limits.memory -- Memory resource limits. + memory: "40Mi" + +# proxyInit -- InitContainer within pod which redirects traffic to the proxy container. +proxyInit: + image: + # proxyInit.image.registry -- Registry for the docker image + registry: + # proxyInit.image.repository - Repository for the docker image + repository: "stackstate/http-header-injector-proxy-init" + # proxyInit.image.pullPolicy -- Policy when pulling an image + pullPolicy: IfNotPresent + # proxyInit.image.tag -- The tag for the docker image + tag: sha-5ff79451 + +# sidecarInjector -- Service for injecting the proxy sidecar into pods +sidecarInjector: + image: + # sidecarInjector.image.registry -- Registry for the docker image. + registry: + # sidecarInjector.image.repository - Repository for the docker image + repository: "stackstate/generic-sidecar-injector" + # sidecarInjector.image.pullPolicy -- Policy when pulling an image + pullPolicy: IfNotPresent + # sidecarInjector.image.tag -- The tag for the docker image + tag: sha-9c852245 + +# certificatePrehook -- Helm prehook to setup/remove a certificate for the sidecarInjector mutationwebhook +certificatePrehook: + image: + # certificatePrehook.image.registry -- Registry for the docker image. + registry: + # certificatePrehook.image.repository - Repository for the docker image. + repository: stackstate/container-tools + # certificatePrehook.image.pullPolicy -- Policy when pulling an image + pullPolicy: IfNotPresent + # certificatePrehook.image.tag -- The tag for the docker image + tag: 1.4.0 + resources: + limits: + cpu: "100m" + memory: "200Mi" + requests: + cpu: "100m" + memory: "200Mi" + +# webhook -- MutationWebhook that will be installed to inject a sidecar into pods +webhook: + # webhook.failurePolicy -- How should the webhook fail? Best is to use Ignore, because there is a brief moment at initialization when the hook s there but the service not. Also, putting this to fail can cause the control plane be unresponsive. + failurePolicy: Ignore + tls: + # webhook.tls.mode -- The mode for the webhook. Can be "provided", "generated", "secret" or "cert-manager". If you want to use cert-manager, you need to install it first. NOTE: If you choose "generated", additional privileges are required to create the certificate and webhook at runtime. + mode: "generated" + provided: + # webhook.tls.provided.caBundle -- The caBundle that is used for the webhook. This is the certificate that is used to sign the webhook. Only used if you set webhook.tls.mode to "provided". + caBundle: "" + # webhook.tls.provided.crt -- The certificate that is used for the webhook. Only used if you set webhook.tls.mode to "provided". + crt: "" + # webhook.tls.provided.key -- The key that is used for the webhook. Only used if you set webhook.tls.mode to "provided". + key: "" + certManager: + # webhook.tls.certManager.issuer -- The issuer that is used for the webhook. Only used if you set webhook.tls.mode to "cert-manager". + issuer: "" + # webhook.tls.certManager.issuerKind -- The issuer kind that is used for the webhook, valid values are "Issuer" or "ClusterIssuer". Only used if you set webhook.tls.mode to "cert-manager". + issuerKind: "ClusterIssuer" + # webhook.tls.certManager.issuerNamespace -- The namespace the cert-manager issuer is located in. If left empty defaults to the release's namespace that is used for the webhook. Only used if you set webhook.tls.mode to "cert-manager". + issuerNamespace: "" + secret: + # webhook.tls.secret.name -- The name of the secret containing the pre-provisioned certificate data that is used for the webhook. Only used if you set webhook.tls.mode to "secret". + name: "" diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/questions.yml b/charts/stackstate/stackstate-k8s-agent/1.0.98/questions.yml new file mode 100644 index 0000000000..5d6e6a0112 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/questions.yml @@ -0,0 +1,184 @@ +questions: + - variable: stackstate.apiKey + label: "StackState API Key" + type: string + description: "The API key for StackState." + required: true + group: General + - variable: stackstate.url + label: "StackState URL" + type: string + description: "The URL where StackState is running." + required: true + group: General + - variable: stackstate.cluster.name + label: "StackState Cluster Name" + type: string + description: "The StackState Cluster Name given when installing the instance of the Kubernetes StackPack in StackState. This is used to identify the cluster in StackState." + required: true + group: General + - variable: all.registry.override + label: "Override Default Image Registry" + type: boolean + description: "Whether or not to override the default image registry." + default: false + group: "General" + show_subquestions_if: true + subquestions: + - variable: all.image.registry + label: "Docker Image Registry" + type: string + description: "The registry to pull the StackState Agent images from." + default: "quay.io" + - variable: global.imagePullCredentials.username + label: "Docker Image Pull Username" + type: string + description: "The username to use when pulling the StackState Agent images." + - variable: global.imagePullCredentials.password + label: "Docker Image Pull Password" + type: secret + description: "The password to use when pulling the StackState Agent images." + - variable: nodeAgent.containers.agent.resources.override + label: "Override Node Agent Resource Allocation" + type: boolean + description: "Whether or not to override the default resources." + default: "false" + group: "Node Agent" + show_subquestions_if: true + subquestions: + - variable: nodeAgent.containers.agent.resources.requests.cpu + label: "CPU Requests" + type: string + description: "The requested CPU for the Node Agent." + default: "20m" + - variable: nodeAgent.containers.agent.resources.requests.memory + label: "Memory Requests" + type: string + description: "The requested memory for the Node Agent." + default: "180Mi" + - variable: nodeAgent.containers.agent.resources.limits.cpu + label: "CPU Limit" + type: string + description: "The CPU limit for the Node Agent." + default: "270m" + - variable: nodeAgent.containers.agent.resources.limits.memory + label: "Memory Limit" + type: string + description: "The memory limit for the Node Agent." + default: "420Mi" + - variable: nodeAgent.containers.processAgent.enabled + label: "Enable Process Agent" + type: boolean + description: "Whether or not to enable the Process Agent." + default: "true" + group: "Process Agent" + - variable: nodeAgent.skipKubeletTLSVerify + label: "Skip Kubelet TLS Verify" + type: boolean + description: "Whether or not to skip TLS verification when connecting to the kubelet API." + default: "true" + group: "Process Agent" + - variable: nodeAgent.containers.processAgent.resources.override + label: "Override Process Agent Resource Allocation" + type: boolean + description: "Whether or not to override the default resources." + default: "false" + group: "Process Agent" + show_subquestions_if: true + subquestions: + - variable: nodeAgent.containers.processAgent.resources.requests.cpu + label: "CPU Requests" + type: string + description: "The requested CPU for the Process Agent." + default: "25m" + - variable: nodeAgent.containers.processAgent.resources.requests.memory + label: "Memory Requests" + type: string + description: "The requested memory for the Process Agent." + default: "128Mi" + - variable: nodeAgent.containers.processAgent.resources.limits.cpu + label: "CPU Limit" + type: string + description: "The CPU limit for the Process Agent." + default: "125m" + - variable: nodeAgent.containers.processAgent.resources.limits.memory + label: "Memory Limit" + type: string + description: "The memory limit for the Process Agent." + default: "400Mi" + - variable: clusterAgent.enabled + label: "Enable Cluster Agent" + type: boolean + description: "Whether or not to enable the Cluster Agent." + default: "true" + group: "Cluster Agent" + - variable: clusterAgent.collection.kubernetesResources.secrets + label: "Collect Secret Resources" + type: boolean + description: | + Whether or not to collect Kubernetes Secrets. + NOTE: StackState will not send the actual data of the secrets, only the metadata and a secure hash of the data. + default: "true" + group: "Cluster Agent" + - variable: clusterAgent.resources.override + label: "Override Cluster Agent Resource Allocation" + type: boolean + description: "Whether or not to override the default resources." + default: "false" + group: "Cluster Agent" + show_subquestions_if: true + subquestions: + - variable: clusterAgent.resources.requests.cpu + label: "CPU Requests" + type: string + description: "The requested CPU for the Cluster Agent." + default: "70m" + - variable: clusterAgent.resources.requests.memory + label: "Memory Requests" + type: string + description: "The requested memory for the Cluster Agent." + default: "512Mi" + - variable: clusterAgent.resources.limits.cpu + label: "CPU Limit" + type: string + description: "The CPU limit for the Cluster Agent." + default: "400m" + - variable: clusterAgent.resources.limits.memory + label: "Memory Limit" + type: string + description: "The memory limit for the Cluster Agent." + default: "800Mi" + - variable: logsAgent.enabled + label: "Enable Logs Agent" + type: boolean + description: "Whether or not to enable the Logs Agent." + default: "true" + group: "Logs Agent" + - variable: logsAgent.resources.override + label: "Override Logs Agent Resource Allocation" + type: boolean + description: "Whether or not to override the default resources." + default: "false" + group: "Logs Agent" + show_subquestions_if: true + subquestions: + - variable: logsAgent.resources.requests.cpu + label: "CPU Requests" + type: string + description: "The requested CPU for the Logs Agent." + default: "20m" + - variable: logsAgent.resources.requests.memory + label: "Memory Requests" + type: string + description: "The requested memory for the Logs Agent." + default: "100Mi" + - variable: logsAgent.resources.limits.cpu + label: "CPU Limit" + type: string + description: "The CPU limit for the Logs Agent." + default: "1300m" + - variable: logsAgent.resources.limits.memory + label: "Memory Limit" + type: string + description: "The memory limit for the Logs Agent." + default: "192Mi" diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_cluster-agent-kube-state-metrics.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_cluster-agent-kube-state-metrics.yaml new file mode 100644 index 0000000000..f99fbf6187 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_cluster-agent-kube-state-metrics.yaml @@ -0,0 +1,62 @@ +{{- define "cluster-agent-kube-state-metrics" -}} +{{- $kubeRes := .Values.clusterAgent.collection.kubernetesResources }} +{{- if .Values.clusterAgent.collection.kubeStateMetrics.clusterCheck }} +cluster_check: true +{{- end }} +init_config: +instances: + - collectors: + - nodes + - pods + - services + {{- if $kubeRes.persistentvolumeclaims }} + - persistentvolumeclaims + {{- end }} + {{- if $kubeRes.persistentvolumes }} + - persistentvolumes + {{- end }} + {{- if $kubeRes.namespaces }} + - namespaces + {{- end }} + {{- if $kubeRes.endpoints }} + - endpoints + {{- end }} + {{- if $kubeRes.daemonsets }} + - daemonsets + {{- end }} + {{- if $kubeRes.deployments }} + - deployments + {{- end }} + {{- if $kubeRes.replicasets }} + - replicasets + {{- end }} + {{- if $kubeRes.statefulsets }} + - statefulsets + {{- end }} + {{- if $kubeRes.cronjobs }} + - cronjobs + {{- end }} + {{- if $kubeRes.jobs }} + - jobs + {{- end }} + {{- if $kubeRes.ingresses }} + - ingresses + {{- end }} + {{- if $kubeRes.secrets }} + - secrets + {{- end }} + - resourcequotas + - replicationcontrollers + - limitranges + - horizontalpodautoscalers + - poddisruptionbudgets + - storageclasses + - volumeattachments + {{- if .Values.clusterAgent.collection.kubeStateMetrics.clusterCheck }} + skip_leader_election: true + {{- end }} + labels_as_tags: + {{ .Values.clusterAgent.collection.kubeStateMetrics.labelsAsTags | toYaml | indent 8 }} + annotations_as_tags: + {{ .Values.clusterAgent.collection.kubeStateMetrics.annotationsAsTags | toYaml | indent 8 }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-agent.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-agent.yaml new file mode 100644 index 0000000000..09f9591c63 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-agent.yaml @@ -0,0 +1,191 @@ +{{- define "container-agent" -}} +- name: node-agent +{{- if .Values.all.hardening.enabled}} + lifecycle: + preStop: + exec: + command: [ "/bin/sh", "-c", "echo 'Giving slim.ai monitor time to submit data...'; sleep 120" ] +{{- end }} + image: "{{ include "stackstate-k8s-agent.imageRegistry" . }}/{{ .Values.nodeAgent.containers.agent.image.repository }}:{{ .Values.nodeAgent.containers.agent.image.tag }}" + imagePullPolicy: "{{ .Values.nodeAgent.containers.agent.image.pullPolicy }}" + env: + {{ include "stackstate-k8s-agent.apiKeyEnv" . | nindent 4 }} + - name: STS_KUBERNETES_KUBELET_HOST + valueFrom: + fieldRef: + fieldPath: status.hostIP + - name: KUBERNETES_HOSTNAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: STS_HOSTNAME + value: "$(KUBERNETES_HOSTNAME)-{{ .Values.stackstate.cluster.name}}" + - name: AGENT_VERSION + value: {{ .Values.nodeAgent.containers.agent.image.tag | quote }} + - name: HOST_PROC + value: "/host/proc" + - name: HOST_SYS + value: "/host/sys" + - name: KUBERNETES + value: "true" + - name: STS_APM_ENABLED + value: {{ .Values.nodeAgent.apm.enabled | quote }} + - name: STS_APM_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + - name: STS_CLUSTER_AGENT_ENABLED + value: {{ .Values.clusterAgent.enabled | quote }} + {{- if .Values.clusterAgent.enabled }} + - name: STS_CLUSTER_AGENT_KUBERNETES_SERVICE_NAME + value: {{ .Release.Name }}-cluster-agent + {{ include "stackstate-k8s-agent.clusterAgentAuthTokenEnv" . | nindent 4 }} + {{- end }} + - name: STS_CLUSTER_NAME + value: {{ .Values.stackstate.cluster.name | quote }} + - name: STS_SKIP_VALIDATE_CLUSTERNAME + value: "true" + - name: STS_CHECKS_TAG_CARDINALITY + value: {{ .Values.nodeAgent.checksTagCardinality | quote }} + {{- if .Values.checksAgent.enabled }} + - name: STS_EXTRA_CONFIG_PROVIDERS + value: "endpointschecks" + {{- end }} + - name: STS_HEALTH_PORT + value: "5555" + - name: STS_LEADER_ELECTION + value: "false" + - name: LOG_LEVEL + value: {{ .Values.nodeAgent.containers.agent.logLevel | default .Values.nodeAgent.logLevel | quote }} + - name: STS_LOG_LEVEL + value: {{ .Values.nodeAgent.containers.agent.logLevel | default .Values.nodeAgent.logLevel | quote }} + - name: STS_NETWORK_TRACING_ENABLED + value: {{ .Values.nodeAgent.networkTracing.enabled | quote }} + - name: STS_PROTOCOL_INSPECTION_ENABLED + value: {{ .Values.nodeAgent.protocolInspection.enabled | quote }} + - name: STS_PROCESS_AGENT_ENABLED + value: {{ .Values.nodeAgent.containers.agent.processAgent.enabled | quote }} + - name: STS_CONTAINER_CHECK_INTERVAL + value: {{ .Values.processAgent.checkIntervals.container | quote }} + - name: STS_CONNECTION_CHECK_INTERVAL + value: {{ .Values.processAgent.checkIntervals.connections | quote }} + - name: STS_PROCESS_CHECK_INTERVAL + value: {{ .Values.processAgent.checkIntervals.process | quote }} + - name: STS_PROCESS_AGENT_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + + - name: STS_SKIP_SSL_VALIDATION + value: {{ or .Values.global.skipSslValidation .Values.nodeAgent.skipSslValidation | quote }} + - name: STS_SKIP_KUBELET_TLS_VERIFY + value: {{ .Values.nodeAgent.skipKubeletTLSVerify | quote }} + - name: STS_STS_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + {{- if .Values.nodeAgent.containerRuntime.customSocketPath }} + - name: STS_CRI_SOCKET_PATH + value: {{ .Values.nodeAgent.containerRuntime.customSocketPath }} + {{- end }} + {{- if .Values.global.proxy.url }} + - name: STS_PROXY_HTTPS + value: {{ .Values.global.proxy.url | quote }} + - name: STS_PROXY_HTTP + value: {{ .Values.global.proxy.url | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.open }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.secret }} + - name: {{ $key }} + valueFrom: + secretKeyRef: + name: {{ include "stackstate-k8s-agent.fullname" . }} + key: {{ $key }} + {{- end }} + {{- range $key, $value := .Values.nodeAgent.containers.agent.env }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- if .Values.nodeAgent.containers.agent.livenessProbe.enabled }} + livenessProbe: + httpGet: + path: /health + port: healthport + failureThreshold: {{ .Values.nodeAgent.containers.agent.livenessProbe.failureThreshold }} + initialDelaySeconds: {{ .Values.nodeAgent.containers.agent.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.nodeAgent.containers.agent.livenessProbe.periodSeconds }} + successThreshold: {{ .Values.nodeAgent.containers.agent.livenessProbe.successThreshold }} + timeoutSeconds: {{ .Values.nodeAgent.containers.agent.livenessProbe.timeoutSeconds }} + {{- end }} + {{- if .Values.nodeAgent.containers.agent.readinessProbe.enabled }} + readinessProbe: + httpGet: + path: /health + port: healthport + failureThreshold: {{ .Values.nodeAgent.containers.agent.readinessProbe.failureThreshold }} + initialDelaySeconds: {{ .Values.nodeAgent.containers.agent.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.nodeAgent.containers.agent.readinessProbe.periodSeconds }} + successThreshold: {{ .Values.nodeAgent.containers.agent.readinessProbe.successThreshold }} + timeoutSeconds: {{ .Values.nodeAgent.containers.agent.readinessProbe.timeoutSeconds }} + {{- end }} + ports: + - containerPort: 8126 + name: traceport + protocol: TCP + - containerPort: 5555 + name: healthport + protocol: TCP + {{- with .Values.nodeAgent.containers.agent.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.nodeAgent.containerRuntime.customSocketPath }} + - name: customcrisocket + mountPath: {{ .Values.nodeAgent.containerRuntime.customSocketPath }} + readOnly: true + {{- end }} + - name: crisocket + mountPath: /var/run/crio/crio.sock + readOnly: true + - name: containerdsocket + mountPath: /var/run/containerd/containerd.sock + readOnly: true + - name: kubelet + mountPath: /var/lib/kubelet + readOnly: true + - name: nfs + mountPath: /var/lib/nfs + readOnly: true + - name: dockersocket + mountPath: /var/run/docker.sock + readOnly: true + - name: dockernetns + mountPath: /run/docker/netns + readOnly: true + - name: dockeroverlay2 + mountPath: /var/lib/docker/overlay2 + readOnly: true + - name: procdir + mountPath: /host/proc + readOnly: true + - name: cgroups + mountPath: /host/sys/fs/cgroup + readOnly: true + {{- if .Values.nodeAgent.config.override }} + {{- range .Values.nodeAgent.config.override }} + - name: config-override-volume + mountPath: {{ .path }}/{{ .name }} + subPath: {{ .path | replace "/" "_"}}_{{ .name }} + readOnly: true + {{- end }} + {{- end }} +{{- if .Values.all.hardening.enabled}} + securityContext: + privileged: true + runAsUser: 0 # root + capabilities: + add: [ "ALL" ] + readOnlyRootFilesystem: false +{{- else }} + securityContext: + privileged: false +{{- end }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-process-agent.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-process-agent.yaml new file mode 100644 index 0000000000..893f11581a --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_container-process-agent.yaml @@ -0,0 +1,160 @@ +{{- define "container-process-agent" -}} +- name: process-agent +{{ if .Values.nodeAgent.containers.processAgent.image.registry }} + image: "{{ .Values.nodeAgent.containers.processAgent.image.registry }}/{{ .Values.nodeAgent.containers.processAgent.image.repository }}:{{ .Values.nodeAgent.containers.processAgent.image.tag }}" +{{ else }} + image: "{{ include "stackstate-k8s-agent.imageRegistry" . }}/{{ .Values.nodeAgent.containers.processAgent.image.repository }}:{{ .Values.nodeAgent.containers.processAgent.image.tag }}" +{{- end }} + imagePullPolicy: "{{ .Values.nodeAgent.containers.processAgent.image.pullPolicy }}" + ports: + - containerPort: 6063 + env: + {{ include "stackstate-k8s-agent.apiKeyEnv" . | nindent 4 }} + - name: STS_KUBERNETES_KUBELET_HOST + valueFrom: + fieldRef: + fieldPath: status.hostIP + - name: KUBERNETES_HOSTNAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: STS_HOSTNAME + value: "$(KUBERNETES_HOSTNAME)-{{ .Values.stackstate.cluster.name}}" + - name: AGENT_VERSION + value: {{ .Values.nodeAgent.containers.processAgent.image.tag | quote }} + - name: STS_LOG_TO_CONSOLE + value: "true" + - name: HOST_PROC + value: "/host/proc" + - name: HOST_SYS + value: "/host/sys" + - name: HOST_ETC + value: "/host/etc" + - name: KUBERNETES + value: "true" + - name: STS_CLUSTER_AGENT_ENABLED + value: {{ .Values.clusterAgent.enabled | quote }} + {{- if .Values.clusterAgent.enabled }} + - name: STS_CLUSTER_AGENT_KUBERNETES_SERVICE_NAME + value: {{ .Release.Name }}-cluster-agent + {{ include "stackstate-k8s-agent.clusterAgentAuthTokenEnv" . | nindent 4 }} + {{- end }} + - name: STS_CLUSTER_NAME + value: {{ .Values.stackstate.cluster.name | quote }} + - name: STS_SKIP_VALIDATE_CLUSTERNAME + value: "true" + - name: LOG_LEVEL + value: {{ .Values.nodeAgent.containers.processAgent.logLevel | default .Values.nodeAgent.logLevel | quote }} + - name: STS_LOG_LEVEL + value: {{ .Values.nodeAgent.containers.processAgent.logLevel | default .Values.nodeAgent.logLevel | quote }} + - name: STS_NETWORK_TRACING_ENABLED + value: {{ .Values.nodeAgent.networkTracing.enabled | quote }} + - name: STS_PROTOCOL_INSPECTION_ENABLED + value: {{ .Values.nodeAgent.protocolInspection.enabled | quote }} + - name: STS_PROCESS_AGENT_ENABLED + value: {{ .Values.nodeAgent.containers.processAgent.enabled | quote }} + - name: STS_CONTAINER_CHECK_INTERVAL + value: {{ .Values.processAgent.checkIntervals.container | quote }} + - name: STS_CONNECTION_CHECK_INTERVAL + value: {{ .Values.processAgent.checkIntervals.connections | quote }} + - name: STS_PROCESS_CHECK_INTERVAL + value: {{ .Values.processAgent.checkIntervals.process | quote }} + - name: GOMEMLIMIT + value: {{ .Values.processAgent.softMemoryLimit.goMemLimit | quote }} + - name: STS_HTTP_STATS_BUFFER_SIZE + value: {{ .Values.processAgent.softMemoryLimit.httpStatsBufferSize | quote }} + - name: STS_HTTP_OBSERVATIONS_BUFFER_SIZE + value: {{ .Values.processAgent.softMemoryLimit.httpObservationsBufferSize | quote }} + - name: STS_PROCESS_AGENT_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + - name: STS_SKIP_SSL_VALIDATION + value: {{ or .Values.global.skipSslValidation .Values.nodeAgent.skipSslValidation | quote }} + - name: STS_SKIP_KUBELET_TLS_VERIFY + value: {{ .Values.nodeAgent.skipKubeletTLSVerify | quote }} + - name: STS_STS_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + - name: STS_HTTP_TRACING_ENABLED + value: {{ .Values.nodeAgent.httpTracing.enabled | quote }} + {{- if .Values.nodeAgent.containerRuntime.customSocketPath }} + - name: STS_CRI_SOCKET_PATH + value: {{ .Values.nodeAgent.containerRuntime.customSocketPath }} + {{- end }} + {{- if .Values.global.proxy.url }} + - name: STS_PROXY_HTTPS + value: {{ .Values.global.proxy.url | quote }} + - name: STS_PROXY_HTTP + value: {{ .Values.global.proxy.url | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.open }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.secret }} + - name: {{ $key }} + valueFrom: + secretKeyRef: + name: {{ include "stackstate-k8s-agent.fullname" . }} + key: {{ $key }} + {{- end }} + {{- range $key, $value := .Values.nodeAgent.containers.processAgent.env }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- with .Values.nodeAgent.containers.processAgent.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + {{- if .Values.nodeAgent.containerRuntime.customSocketPath }} + - name: customcrisocket + mountPath: {{ .Values.nodeAgent.containerRuntime.customSocketPath }} + readOnly: true + {{- end }} + - name: crisocket + mountPath: /var/run/crio/crio.sock + readOnly: true + - name: containerdsocket + mountPath: /var/run/containerd/containerd.sock + readOnly: true + - name: sys-kernel-debug + mountPath: /sys/kernel/debug + # Having sys-kernel-debug as read only breaks specific monitors from receiving metrics + # readOnly: true + - name: dockersocket + mountPath: /var/run/docker.sock + readOnly: true + # The agent needs access to /etc to figure out what os it is running on. + - name: etcdir + mountPath: /host/etc + readOnly: true + - name: procdir + mountPath: /host/proc + # We have an agent option STS_DISABLE_BPF_JIT_HARDEN that write to /proc. this is a debug setting but if we want to use + # it, we have the option to make /proc writable. + readOnly: {{ .Values.nodeAgent.containers.processAgent.procVolumeReadOnly }} + - name: passwd + mountPath: /etc/passwd + readOnly: true + - name: cgroups + mountPath: /host/sys/fs/cgroup + readOnly: true + {{- if .Values.nodeAgent.config.override }} + {{- range .Values.nodeAgent.config.override }} + - name: config-override-volume + mountPath: {{ .path }}/{{ .name }} + subPath: {{ .path | replace "/" "_"}}_{{ .name }} + readOnly: true + {{- end }} + {{- end }} +{{- if .Values.all.hardening.enabled}} + securityContext: + privileged: true + runAsUser: 0 # root + capabilities: + add: [ "ALL" ] + readOnlyRootFilesystem: false +{{- else }} + securityContext: + privileged: true +{{- end }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_helpers.tpl b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_helpers.tpl new file mode 100644 index 0000000000..3c51bc3087 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/_helpers.tpl @@ -0,0 +1,219 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "stackstate-k8s-agent.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 "stackstate-k8s-agent.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 "stackstate-k8s-agent.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Common labels +*/}} +{{- define "stackstate-k8s-agent.labels" -}} +app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +helm.sh/chart: {{ include "stackstate-k8s-agent.chart" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end -}} + +{{/* +Cluster agent checksum annotations +*/}} +{{- define "stackstate-k8s-agent.checksum-configs" }} +checksum/secret: {{ include (print $.Template.BasePath "/secret.yaml") . | sha256sum }} +{{- end }} + +{{/* +StackState URL function +*/}} +{{- define "stackstate-k8s-agent.stackstate.url" -}} +{{ tpl .Values.stackstate.url . | quote }} +{{- end }} + +{{- define "stackstate-k8s-agent.configmap.override.checksum" -}} +{{- if .Values.clusterAgent.config.override }} +checksum/override-configmap: {{ include (print $.Template.BasePath "/cluster-agent-configmap.yaml") . | sha256sum }} +{{- end }} +{{- end }} + +{{- define "stackstate-k8s-agent.nodeAgent.configmap.override.checksum" -}} +{{- if .Values.nodeAgent.config.override }} +checksum/override-configmap: {{ include (print $.Template.BasePath "/node-agent-configmap.yaml") . | sha256sum }} +{{- end }} +{{- end }} + +{{- define "stackstate-k8s-agent.logsAgent.configmap.override.checksum" -}} +checksum/override-configmap: {{ include (print $.Template.BasePath "/logs-agent-configmap.yaml") . | sha256sum }} +{{- end }} + +{{- define "stackstate-k8s-agent.checksAgent.configmap.override.checksum" -}} +{{- if .Values.checksAgent.config.override }} +checksum/override-configmap: {{ include (print $.Template.BasePath "/checks-agent-configmap.yaml") . | sha256sum }} +{{- end }} +{{- end }} + + +{{/* +Return the image registry +*/}} +{{- define "stackstate-k8s-agent.imageRegistry" -}} + {{- if .Values.global }} + {{- .Values.global.imageRegistry | default .Values.all.image.registry -}} + {{- else -}} + {{- .Values.all.image.registry -}} + {{- end -}} +{{- end -}} + +{{/* +Renders a value that contains a template. +Usage: +{{ include "stackstate-k8s-agent.tplvalue.render" ( dict "value" .Values.path.to.the.Value "context" $) }} +*/}} +{{- define "stackstate-k8s-agent.tplvalue.render" -}} + {{- if typeIs "string" .value }} + {{- tpl .value .context }} + {{- else }} + {{- tpl (.value | toYaml) .context }} + {{- end }} +{{- end -}} + +{{- define "stackstate-k8s-agent.pull-secret.name" -}} +{{ include "stackstate-k8s-agent.fullname" . }}-pull-secret +{{- end -}} + +{{/* +Return the proper Docker Image Registry Secret Names evaluating values as templates +{{ include "stackstate-k8s-agent.image.pullSecrets" ( dict "images" (list .Values.path.to.the.image1, .Values.path.to.the.image2) "context" $) }} +*/}} +{{- define "stackstate-k8s-agent.image.pullSecrets" -}} + {{- $pullSecrets := list }} + {{- $context := .context }} + {{- if $context.Values.global }} + {{- range $context.Values.global.imagePullSecrets -}} + {{/* Is plain array of strings, compatible with all bitnami charts */}} + {{- $pullSecrets = append $pullSecrets (include "stackstate-k8s-agent.tplvalue.render" (dict "value" . "context" $context)) -}} + {{- end -}} + {{- end -}} + {{- range $context.Values.imagePullSecrets -}} + {{- $pullSecrets = append $pullSecrets (include "stackstate-k8s-agent.tplvalue.render" (dict "value" .name "context" $context)) -}} + {{- end -}} + {{- range .images -}} + {{- if .pullSecretName -}} + {{- $pullSecrets = append $pullSecrets (include "stackstate-k8s-agent.tplvalue.render" (dict "value" .pullSecretName "context" $context)) -}} + {{- end -}} + {{- end -}} + {{- $pullSecrets = append $pullSecrets (include "stackstate-k8s-agent.pull-secret.name" $context) -}} + {{- if (not (empty $pullSecrets)) -}} +imagePullSecrets: + {{- range $pullSecrets | uniq }} + - name: {{ . }} + {{- end }} + {{- end }} +{{- end -}} + +{{/* +Check whether the kubernetes-state-metrics configuration is overridden. If so, return 'true' else return nothing (which is false). +{{ include "stackstate-k8s-agent.kube-state-metrics.overridden" $ }} +*/}} +{{- define "stackstate-k8s-agent.kube-state-metrics.overridden" -}} +{{- if .Values.clusterAgent.config.override }} + {{- range $i, $val := .Values.clusterAgent.config.override }} + {{- if and (eq $val.name "conf.yaml") (eq $val.path "/etc/stackstate-agent/conf.d/kubernetes_state.d") }} +true + {{- end }} + {{- end }} +{{- end }} +{{- end -}} + +{{- define "stackstate-k8s-agent.nodeAgent.kube-state-metrics.overridden" -}} +{{- if .Values.nodeAgent.config.override }} + {{- range $i, $val := .Values.nodeAgent.config.override }} + {{- if and (eq $val.name "auto_conf.yaml") (eq $val.path "/etc/stackstate-agent/conf.d/kubernetes_state.d") }} +true + {{- end }} + {{- end }} +{{- end }} +{{- end -}} + +{{/* +Return the appropriate os label +*/}} +{{- define "label.os" -}} +{{- if semverCompare "^1.14-0" .Capabilities.KubeVersion.GitVersion -}} +kubernetes.io/os +{{- else -}} +beta.kubernetes.io/os +{{- end -}} +{{- end -}} + +{{/* +Returns a YAML with extra annotations +*/}} +{{- define "stackstate-k8s-agent.global.extraAnnotations" -}} +{{- with .Values.global.extraAnnotations }} +{{- toYaml . }} +{{- end }} +{{- end -}} + +{{/* +Returns a YAML with extra labels +*/}} +{{- define "stackstate-k8s-agent.global.extraLabels" -}} +{{- with .Values.global.extraLabels }} +{{- toYaml . }} +{{- end }} +{{- end -}} + +{{- define "stackstate-k8s-agent.apiKeyEnv" -}} +- name: STS_API_KEY + valueFrom: + secretKeyRef: +{{- if not .Values.stackstate.manageOwnSecrets }} + name: {{ include "stackstate-k8s-agent.fullname" . }} + key: sts-api-key +{{- else }} + name: {{ .Values.stackstate.customSecretName | quote }} + key: {{ .Values.stackstate.customApiKeySecretKey | quote }} +{{- end }} +{{- end -}} + +{{- define "stackstate-k8s-agent.clusterAgentAuthTokenEnv" -}} +- name: STS_CLUSTER_AGENT_AUTH_TOKEN + valueFrom: + secretKeyRef: +{{- if not .Values.stackstate.manageOwnSecrets }} + name: {{ include "stackstate-k8s-agent.fullname" . }} + key: sts-cluster-auth-token +{{- else }} + name: {{ .Values.stackstate.customSecretName | quote }} + key: {{ .Values.stackstate.customClusterAuthTokenSecretKey | quote }} +{{- end }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-clusterrolebinding.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-clusterrolebinding.yaml new file mode 100644 index 0000000000..4fd0eadbcd --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-clusterrolebinding.yaml @@ -0,0 +1,21 @@ +{{- if .Values.checksAgent.enabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ .Release.Name }}-checks-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: checks-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-node-agent +subjects: +- apiGroup: "" + kind: ServiceAccount + name: {{ .Release.Name }}-checks-agent + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-configmap.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-configmap.yaml new file mode 100644 index 0000000000..54a1abf2fe --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-configmap.yaml @@ -0,0 +1,17 @@ +{{- if and .Values.checksAgent.enabled .Values.checksAgent.config.override }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ .Release.Name }}-checks-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: checks-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +data: +{{- range .Values.checksAgent.config.override }} + {{ .path | replace "/" "_"}}_{{ .name }}: | +{{ .data | indent 4 -}} +{{- end -}} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-deployment.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-deployment.yaml new file mode 100644 index 0000000000..37a0b1a1db --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-deployment.yaml @@ -0,0 +1,185 @@ +{{- if .Values.checksAgent.enabled }} +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Release.Name }}-checks-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: checks-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + selector: + matchLabels: + app.kubernetes.io/component: checks-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} + replicas: {{ .Values.checksAgent.replicas }} +{{- with .Values.checksAgent.strategy }} + strategy: + {{- toYaml . | nindent 4 }} +{{- end }} + template: + metadata: + annotations: + {{- include "stackstate-k8s-agent.checksum-configs" . | nindent 8 }} + {{- include "stackstate-k8s-agent.nodeAgent.configmap.override.checksum" . | nindent 8 }} +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 8 }} + labels: + app.kubernetes.io/component: checks-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 8 }} + spec: + {{- include "stackstate-k8s-agent.image.pullSecrets" (dict "images" (list .Values.checksAgent.image .Values.all.image) "context" $) | nindent 6 }} + {{- if .Values.all.hardening.enabled}} + terminationGracePeriodSeconds: 240 + {{- end }} + containers: + - name: {{ .Chart.Name }} + image: "{{ include "stackstate-k8s-agent.imageRegistry" . }}/{{ .Values.checksAgent.image.repository }}:{{ .Values.checksAgent.image.tag }}" + imagePullPolicy: "{{ .Values.checksAgent.image.pullPolicy }}" + {{- if .Values.all.hardening.enabled}} + lifecycle: + preStop: + exec: + command: [ "/bin/sh", "-c", "echo 'Giving slim.ai monitor time to submit data...'; sleep 120" ] + {{- end }} + env: + {{ include "stackstate-k8s-agent.apiKeyEnv" . | nindent 10 }} + - name: KUBERNETES_HOSTNAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: STS_HOSTNAME + value: "$(KUBERNETES_HOSTNAME)-{{ .Values.stackstate.cluster.name}}" + - name: AGENT_VERSION + value: {{ .Values.checksAgent.image.tag | quote }} + - name: LOG_LEVEL + value: {{ .Values.checksAgent.logLevel | quote }} + - name: STS_APM_ENABLED + value: "false" + - name: STS_CLUSTER_AGENT_ENABLED + value: {{ .Values.clusterAgent.enabled | quote }} + {{- if .Values.clusterAgent.enabled }} + - name: STS_CLUSTER_AGENT_KUBERNETES_SERVICE_NAME + value: {{ .Release.Name }}-cluster-agent + {{ include "stackstate-k8s-agent.clusterAgentAuthTokenEnv" . | nindent 10 }} + {{- end }} + - name: STS_CLUSTER_NAME + value: {{ .Values.stackstate.cluster.name | quote }} + - name: STS_SKIP_VALIDATE_CLUSTERNAME + value: "true" + - name: STS_CHECKS_TAG_CARDINALITY + value: {{ .Values.checksAgent.checksTagCardinality | quote }} + - name: STS_EXTRA_CONFIG_PROVIDERS + value: "clusterchecks" + - name: STS_HEALTH_PORT + value: "5555" + - name: STS_LEADER_ELECTION + value: "false" + - name: STS_LOG_LEVEL + value: {{ .Values.checksAgent.logLevel | quote }} + - name: STS_NETWORK_TRACING_ENABLED + value: "false" + - name: STS_PROCESS_AGENT_ENABLED + value: "false" + - name: STS_SKIP_SSL_VALIDATION + value: {{ or .Values.global.skipSslValidation .Values.checksAgent.skipSslValidation | quote }} + - name: STS_STS_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + {{- if .Values.global.proxy.url }} + - name: STS_PROXY_HTTPS + value: {{ .Values.global.proxy.url | quote }} + - name: STS_PROXY_HTTP + value: {{ .Values.global.proxy.url | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.open }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.secret }} + - name: {{ $key }} + valueFrom: + secretKeyRef: + name: {{ include "stackstate-k8s-agent.fullname" . }} + key: {{ $key }} + {{- end }} + livenessProbe: + httpGet: + path: /health + port: healthport + failureThreshold: {{ .Values.checksAgent.livenessProbe.failureThreshold }} + initialDelaySeconds: {{ .Values.checksAgent.livenessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.checksAgent.livenessProbe.periodSeconds }} + successThreshold: {{ .Values.checksAgent.livenessProbe.successThreshold }} + timeoutSeconds: {{ .Values.checksAgent.livenessProbe.timeoutSeconds }} + readinessProbe: + httpGet: + path: /health + port: healthport + failureThreshold: {{ .Values.checksAgent.readinessProbe.failureThreshold }} + initialDelaySeconds: {{ .Values.checksAgent.readinessProbe.initialDelaySeconds }} + periodSeconds: {{ .Values.checksAgent.readinessProbe.periodSeconds }} + successThreshold: {{ .Values.checksAgent.readinessProbe.successThreshold }} + timeoutSeconds: {{ .Values.checksAgent.readinessProbe.timeoutSeconds }} + ports: + - containerPort: 5555 + name: healthport + protocol: TCP + {{- if .Values.all.hardening.enabled}} + securityContext: + privileged: true + runAsUser: 0 # root + capabilities: + add: [ "ALL" ] + readOnlyRootFilesystem: false + {{- else }} + securityContext: + privileged: false + {{- end }} + {{- with .Values.checksAgent.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: confd-empty-volume + mountPath: /etc/stackstate-agent/conf.d +# setting as readOnly: false because we need the ability to write data on /etc/stackstate-agent/conf.d as we enable checks to run. + readOnly: false + {{- if .Values.checksAgent.config.override }} + {{- range .Values.checksAgent.config.override }} + - name: config-override-volume + mountPath: {{ .path }}/{{ .name }} + subPath: {{ .path | replace "/" "_"}}_{{ .name }} + readOnly: true + {{- end }} + {{- end }} + {{- if .Values.checksAgent.priorityClassName }} + priorityClassName: {{ .Values.checksAgent.priorityClassName }} + {{- end }} + serviceAccountName: {{ .Release.Name }}-checks-agent + nodeSelector: + {{ template "label.os" . }}: {{ .Values.targetSystem }} + {{- with .Values.checksAgent.nodeSelector }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.checksAgent.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.checksAgent.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: confd-empty-volume + emptyDir: {} + {{- if .Values.checksAgent.config.override }} + - name: config-override-volume + configMap: + name: {{ .Release.Name }}-checks-agent + {{- end }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-poddisruptionbudget.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-poddisruptionbudget.yaml new file mode 100644 index 0000000000..19d3924ea3 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-poddisruptionbudget.yaml @@ -0,0 +1,23 @@ +{{- if .Values.checksAgent.enabled }} +{{- if .Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" }} +apiVersion: policy/v1 +{{- else }} +apiVersion: policy/v1beta1 +{{- end }} +kind: PodDisruptionBudget +metadata: + name: {{ .Release.Name }}-checks-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: checks-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + maxUnavailable: 1 + selector: + matchLabels: + app.kubernetes.io/component: checks-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-serviceaccount.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-serviceaccount.yaml new file mode 100644 index 0000000000..a90a43589d --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/checks-agent-serviceaccount.yaml @@ -0,0 +1,16 @@ +{{- if .Values.checksAgent.enabled }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ .Release.Name }}-checks-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: checks-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +{{- with .Values.checksAgent.serviceaccount.annotations }} + {{- toYaml . | nindent 4 }} +{{- end }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrole.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrole.yaml new file mode 100644 index 0000000000..021a43ebdf --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrole.yaml @@ -0,0 +1,152 @@ +{{- $kubeRes := .Values.clusterAgent.collection.kubernetesResources }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +rules: +- apiGroups: + - "" + resources: + - events + - nodes + - pods + - services + {{- if $kubeRes.namespaces }} + - namespaces + {{- end }} + {{- if .Values.clusterAgent.collection.kubernetesMetrics }} + - componentstatuses + {{- end }} + {{- if $kubeRes.configmaps }} + - configmaps + {{- end }} + {{- if $kubeRes.endpoints }} + - endpoints + {{- end }} + {{- if $kubeRes.persistentvolumeclaims }} + - persistentvolumeclaims + {{- end }} + {{- if $kubeRes.persistentvolumes }} + - persistentvolumes + {{- end }} + {{- if $kubeRes.secrets }} + - secrets + {{- end }} + {{- if $kubeRes.resourcequotas }} + - resourcequotas + {{- end }} + verbs: + - get + - list + - watch +{{- if or $kubeRes.daemonsets $kubeRes.deployments $kubeRes.replicasets $kubeRes.statefulsets }} +- apiGroups: + - "apps" + resources: + {{- if $kubeRes.daemonsets }} + - daemonsets + {{- end }} + {{- if $kubeRes.deployments }} + - deployments + {{- end }} + {{- if $kubeRes.replicasets }} + - replicasets + {{- end }} + {{- if $kubeRes.statefulsets }} + - statefulsets + {{- end }} + verbs: + - get + - list + - watch +{{- end}} +{{- if $kubeRes.ingresses }} +- apiGroups: + - "extensions" + - "networking.k8s.io" + resources: + - ingresses + verbs: + - get + - list + - watch +{{- end}} +{{- if or $kubeRes.cronjobs $kubeRes.jobs }} +- apiGroups: + - "batch" + resources: + {{- if $kubeRes.cronjobs }} + - cronjobs + {{- end }} + {{- if $kubeRes.jobs }} + - jobs + {{- end }} + verbs: + - get + - list + - watch +{{- end}} +- nonResourceURLs: + - "/healthz" + - "/version" + verbs: + - get +- apiGroups: + - "storage.k8s.io" + resources: + {{- if $kubeRes.volumeattachments }} + - volumeattachments + {{- end }} + {{- if $kubeRes.storageclasses }} + - storageclasses + {{- end }} + verbs: + - get + - list + - watch +- apiGroups: + - "policy" + resources: + {{- if $kubeRes.poddisruptionbudgets }} + - poddisruptionbudgets + {{- end }} + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + {{- if $kubeRes.replicationcontrollers }} + - replicationcontrollers + {{- end }} + verbs: + - get + - list + - watch +- apiGroups: + - "autoscaling" + resources: + {{- if $kubeRes.horizontalpodautoscalers }} + - horizontalpodautoscalers + {{- end }} + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + {{- if $kubeRes.limitranges }} + - limitranges + {{- end }} + verbs: + - get + - list + - watch diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrolebinding.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrolebinding.yaml new file mode 100644 index 0000000000..207613dd9e --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-clusterrolebinding.yaml @@ -0,0 +1,19 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "stackstate-k8s-agent.fullname" . }} +subjects: +- apiGroup: "" + kind: ServiceAccount + name: {{ include "stackstate-k8s-agent.fullname" . }} + namespace: {{ .Release.Namespace }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-configmap.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-configmap.yaml new file mode 100644 index 0000000000..37d10217f8 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-configmap.yaml @@ -0,0 +1,31 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ .Release.Name }}-cluster-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +data: + kubernetes_api_events_conf: | + init_config: + instances: + - collect_events: {{ .Values.clusterAgent.collection.kubernetesEvents }} + event_categories:{{ .Values.clusterAgent.config.events.categories | toYaml | nindent 10 }} + kubernetes_api_topology_conf: | + init_config: + instances: + - collection_interval: {{ .Values.clusterAgent.config.topology.collectionInterval }} + resources:{{ .Values.clusterAgent.collection.kubernetesResources | toYaml | nindent 10 }} + {{- if .Values.clusterAgent.collection.kubeStateMetrics.enabled }} + kube_state_metrics_core_conf: | + {{- include "cluster-agent-kube-state-metrics" . | nindent 6 }} + {{- end }} +{{- if .Values.clusterAgent.config.override }} +{{- range .Values.clusterAgent.config.override }} + {{ .path | replace "/" "_"}}_{{ .name }}: | +{{ .data | indent 4 -}} +{{- end -}} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-deployment.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-deployment.yaml new file mode 100644 index 0000000000..51025a6703 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-deployment.yaml @@ -0,0 +1,169 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Release.Name }}-cluster-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + replicas: {{ .Values.clusterAgent.replicaCount }} + selector: + matchLabels: + app.kubernetes.io/component: cluster-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{- with .Values.clusterAgent.strategy }} + strategy: + {{- toYaml . | nindent 4 }} +{{- end }} + template: + metadata: + annotations: + {{- include "stackstate-k8s-agent.checksum-configs" . | nindent 8 }} + {{- include "stackstate-k8s-agent.configmap.override.checksum" . | nindent 8 }} +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 8 }} + labels: + app.kubernetes.io/component: cluster-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 8 }} + spec: + {{- include "stackstate-k8s-agent.image.pullSecrets" (dict "images" (list .Values.clusterAgent.image .Values.all.image) "context" $) | nindent 6 }} + {{- if .Values.clusterAgent.priorityClassName }} + priorityClassName: {{ .Values.clusterAgent.priorityClassName }} + {{- end }} + serviceAccountName: {{ include "stackstate-k8s-agent.fullname" . }} + {{- if .Values.all.hardening.enabled}} + terminationGracePeriodSeconds: 240 + {{- end }} + containers: + - name: cluster-agent + image: "{{ include "stackstate-k8s-agent.imageRegistry" . }}/{{ .Values.clusterAgent.image.repository }}:{{ .Values.clusterAgent.image.tag }}" + imagePullPolicy: "{{ .Values.clusterAgent.image.pullPolicy }}" + {{- if .Values.all.hardening.enabled}} + lifecycle: + preStop: + exec: + command: [ "/bin/sh", "-c", "echo 'Giving slim.ai monitor time to submit data...'; sleep 120" ] + {{- end }} + env: + {{ include "stackstate-k8s-agent.apiKeyEnv" . | nindent 10 }} + {{ include "stackstate-k8s-agent.clusterAgentAuthTokenEnv" . | nindent 10 }} + - name: KUBERNETES_HOSTNAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + - name: STS_HOSTNAME + value: "$(KUBERNETES_HOSTNAME)-{{ .Values.stackstate.cluster.name}}" + - name: LOG_LEVEL + value: {{ .Values.clusterAgent.logLevel | quote }} + {{- if .Values.checksAgent.enabled }} + - name: STS_CLUSTER_CHECKS_ENABLED + value: "true" + - name: STS_EXTRA_CONFIG_PROVIDERS + value: "kube_endpoints kube_services" + - name: STS_EXTRA_LISTENERS + value: "kube_endpoints kube_services" + {{- end }} + - name: STS_CLUSTER_NAME + value: {{.Values.stackstate.cluster.name | quote }} + - name: STS_SKIP_VALIDATE_CLUSTERNAME + value: "true" + - name: STS_SKIP_SSL_VALIDATION + value: {{ or .Values.global.skipSslValidation .Values.clusterAgent.skipSslValidation | quote }} + - name: STS_COLLECT_KUBERNETES_METRICS + value: {{ .Values.clusterAgent.collection.kubernetesMetrics | quote }} + - name: STS_COLLECT_KUBERNETES_TIMEOUT + value: {{ .Values.clusterAgent.collection.kubernetesTimeout | quote }} + - name: STS_COLLECT_KUBERNETES_TOPOLOGY + value: {{ .Values.clusterAgent.collection.kubernetesTopology | quote }} + - name: STS_LEADER_ELECTION + value: "true" + - name: STS_LOG_LEVEL + value: {{ .Values.clusterAgent.logLevel | quote }} + - name: STS_CLUSTER_AGENT_CMD_PORT + value: {{ .Values.clusterAgent.service.targetPort | quote }} + - name: STS_STS_URL + value: {{ include "stackstate-k8s-agent.stackstate.url" . }} + {{- if .Values.clusterAgent.config.configMap.maxDataSize }} + - name: STS_CONFIGMAP_MAX_DATASIZE + value: {{ .Values.clusterAgent.config.configMap.maxDataSize | quote }} + {{- end}} + {{- if .Values.global.proxy.url }} + - name: STS_PROXY_HTTPS + value: {{ .Values.global.proxy.url | quote }} + - name: STS_PROXY_HTTP + value: {{ .Values.global.proxy.url | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.open }} + - name: {{ $key }} + value: {{ $value | quote }} + {{- end }} + {{- range $key, $value := .Values.global.extraEnv.secret }} + - name: {{ $key }} + valueFrom: + secretKeyRef: + name: {{ include "stackstate-k8s-agent.fullname" . }} + key: {{ $key }} + {{- end }} + {{- if .Values.all.hardening.enabled}} + securityContext: + privileged: true + runAsUser: 0 # root + capabilities: + add: [ "ALL" ] + readOnlyRootFilesystem: false + {{- else }} + securityContext: + privileged: false + {{- end }} + {{- with .Values.clusterAgent.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: logs + mountPath: /var/log/stackstate-agent + - name: config-override-volume + mountPath: /etc/stackstate-agent/conf.d/kubernetes_api_events.d/conf.yaml + subPath: kubernetes_api_events_conf + - name: config-override-volume + mountPath: /etc/stackstate-agent/conf.d/kubernetes_api_topology.d/conf.yaml + subPath: kubernetes_api_topology_conf + readOnly: true + {{- if .Values.clusterAgent.collection.kubeStateMetrics.enabled }} + - name: config-override-volume + mountPath: /etc/stackstate-agent/conf.d/kubernetes_state_core.d/conf.yaml + subPath: kube_state_metrics_core_conf + readOnly: true + {{- end }} + {{- if .Values.clusterAgent.config.override }} + {{- range .Values.clusterAgent.config.override }} + - name: config-override-volume + mountPath: {{ .path }}/{{ .name }} + subPath: {{ .path | replace "/" "_"}}_{{ .name }} + readOnly: true + {{- end }} + {{- end }} + nodeSelector: + {{ template "label.os" . }}: {{ .Values.targetSystem }} + {{- with .Values.clusterAgent.nodeSelector }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.clusterAgent.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.clusterAgent.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: logs + emptyDir: {} + - name: config-override-volume + configMap: + name: {{ .Release.Name }}-cluster-agent diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-poddisruptionbudget.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-poddisruptionbudget.yaml new file mode 100644 index 0000000000..64a265b7db --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-poddisruptionbudget.yaml @@ -0,0 +1,21 @@ +{{- if .Capabilities.APIVersions.Has "policy/v1/PodDisruptionBudget" }} +apiVersion: policy/v1 +{{- else }} +apiVersion: policy/v1beta1 +{{- end }} +kind: PodDisruptionBudget +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + maxUnavailable: 1 + selector: + matchLabels: + app.kubernetes.io/component: cluster-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-role.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-role.yaml new file mode 100644 index 0000000000..eabc5bde36 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-role.yaml @@ -0,0 +1,21 @@ +{{- $kubeRes := .Values.clusterAgent.collection.kubernetesResources }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +rules: +- apiGroups: + - "" + resources: + - configmaps + verbs: + - create + - get + - patch + - update diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-rolebinding.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-rolebinding.yaml new file mode 100644 index 0000000000..adabad45e0 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-rolebinding.yaml @@ -0,0 +1,18 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "stackstate-k8s-agent.fullname" . }} +subjects: +- apiGroup: "" + kind: ServiceAccount + name: {{ include "stackstate-k8s-agent.fullname" . }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-service.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-service.yaml new file mode 100644 index 0000000000..8b687e8f76 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-service.yaml @@ -0,0 +1,21 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ .Release.Name }}-cluster-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + ports: + - name: clusteragent + port: {{int .Values.clusterAgent.service.port }} + protocol: TCP + targetPort: {{int .Values.clusterAgent.service.targetPort }} + selector: + app.kubernetes.io/component: cluster-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-serviceaccount.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-serviceaccount.yaml new file mode 100644 index 0000000000..6cbc89699e --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/cluster-agent-serviceaccount.yaml @@ -0,0 +1,14 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: cluster-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +{{- with .Values.clusterAgent.serviceaccount.annotations }} + {{- toYaml . | nindent 4 }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrole.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrole.yaml new file mode 100644 index 0000000000..da6cd59dd5 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrole.yaml @@ -0,0 +1,23 @@ +{{- if .Values.logsAgent.enabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ .Release.Name }}-logs-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: logs-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +rules: +- apiGroups: # Kubelet connectivity + - "" + resources: + - nodes + - services + - pods + verbs: + - get + - watch + - list +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrolebinding.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrolebinding.yaml new file mode 100644 index 0000000000..1f6e7cfcff --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-clusterrolebinding.yaml @@ -0,0 +1,21 @@ +{{- if .Values.logsAgent.enabled }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ .Release.Name }}-logs-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: logs-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-logs-agent +subjects: +- apiGroup: "" + kind: ServiceAccount + name: {{ .Release.Name }}-logs-agent + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-configmap.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-configmap.yaml new file mode 100644 index 0000000000..ff9440a4b7 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-configmap.yaml @@ -0,0 +1,63 @@ +{{- if .Values.logsAgent.enabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ .Release.Name }}-logs-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: logs-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +data: + promtail.yaml: | + server: + http_listen_port: 9080 + grpc_listen_port: 0 + + clients: + - url: {{ tpl .Values.stackstate.url . }}/logs/k8s?api_key=${STS_API_KEY} + external_labels: + sts_cluster_name: {{ .Values.stackstate.cluster.name | quote }} + {{- if .Values.global.proxy.url }} + proxy_url: {{ .Values.global.proxy.url | quote }} + {{- end }} + tls_config: + insecure_skip_verify: {{ or .Values.global.skipSslValidation .Values.logsAgent.skipSslValidation }} + + + positions: + filename: /tmp/positions.yaml + target_config: + sync_period: 10s + scrape_configs: + - job_name: pod-logs + kubernetes_sd_configs: + - role: pod + pipeline_stages: + - docker: {} + - cri: {} + relabel_configs: + - action: replace + source_labels: + - __meta_kubernetes_pod_name + target_label: pod_name + - action: replace + source_labels: + - __meta_kubernetes_pod_uid + target_label: pod_uid + - action: replace + source_labels: + - __meta_kubernetes_pod_container_name + target_label: container_name + # The __path__ is required by the promtail client + - replacement: /var/log/pods/*$1/*.log + separator: / + source_labels: + - __meta_kubernetes_pod_uid + - __meta_kubernetes_pod_container_name + target_label: __path__ + # Drop all remaining labels, we do not need those + - action: drop + regex: __meta_(.*) +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-daemonset.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-daemonset.yaml new file mode 100644 index 0000000000..23cfce31f0 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-daemonset.yaml @@ -0,0 +1,91 @@ +{{- if .Values.logsAgent.enabled }} +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ .Release.Name }}-logs-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: logs-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + selector: + matchLabels: + app.kubernetes.io/component: logs-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{- with .Values.logsAgent.updateStrategy }} + updateStrategy: + {{- toYaml . | nindent 4 }} +{{- end }} + template: + metadata: + annotations: + {{- include "stackstate-k8s-agent.checksum-configs" . | nindent 8 }} + {{- include "stackstate-k8s-agent.logsAgent.configmap.override.checksum" . | nindent 8 }} +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 8 }} + labels: + app.kubernetes.io/component: logs-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 8 }} + spec: + {{- include "stackstate-k8s-agent.image.pullSecrets" (dict "images" (list .Values.logsAgent.image .Values.all.image) "context" $) | nindent 6 }} + containers: + - name: logs-agent + image: "{{ include "stackstate-k8s-agent.imageRegistry" . }}/{{ .Values.logsAgent.image.repository }}:{{ .Values.logsAgent.image.tag }}" + args: + - -config.expand-env=true + - -config.file=/etc/promtail/promtail.yaml + imagePullPolicy: "{{ .Values.logsAgent.image.pullPolicy }}" + env: + {{ include "stackstate-k8s-agent.apiKeyEnv" . | nindent 10 }} + - name: "HOSTNAME" # needed when using kubernetes_sd_configs + valueFrom: + fieldRef: + fieldPath: "spec.nodeName" + securityContext: + privileged: false + {{- with .Values.logsAgent.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: logs + mountPath: /var/log + readOnly: true + - name: logs-agent-config + mountPath: /etc/promtail + readOnly: true + - name: varlibdockercontainers + mountPath: /var/lib/docker/containers + readOnly: true + {{- if .Values.logsAgent.priorityClassName }} + priorityClassName: {{ .Values.logsAgent.priorityClassName }} + {{- end }} + serviceAccountName: {{ .Release.Name }}-logs-agent + {{- with .Values.logsAgent.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.logsAgent.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.logsAgent.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: logs + hostPath: + path: /var/log + - name: varlibdockercontainers + hostPath: + path: /var/lib/docker/containers + - name: logs-agent-config + configMap: + name: {{ .Release.Name }}-logs-agent +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-serviceaccount.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-serviceaccount.yaml new file mode 100644 index 0000000000..91cfdb137d --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/logs-agent-serviceaccount.yaml @@ -0,0 +1,16 @@ +{{- if .Values.logsAgent.enabled }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ .Release.Name }}-logs-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: logs-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +{{- with .Values.logsAgent.serviceaccount.annotations }} + {{- toYaml . | nindent 4 }} +{{- end }} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrole.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrole.yaml new file mode 100644 index 0000000000..1ded16cc29 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrole.yaml @@ -0,0 +1,21 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ .Release.Name }}-node-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: node-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +rules: +- apiGroups: # Kubelet connectivity + - "" + resources: + - nodes/metrics + - nodes/proxy + - nodes/spec + - endpoints + verbs: + - get + - list diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrolebinding.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrolebinding.yaml new file mode 100644 index 0000000000..b3f033ebbf --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-clusterrolebinding.yaml @@ -0,0 +1,19 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ .Release.Name }}-node-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: node-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-node-agent +subjects: +- apiGroup: "" + kind: ServiceAccount + name: {{ .Release.Name }}-node-agent + namespace: {{ .Release.Namespace }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-configmap.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-configmap.yaml new file mode 100644 index 0000000000..8f6b2ed3ac --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-configmap.yaml @@ -0,0 +1,17 @@ +{{- if .Values.nodeAgent.config.override }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ .Release.Name }}-node-agent + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: node-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +data: +{{- range .Values.nodeAgent.config.override }} + {{ .path | replace "/" "_"}}_{{ .name }}: | +{{ .data | indent 4 -}} +{{- end -}} +{{- end -}} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-daemonset.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-daemonset.yaml new file mode 100644 index 0000000000..4c8dbdd5a9 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-daemonset.yaml @@ -0,0 +1,110 @@ +--- +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ .Release.Name }}-node-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: node-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + selector: + matchLabels: + app.kubernetes.io/component: node-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{- with .Values.nodeAgent.updateStrategy }} + updateStrategy: + {{- toYaml . | nindent 4 }} +{{- end }} + template: + metadata: + annotations: + {{- include "stackstate-k8s-agent.checksum-configs" . | nindent 8 }} + {{- include "stackstate-k8s-agent.nodeAgent.configmap.override.checksum" . | nindent 8 }} +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 8 }} + labels: + app.kubernetes.io/component: node-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 8 }} + spec: + {{- include "stackstate-k8s-agent.image.pullSecrets" (dict "images" (list .Values.nodeAgent.containers.agent.image .Values.all.image) "context" $) | nindent 6 }} + {{- if .Values.all.hardening.enabled}} + terminationGracePeriodSeconds: 240 + {{- end }} + containers: + {{- include "container-agent" . | nindent 6 }} + {{- if .Values.nodeAgent.containers.processAgent.enabled }} + {{- include "container-process-agent" . | nindent 6 }} + {{- end }} + dnsPolicy: ClusterFirstWithHostNet + hostNetwork: true + hostPID: true + {{- if .Values.nodeAgent.priorityClassName }} + priorityClassName: {{ .Values.nodeAgent.priorityClassName }} + {{- end }} + serviceAccountName: {{ .Release.Name }}-node-agent + nodeSelector: + {{ template "label.os" . }}: {{ .Values.targetSystem }} + {{- with .Values.nodeAgent.nodeSelector }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeAgent.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeAgent.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + {{- if .Values.nodeAgent.containerRuntime.customSocketPath }} + - hostPath: + path: {{ .Values.nodeAgent.containerRuntime.customSocketPath }} + name: customcrisocket + {{- end }} + - hostPath: + path: /var/lib/kubelet + name: kubelet + - hostPath: + path: /var/lib/nfs + name: nfs + - hostPath: + path: /var/lib/docker/overlay2 + name: dockeroverlay2 + - hostPath: + path: /run/docker/netns + name: dockernetns + - hostPath: + path: /var/run/crio/crio.sock + name: crisocket + - hostPath: + path: /var/run/containerd/containerd.sock + name: containerdsocket + - hostPath: + path: /sys/kernel/debug + name: sys-kernel-debug + - hostPath: + path: /var/run/docker.sock + name: dockersocket + - hostPath: + path: {{ .Values.nodeAgent.containerRuntime.hostProc }} + name: procdir + - hostPath: + path: /etc + name: etcdir + - hostPath: + path: /etc/passwd + name: passwd + - hostPath: + path: /sys/fs/cgroup + name: cgroups + {{- if .Values.nodeAgent.config.override }} + - name: config-override-volume + configMap: + name: {{ .Release.Name }}-node-agent + {{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-podautoscaler.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-podautoscaler.yaml new file mode 100644 index 0000000000..38298d4147 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-podautoscaler.yaml @@ -0,0 +1,39 @@ +--- +{{- if .Values.nodeAgent.autoScalingEnabled }} +apiVersion: "autoscaling.k8s.io/v1" +kind: VerticalPodAutoscaler +metadata: + name: {{ .Release.Name }}-node-agent-vpa + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +spec: + targetRef: + apiVersion: "apps/v1" + kind: DaemonSet + name: {{ .Release.Name }}-node-agent + resourcePolicy: + containerPolicies: + - containerName: 'node-agent' + minAllowed: + cpu: {{ .Values.nodeAgent.scaling.autoscalerLimits.agent.minimum.cpu }} + memory: {{ .Values.nodeAgent.scaling.autoscalerLimits.agent.minimum.memory }} + maxAllowed: + cpu: {{ .Values.nodeAgent.scaling.autoscalerLimits.agent.maximum.cpu }} + memory: {{ .Values.nodeAgent.scaling.autoscalerLimits.agent.maximum.memory }} + controlledResources: ["cpu", "memory"] + controlledValues: RequestsAndLimits + - containerName: 'process-agent' + minAllowed: + cpu: {{ .Values.nodeAgent.scaling.autoscalerLimits.processAgent.minimum.cpu }} + memory: {{ .Values.nodeAgent.scaling.autoscalerLimits.processAgent.minimum.memory }} + maxAllowed: + cpu: {{ .Values.nodeAgent.scaling.autoscalerLimits.processAgent.maximum.cpu }} + memory: {{ .Values.nodeAgent.scaling.autoscalerLimits.processAgent.maximum.memory }} + controlledResources: ["cpu", "memory"] + controlledValues: RequestsAndLimits + updatePolicy: + updateMode: "Auto" +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-scc.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-scc.yaml new file mode 100644 index 0000000000..a09da78c17 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-scc.yaml @@ -0,0 +1,60 @@ +{{- if .Values.nodeAgent.scc.enabled }} +allowHostDirVolumePlugin: true +# was true +allowHostIPC: true +# was true +allowHostNetwork: true +# Allow host PID for dogstatsd origin detection +allowHostPID: true +# Allow host ports for dsd / trace / logs intake +allowHostPorts: true +allowPrivilegeEscalation: true +# was true +allowPrivilegedContainer: true +# was - '*' +allowedCapabilities: [] +allowedUnsafeSysctls: +- '*' +apiVersion: security.openshift.io/v1 +defaultAddCapabilities: null +fsGroup: +# was RunAsAny + type: MustRunAs +groups: [] +kind: SecurityContextConstraints +metadata: + name: {{ .Release.Name }}-node-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +priority: null +readOnlyRootFilesystem: false +requiredDropCapabilities: null +# was RunAsAny +runAsUser: + type: MustRunAsRange +# Use the `spc_t` selinux type to access the +# docker socket + proc and cgroup stats +seLinuxContext: + type: RunAsAny + seLinuxOptions: + user: "system_u" + role: "system_r" + type: "spc_t" + level: "s0" +# was - '*' +seccompProfiles: [] +supplementalGroups: + type: RunAsAny +users: +- system:serviceaccount:{{ .Release.Namespace }}:{{ .Release.Name }}-node-agent +# Allow hostPath for docker / process metrics +volumes: + - configMap + - downwardAPI + - emptyDir + - hostPath + - secret +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-service.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-service.yaml new file mode 100644 index 0000000000..0b6cd6ec03 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-service.yaml @@ -0,0 +1,28 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ .Release.Name }}-node-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: node-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +{{- with .Values.nodeAgent.service.annotations }} + {{- toYaml . | nindent 4 }} +{{- end }} +spec: + type: {{ .Values.nodeAgent.service.type }} +{{- if eq .Values.nodeAgent.service.type "LoadBalancer" }} + loadBalancerSourceRanges: {{ toYaml .Values.nodeAgent.service.loadBalancerSourceRanges | nindent 4}} +{{- end }} + ports: + - name: traceport + port: 8126 + protocol: TCP + targetPort: 8126 + selector: + app.kubernetes.io/component: node-agent + app.kubernetes.io/instance: {{ .Release.Name }} + app.kubernetes.io/name: {{ include "stackstate-k8s-agent.name" . }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-serviceaccount.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-serviceaccount.yaml new file mode 100644 index 0000000000..803d184ef7 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/node-agent-serviceaccount.yaml @@ -0,0 +1,14 @@ +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ .Release.Name }}-node-agent + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + app.kubernetes.io/component: node-agent + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +{{- with .Values.nodeAgent.serviceaccount.annotations }} + {{- toYaml . | nindent 4 }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/openshift-logging-secret.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/openshift-logging-secret.yaml new file mode 100644 index 0000000000..ed0707f1fa --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/openshift-logging-secret.yaml @@ -0,0 +1,22 @@ +{{- if not .Values.stackstate.manageOwnSecrets }} +{{- if .Values.openShiftLogging.installSecret }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }}-logging-secret + namespace: openshift-logging + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +type: Opaque +data: + username: {{ "apikey" | b64enc | quote }} +{{- if .Values.global.receiverApiKey }} + password: {{ .Values.global.receiverApiKey | b64enc | quote }} +{{- else }} + password: {{ .Values.stackstate.apiKey | b64enc | quote }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/pull-secret.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/pull-secret.yaml new file mode 100644 index 0000000000..9169416657 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/pull-secret.yaml @@ -0,0 +1,39 @@ +{{- $defaultRegistry := .Values.global.imageRegistry }} +{{- $top := . }} +{{- $registryAuthMap := dict }} + +{{- range $registry, $credentials := .Values.global.imagePullCredentials }} + {{- $registryAuthDocument := dict -}} + {{- $_ := set $registryAuthDocument "username" $credentials.username }} + {{- $_ := set $registryAuthDocument "password" $credentials.password }} + {{- $authMessage := printf "%s:%s" $registryAuthDocument.username $registryAuthDocument.password | b64enc }} + {{- $_ := set $registryAuthDocument "auth" $authMessage }} + {{- if eq $registry "default" }} + {{- $registryAuthMap := set $registryAuthMap (include "stackstate-k8s-agent.imageRegistry" $top) $registryAuthDocument }} + {{ else }} + {{- $registryAuthMap := set $registryAuthMap $registry $registryAuthDocument }} + {{- end }} +{{- end }} + +{{- if .Values.all.image.pullSecretUsername }} + {{- $registryAuthDocument := dict -}} + {{- $_ := set $registryAuthDocument "username" .Values.all.image.pullSecretUsername }} + {{- $_ := set $registryAuthDocument "password" .Values.all.image.pullSecretPassword }} + {{- $authMessage := printf "%s:%s" $registryAuthDocument.username $registryAuthDocument.password | b64enc }} + {{- $_ := set $registryAuthDocument "auth" $authMessage }} + {{- $registryAuthMap := set $registryAuthMap (include "stackstate-k8s-agent.imageRegistry" $top) $registryAuthDocument }} +{{- end }} + +{{- $dockerAuthsDocuments := dict "auths" $registryAuthMap }} + +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "stackstate-k8s-agent.pull-secret.name" . }} + labels: +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +data: + .dockerconfigjson: {{ $dockerAuthsDocuments | toJson | b64enc | quote }} +type: kubernetes.io/dockerconfigjson diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/secret.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/secret.yaml new file mode 100644 index 0000000000..5e0f5f74cd --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/templates/secret.yaml @@ -0,0 +1,27 @@ +{{- if not .Values.stackstate.manageOwnSecrets }} +apiVersion: v1 +kind: Secret +metadata: + name: {{ include "stackstate-k8s-agent.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: +{{ include "stackstate-k8s-agent.labels" . | indent 4 }} +{{ include "stackstate-k8s-agent.global.extraLabels" . | indent 4 }} + annotations: +{{ include "stackstate-k8s-agent.global.extraAnnotations" . | indent 4 }} +type: Opaque +data: +{{- if .Values.global.receiverApiKey }} + sts-api-key: {{ .Values.global.receiverApiKey | b64enc | quote }} +{{- else }} + sts-api-key: {{ .Values.stackstate.apiKey | b64enc | quote }} +{{- end }} +{{- if .Values.stackstate.cluster.authToken }} + sts-cluster-auth-token: {{ .Values.stackstate.cluster.authToken | b64enc | quote }} +{{- else }} + sts-cluster-auth-token: {{ randAlphaNum 32 | b64enc | quote }} +{{- end }} +{{- range $key, $value := .Values.global.extraEnv.secret }} + {{ $key }}: {{ $value | b64enc | quote }} +{{- end }} +{{- end }} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/clusteragent_resources_test.go b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/clusteragent_resources_test.go new file mode 100644 index 0000000000..25875e871a --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/clusteragent_resources_test.go @@ -0,0 +1,145 @@ +package test + +import ( + "regexp" + "strings" + "testing" + + v1 "k8s.io/api/rbac/v1" + + "github.com/stretchr/testify/assert" + "gitlab.com/StackVista/DevOps/helm-charts/helmtestutil" +) + +var requiredRules = []string{ + "events+get,list,watch", + "nodes+get,list,watch", + "pods+get,list,watch", + "services+get,list,watch", + "configmaps+create,get,patch,update", +} + +var optionalRules = []string{ + "namespaces+get,list,watch", + "componentstatuses+get,list,watch", + "configmaps+list,watch", // get is already required + "endpoints+get,list,watch", + "persistentvolumeclaims+get,list,watch", + "persistentvolumes+get,list,watch", + "secrets+get,list,watch", + "apps/daemonsets+get,list,watch", + "apps/deployments+get,list,watch", + "apps/replicasets+get,list,watch", + "apps/statefulsets+get,list,watch", + "extensions/ingresses+get,list,watch", + "batch/cronjobs+get,list,watch", + "batch/jobs+get,list,watch", +} + +var roleDescriptionRegexp = regexp.MustCompile(`^((?P\w+)/)?(?P\w+)\+(?P[\w,]+)`) + +type Rule struct { + Group string + ResourceName string + Verb string +} + +func assertRuleExistence(t *testing.T, rules []v1.PolicyRule, roleDescription string, shouldBePresent bool) { + match := roleDescriptionRegexp.FindStringSubmatch(roleDescription) + assert.NotNil(t, match) + + var roleRules []Rule + for _, rule := range rules { + for _, group := range rule.APIGroups { + for _, resource := range rule.Resources { + for _, verb := range rule.Verbs { + roleRules = append(roleRules, Rule{group, resource, verb}) + } + } + } + } + + resGroup := match[roleDescriptionRegexp.SubexpIndex("group")] + resName := match[roleDescriptionRegexp.SubexpIndex("name")] + verbs := strings.Split(match[roleDescriptionRegexp.SubexpIndex("verbs")], ",") + + for _, verb := range verbs { + requiredRule := Rule{resGroup, resName, verb} + found := false + for _, rule := range roleRules { + if rule == requiredRule { + found = true + break + } + } + if shouldBePresent { + assert.Truef(t, found, "Rule %v has not been found", requiredRule) + } else { + assert.Falsef(t, found, "Rule %v should not be present", requiredRule) + } + } +} + +func TestAllResourcesAreEnabled(t *testing.T) { + output := helmtestutil.RenderHelmTemplate(t, "stackstate-k8s-agent", "values/minimal.yaml") + resources := helmtestutil.NewKubernetesResources(t, output) + + assert.Contains(t, resources.ClusterRoles, "stackstate-k8s-agent") + assert.Contains(t, resources.Roles, "stackstate-k8s-agent") + rules := resources.ClusterRoles["stackstate-k8s-agent"].Rules + rules = append(rules, resources.Roles["stackstate-k8s-agent"].Rules...) + + for _, requiredRole := range requiredRules { + assertRuleExistence(t, rules, requiredRole, true) + } + // be default, everything is enabled, so all the optional roles should be present as well + for _, optionalRule := range optionalRules { + assertRuleExistence(t, rules, optionalRule, true) + } +} + +func TestMostOfResourcesAreDisabled(t *testing.T) { + output := helmtestutil.RenderHelmTemplate(t, "stackstate-k8s-agent", "values/minimal.yaml", "values/disable-all-resource.yaml") + resources := helmtestutil.NewKubernetesResources(t, output) + + assert.Contains(t, resources.ClusterRoles, "stackstate-k8s-agent") + assert.Contains(t, resources.Roles, "stackstate-k8s-agent") + rules := resources.ClusterRoles["stackstate-k8s-agent"].Rules + rules = append(rules, resources.Roles["stackstate-k8s-agent"].Rules...) + + for _, requiredRole := range requiredRules { + assertRuleExistence(t, rules, requiredRole, true) + } + + // we expect all optional resources to be removed from ClusterRole with the given values + for _, optionalRule := range optionalRules { + assertRuleExistence(t, rules, optionalRule, false) + } +} + +func TestNoClusterWideModificationRights(t *testing.T) { + output := helmtestutil.RenderHelmTemplate(t, "stackstate-k8s-agent", "values/minimal.yaml", "values/http-header-injector.yaml") + resources := helmtestutil.NewKubernetesResources(t, output) + assert.Contains(t, resources.ClusterRoles, "stackstate-k8s-agent") + illegalVerbs := []string{"create", "patch", "update", "delete"} + + for _, clusterRole := range resources.ClusterRoles { + for _, rule := range clusterRole.Rules { + for _, verb := range rule.Verbs { + assert.NotContains(t, illegalVerbs, verb, "ClusterRole %s should not have %s verb for %s resource", clusterRole.Name, verb, rule.Resources) + } + } + } +} + +func TestServicePortChange(t *testing.T) { + output := helmtestutil.RenderHelmTemplate(t, "stackstate-k8s-agent", "values/minimal.yaml", "values/clustercheck_service_port_override.yaml") + resources := helmtestutil.NewKubernetesResources(t, output) + + cluster_agent_service := resources.Services["stackstate-k8s-agent-cluster-agent"] + + port := cluster_agent_service.Spec.Ports[0] + assert.Equal(t, port.Name, "clusteragent") + assert.Equal(t, port.Port, int32(8008)) + assert.Equal(t, port.TargetPort.IntVal, int32(9009)) +} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/clustername_test.go b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/clustername_test.go new file mode 100644 index 0000000000..55090b9956 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/clustername_test.go @@ -0,0 +1,54 @@ +package test + +import ( + "testing" + + "github.com/gruntwork-io/terratest/modules/helm" + "github.com/stretchr/testify/assert" + + "gitlab.com/StackVista/DevOps/helm-charts/helmtestutil" +) + +func TestHelmBasicRender(t *testing.T) { + output := helmtestutil.RenderHelmTemplate(t, "stackstate-k8s-agent", "values/minimal.yaml") + + // Parse all resources into their corresponding types for validation and further inspection + helmtestutil.NewKubernetesResources(t, output) +} + +func TestClusterNameValidation(t *testing.T) { + testCases := []struct { + Name string + ClusterName string + IsValid bool + }{ + {"not allowed end with special character [.]", "name.", false}, + {"not allowed end with special character [-]", "name.", false}, + {"not allowed start with special character [-]", "-name", false}, + {"not allowed start with special character [.]", ".name", false}, + {"upper case is not allowed", "Euwest1-prod.cool-company.com", false}, + {"upper case is not allowed", "euwest1-PROD.cool-company.com", false}, + {"upper case is not allowed", "euwest1-prod.cool-company.coM", false}, + {"dots and dashes are allowed in the middle", "euwest1-prod.cool-company.com", true}, + {"underscore is not allowed", "why_7", false}, + } + + for _, testCase := range testCases { + t.Run(testCase.Name, func(t *testing.T) { + output, err := helmtestutil.RenderHelmTemplateOpts( + t, "cluster-agent", + &helm.Options{ + ValuesFiles: []string{"values/minimal.yaml"}, + SetStrValues: map[string]string{ + "stackstate.cluster.name": testCase.ClusterName, + }, + }) + if testCase.IsValid { + assert.Nil(t, err) + } else { + assert.NotNil(t, err) + assert.Contains(t, output, "stackstate.cluster.name: Does not match pattern") + } + }) + } +} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_custom_url.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_custom_url.yaml new file mode 100644 index 0000000000..57b973eed9 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_custom_url.yaml @@ -0,0 +1,7 @@ +checksAgent: + enabled: true + kubeStateMetrics: + url: http://my-custom-ksm-url.monitoring.svc.local:8080/metrics +dependencies: + kubeStateMetrics: + enabled: true diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_no_override.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_no_override.yaml new file mode 100644 index 0000000000..b6c817d473 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_no_override.yaml @@ -0,0 +1,5 @@ +checksAgent: + enabled: true +dependencies: + kubeStateMetrics: + enabled: true diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_override.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_override.yaml new file mode 100644 index 0000000000..9ca201345a --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_ksm_override.yaml @@ -0,0 +1,26 @@ +checksAgent: + enabled: true +dependencies: + kubeStateMetrics: + enabled: true +agent: + config: + override: +# agent.config.override -- Disables kubernetes_state check on regular agent pods. + - name: auto_conf.yaml + path: /etc/stackstate-agent/conf.d/kubernetes_state.d + data: | +clusterAgent: + config: + override: +# clusterAgent.config.override -- Defines kubernetes_state check for clusterchecks agents. Auto-discovery +# with ad_identifiers does not work here. Use a specific URL instead. + - name: conf.yaml + path: /etc/stackstate-agent/conf.d/kubernetes_state.d + data: | + cluster_check: true + + init_config: + + instances: + - kube_state_url: http://YOUR_KUBE_STATE_METRICS_SERVICE_NAME:8080/metrics diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_no_ksm_custom_url.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_no_ksm_custom_url.yaml new file mode 100644 index 0000000000..a62691878c --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_no_ksm_custom_url.yaml @@ -0,0 +1,7 @@ +checksAgent: + enabled: true + kubeStateMetrics: + url: http://my-custom-ksm-url.monitoring.svc.local:8080/metrics +dependencies: + kubeStateMetrics: + enabled: false diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_service_port_override.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_service_port_override.yaml new file mode 100644 index 0000000000..c01a98fcb4 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/clustercheck_service_port_override.yaml @@ -0,0 +1,4 @@ +clusterAgent: + service: + port: 8008 + targetPort: 9009 diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/disable-all-resource.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/disable-all-resource.yaml new file mode 100644 index 0000000000..cd33e843ea --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/disable-all-resource.yaml @@ -0,0 +1,17 @@ +clusterAgent: + collection: + kubernetesMetrics: false + kubernetesResources: + namespaces: false + configmaps: false + endpoints: false + persistentvolumes: false + persistentvolumeclaims: false + secrets: false + daemonsets: false + deployments: false + replicasets: false + statefulsets: false + ingresses: false + cronjobs: false + jobs: false diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/http-header-injector.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/http-header-injector.yaml new file mode 100644 index 0000000000..c9392ce2dd --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/http-header-injector.yaml @@ -0,0 +1,8 @@ +httpHeaderInjectorWebhook: + webhook: + tls: + mode: "provided" + provided: + caBundle: insert-ca-here + crt: insert-cert-here + key: insert-key-here diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/minimal.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/minimal.yaml new file mode 100644 index 0000000000..b310c9a093 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/test/values/minimal.yaml @@ -0,0 +1,7 @@ +stackstate: + apiKey: foobar + cluster: + name: some-k8s-cluster + token: some-token + + url: https://stackstate:7000/receiver diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/values.schema.json b/charts/stackstate/stackstate-k8s-agent/1.0.98/values.schema.json new file mode 100644 index 0000000000..57d36a9f34 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/values.schema.json @@ -0,0 +1,78 @@ +{ + "$schema": "https://json-schema.org/draft/2019-09/schema", + "$id": "https://stackstate.io/example.json", + "type": "object", + "default": {}, + "title": "StackState Agent Helm chart values", + "required": [ + "stackstate", + "clusterAgent" + ], + "properties": { + "stackstate": { + "type": "object", + "required": [ + "cluster", + "url" + ], + "properties": { + "apiKey": { + "type": "string" + }, + "cluster": { + "type": "object", + "required": ["name"], + "properties": { + "name": { + "type": "string", + "pattern": "^[a-z0-9]([a-z0-9\\-\\.]*[a-z0-9])$" + }, + "authToken": { + "type": "string" + } + } + }, + "url": { + "type": "string" + } + } + }, + "clusterAgent": { + "type": "object", + "required": [ + "config" + ], + "properties": { + "config": { + "type": "object", + "required": [ + "events" + ], + "properties": { + "events": { + "type": "object", + "properties": { + "categories": { + "type": "object", + "patternProperties": { + ".*": { + "type": [ + "string" + ], + "enum": [ + "Alerts", + "Activities", + "Changes", + "Others" + ] + } + } + } + } + } + } + } + } + } + } +} diff --git a/charts/stackstate/stackstate-k8s-agent/1.0.98/values.yaml b/charts/stackstate/stackstate-k8s-agent/1.0.98/values.yaml new file mode 100644 index 0000000000..c58846a7c0 --- /dev/null +++ b/charts/stackstate/stackstate-k8s-agent/1.0.98/values.yaml @@ -0,0 +1,616 @@ +##################### +# General variables # +##################### + +global: + extraEnv: + # global.extraEnv.open -- Extra open environment variables to inject into pods. + open: {} + # global.extraEnv.secret -- Extra secret environment variables to inject into pods via a `Secret` object. + secret: {} + # global.imagePullSecrets -- Secrets / credentials needed for container image registry. + imagePullSecrets: [] + # global.imagePullCredentials -- Globally define credentials for pulling images. + imagePullCredentials: {} + proxy: + # global.proxy.url -- Proxy for all traffic to stackstate + url: "" + # global.skipSslValidation -- Enable tls validation from client + skipSslValidation: false + + # global.extraLabels -- Extra labels added ta all resources created by the helm chart + extraLabels: {} + # global.extraAnnotations -- Extra annotations added ta all resources created by the helm chart + extraAnnotations: {} + +# nameOverride -- Override the name of the chart. +nameOverride: "" +# fullnameOverride -- Override the fullname of the chart. +fullnameOverride: "" + +# targetSystem -- Target OS for this deployment (possible values: linux) +targetSystem: "linux" + +all: + image: + # all.image.registry -- The image registry to use. + registry: "quay.io" + hardening: + # all.hardening.enabled -- An indication of whether the containers will be evaluated for hardening at runtime + enabled: false + +nodeAgent: + # nodeAgent.autoScalingEnabled -- Enable / disable autoscaling for the node agent pods. + autoScalingEnabled: false + containerRuntime: + # nodeAgent.containerRuntime.customSocketPath -- If the container socket path does not match the default for CRI-O, Containerd or Docker, supply a custom socket path. + customSocketPath: "" + # nodeAgent.containerRuntime.customHostProc -- If the container is launched from a place where /proc is mounted differently, /proc can be changed + hostProc: /proc + + scc: + # nodeAgent.scc.enabled -- Enable / disable the installation of the SecurityContextConfiguration needed for installation on OpenShift. + enabled: false + apm: + # nodeAgent.apm.enabled -- Enable / disable the nodeAgent APM module. + enabled: true + networkTracing: + # nodeAgent.networkTracing.enabled -- Enable / disable the nodeAgent network tracing module. + enabled: true + protocolInspection: + # nodeAgent.protocolInspection.enabled -- Enable / disable the nodeAgent protocol inspection. + enabled: true + httpTracing: + enabled: true + # nodeAgent.skipSslValidation -- Set to true if self signed certificates are used. + skipSslValidation: false + # nodeAgent.skipKubeletTLSVerify -- Set to true if you want to skip kubelet tls verification. + skipKubeletTLSVerify: false + + # nodeAgent.checksTagCardinality -- low, orchestrator or high. Orchestrator level adds pod_name, high adds display_container_name + checksTagCardinality: orchestrator + + # nodeAgent.config -- + config: + # nodeAgent.config.override -- A list of objects containing three keys `name`, `path` and `data`, specifying filenames at specific paths which need to be (potentially) overridden using a mounted configmap + override: [] + + # nodeAgent.priorityClassName -- Priority class for nodeAgent pods. + priorityClassName: "" + + scaling: + autoscalerLimits: + agent: + minimum: + # nodeAgent.scaling.autoscalerLimits.agent.minimum.cpu -- Minimum CPU resource limits for main agent. + cpu: "20m" + # nodeAgent.scaling.autoscalerLimits.agent.minimum.memory -- Minimum memory resource limits for main agent. + memory: "180Mi" + maximum: + # nodeAgent.scaling.autoscalerLimits.agent.maximum.cpu -- Maximum CPU resource limits for main agent. + cpu: "200m" + # nodeAgent.scaling.autoscalerLimits.agent.maximum.memory -- Maximum memory resource limits for main agent. + memory: "450Mi" + processAgent: + minimum: + # nodeAgent.scaling.autoscalerLimits.processAgent.minimum.cpu -- Minimum CPU resource limits for process agent. + cpu: "25m" + # nodeAgent.scaling.autoscalerLimits.processAgent.minimum.memory -- Minimum memory resource limits for process agent. + memory: "100Mi" + maximum: + # nodeAgent.scaling.autoscalerLimits.processAgent.maximum.cpu -- Maximum CPU resource limits for process agent. + cpu: "200m" + # nodeAgent.scaling.autoscalerLimits.processAgent.maximum.memory -- Maximum memory resource limits for process agent. + memory: "500Mi" + + containers: + agent: + image: + # nodeAgent.containers.agent.image.repository -- Base container image repository. + repository: stackstate/stackstate-k8s-agent + # nodeAgent.containers.agent.image.tag -- Default container image tag. + tag: "c4caacef" + # nodeAgent.containers.agent.image.pullPolicy -- Default container image pull policy. + pullPolicy: IfNotPresent + processAgent: + # nodeAgent.containers.agent.processAgent.enabled -- Enable / disable the agent process agent module. - deprecated + enabled: false + # nodeAgent.containers.agent.env -- Additional environment variables for the agent container + env: {} + # nodeAgent.containers.agent.logLevel -- Set logging verbosity, valid log levels are: trace, debug, info, warn, error, critical, and off + ## If not set, fall back to the value of agent.logLevel. + logLevel: # INFO + + resources: + limits: + # nodeAgent.containers.agent.resources.limits.cpu -- CPU resource limits. + cpu: "270m" + # nodeAgent.containers.agent.resources.limits.memory -- Memory resource limits. + memory: "420Mi" + requests: + # nodeAgent.containers.agent.resources.requests.cpu -- CPU resource requests. + cpu: "20m" + # nodeAgent.containers.agent.resources.requests.memory -- Memory resource requests. + memory: "180Mi" + livenessProbe: + # nodeAgent.containers.agent.livenessProbe.enabled -- Enable use of livenessProbe check. + enabled: true + # nodeAgent.containers.agent.livenessProbe.failureThreshold -- `failureThreshold` for the liveness probe. + failureThreshold: 3 + # nodeAgent.containers.agent.livenessProbe.initialDelaySeconds -- `initialDelaySeconds` for the liveness probe. + initialDelaySeconds: 15 + # nodeAgent.containers.agent.livenessProbe.periodSeconds -- `periodSeconds` for the liveness probe. + periodSeconds: 15 + # nodeAgent.containers.agent.livenessProbe.successThreshold -- `successThreshold` for the liveness probe. + successThreshold: 1 + # nodeAgent.containers.agent.livenessProbe.timeoutSeconds -- `timeoutSeconds` for the liveness probe. + timeoutSeconds: 5 + readinessProbe: + # nodeAgent.containers.agent.readinessProbe.enabled -- Enable use of readinessProbe check. + enabled: true + # nodeAgent.containers.agent.readinessProbe.failureThreshold -- `failureThreshold` for the readiness probe. + failureThreshold: 3 + # nodeAgent.containers.agent.readinessProbe.initialDelaySeconds -- `initialDelaySeconds` for the readiness probe. + initialDelaySeconds: 15 + # nodeAgent.containers.agent.readinessProbe.periodSeconds -- `periodSeconds` for the readiness probe. + periodSeconds: 15 + # nodeAgent.containers.agent.readinessProbe.successThreshold -- `successThreshold` for the readiness probe. + successThreshold: 1 + # nodeAgent.containers.agent.readinessProbe.timeoutSeconds -- `timeoutSeconds` for the readiness probe. + timeoutSeconds: 5 + + processAgent: + # nodeAgent.containers.processAgent.enabled -- Enable / disable the process agent container. + enabled: true + image: + # Override to pull the image from an alternate registry + registry: + # nodeAgent.containers.processAgent.image.repository -- Process-agent container image repository. + repository: stackstate/stackstate-k8s-process-agent + # nodeAgent.containers.processAgent.image.tag -- Default process-agent container image tag. + tag: "cae7a4fa" + # nodeAgent.containers.processAgent.image.pullPolicy -- Process-agent container image pull policy. + pullPolicy: IfNotPresent + # nodeAgent.containers.processAgent.env -- Additional environment variables for the process-agent container + env: {} + # nodeAgent.containers.processAgent.logLevel -- Set logging verbosity, valid log levels are: trace, debug, info, warn, error, critical, and off + ## If not set, fall back to the value of agent.logLevel. + logLevel: # INFO + + # nodeAgent.containers.processAgent.procVolumeReadOnly -- Configure whether /host/proc is read only for the process agent container + procVolumeReadOnly: true + + resources: + limits: + # nodeAgent.containers.processAgent.resources.limits.cpu -- CPU resource limits. + cpu: "125m" + # nodeAgent.containers.processAgent.resources.limits.memory -- Memory resource limits. + memory: "400Mi" + requests: + # nodeAgent.containers.processAgent.resources.requests.cpu -- CPU resource requests. + cpu: "25m" + # nodeAgent.containers.processAgent.resources.requests.memory -- Memory resource requests. + memory: "128Mi" + # nodeAgent.service -- The Kubernetes service for the agent + service: + # nodeAgent.service.type -- Type of Kubernetes service: ClusterIP, LoadBalancer, NodePort + type: ClusterIP + # nodeAgent.service.annotations -- Annotations for the service + annotations: {} + # nodeAgent.service.loadBalancerSourceRanges -- The IP4 CIDR allowed to reach LoadBalancer for the service. For LoadBalancer type of service only. + loadBalancerSourceRanges: ["10.0.0.0/8"] + + # nodeAgent.logLevel -- Logging level for agent processes. + logLevel: INFO + + # nodeAgent.updateStrategy -- The update strategy for the DaemonSet object. + updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 100 + + # nodeAgent.nodeSelector -- Node labels for pod assignment. + nodeSelector: {} + + # nodeAgent.tolerations -- Toleration labels for pod assignment. + tolerations: [] + + # nodeAgent.affinity -- Affinity settings for pod assignment. + affinity: {} + + serviceaccount: + # nodeAgent.serviceaccount.annotations -- Annotations for the service account for the agent daemonset pods + annotations: {} + +processAgent: + softMemoryLimit: + # processAgent.softMemoryLimit.goMemLimit -- Soft-limit for golang heap allocation, for sanity, must be around 85% of nodeAgent.containers.processAgent.resources.limits.cpu. + goMemLimit: 340MiB + # processAgent.softMemoryLimit.httpStatsBufferSize -- Sets a maximum for the number of http stats to keep in memory between check runs, to use 40k requires around ~400Mib of memory. + httpStatsBufferSize: 40000 + # processAgent.softMemoryLimit.httpObservationsBufferSize -- Sets a maximum for the number of http observations to keep in memory between check runs, to use 40k requires around ~400Mib of memory. + httpObservationsBufferSize: 40000 + + checkIntervals: + # processAgent.checkIntervals.container -- Override the default value of the container check interval in seconds. + container: 28 + # processAgent.checkIntervals.connections -- Override the default value of the connections check interval in seconds. + connections: 30 + # processAgent.checkIntervals.process -- Override the default value of the process check interval in seconds. + process: 32 + +clusterAgent: + collection: + # clusterAgent.collection.kubernetesEvents -- Enable / disable the cluster agent events collection. + kubernetesEvents: true + # clusterAgent.collection.kubernetesMetrics -- Enable / disable the cluster agent metrics collection. + kubernetesMetrics: true + # clusterAgent.collection.kubernetesTimeout -- Default timeout (in seconds) when obtaining information from the Kubernetes API. + kubernetesTimeout: 10 + # clusterAgent.collection.kubernetesTopology -- Enable / disable the cluster agent topology collection. + kubernetesTopology: true + kubeStateMetrics: + # clusterAgent.collection.kubeStateMetrics.enabled -- Enable / disable the cluster agent kube-state-metrics collection. + enabled: true + # clusterAgent.collection.kubeStateMetrics.clusterCheck -- For large clusters where the Kubernetes State Metrics Check Core needs to be distributed on dedicated workers. + clusterCheck: false + # clusterAgent.collection.kubeStateMetrics.labelsAsTags -- Extra labels to collect from resources and to turn into StackState tag. + ## It has the following structure: + ## labelsAsTags: + ## : # can be pod, deployment, node, etc. + ## : # where is the kubernetes label and is the StackState tag + ## : + ## : + ## : + ## + ## Warning: the label must match the transformation done by kube-state-metrics, + ## for example tags.stackstate/version becomes tags_stackstate_version. + labelsAsTags: {} + # pod: + # app: app + # node: + # zone: zone + # team: team + + # clusterAgent.collection.kubeStateMetrics.annotationsAsTags -- Extra annotations to collect from resources and to turn into StackState tag. + + ## It has the following structure: + ## annotationsAsTags: + ## : # can be pod, deployment, node, etc. + ## : # where is the kubernetes annotation and is the StackState tag + ## : + ## : + ## : + ## + ## Warning: the annotation must match the transformation done by kube-state-metrics, + ## for example tags.stackstate/version becomes tags_stackstate_version. + annotationsAsTags: {} + kubernetesResources: + # clusterAgent.collection.kubernetesResources.limitranges -- Enable / disable collection of LimitRanges. + limitranges: true + # clusterAgent.collection.kubernetesResources.horizontalpodautoscalers -- Enable / disable collection of HorizontalPodAutoscalers. + horizontalpodautoscalers: true + # clusterAgent.collection.kubernetesResources.replicationcontrollers -- Enable / disable collection of ReplicationControllers. + replicationcontrollers: true + # clusterAgent.collection.kubernetesResources.poddisruptionbudgets -- Enable / disable collection of PodDisruptionBudgets. + poddisruptionbudgets: true + # clusterAgent.collection.kubernetesResources.storageclasses -- Enable / disable collection of StorageClasses. + storageclasses: true + # clusterAgent.collection.kubernetesResources.volumeattachments -- Enable / disable collection of Volume Attachments. Used to bind Nodes to Persistent Volumes. + volumeattachments: true + # clusterAgent.collection.kubernetesResources.namespaces -- Enable / disable collection of Namespaces. + namespaces: true + # clusterAgent.collection.kubernetesResources.configmaps -- Enable / disable collection of ConfigMaps. + configmaps: true + # clusterAgent.collection.kubernetesResources.endpoints -- Enable / disable collection of Endpoints. If endpoints are disabled then StackState won't be able to connect a Service to Pods that serving it + endpoints: true + # clusterAgent.collection.kubernetesResources.persistentvolumes -- Enable / disable collection of PersistentVolumes. + persistentvolumes: true + # clusterAgent.collection.kubernetesResources.persistentvolumeclaims -- Enable / disable collection of PersistentVolumeClaims. Disabling these will not let StackState connect PersistentVolumes to pods they are attached to + persistentvolumeclaims: true + # clusterAgent.collection.kubernetesResources.secrets -- Enable / disable collection of Secrets. + secrets: true + # clusterAgent.collection.kubernetesResources.daemonsets -- Enable / disable collection of DaemonSets. + daemonsets: true + # clusterAgent.collection.kubernetesResources.deployments -- Enable / disable collection of Deployments. + deployments: true + # clusterAgent.collection.kubernetesResources.replicasets -- Enable / disable collection of ReplicaSets. + replicasets: true + # clusterAgent.collection.kubernetesResources.statefulsets -- Enable / disable collection of StatefulSets. + statefulsets: true + # clusterAgent.collection.kubernetesResources.ingresses -- Enable / disable collection of Ingresses. + ingresses: true + # clusterAgent.collection.kubernetesResources.cronjobs -- Enable / disable collection of CronJobs. + cronjobs: true + # clusterAgent.collection.kubernetesResources.jobs -- Enable / disable collection of Jobs. + jobs: true + # clusterAgent.collection.kubernetesResources.resourcequotas -- Enable / disable collection of ResourceQuotas. + resourcequotas: true + + # clusterAgent.config -- + config: + events: + # clusterAgent.config.events.categories -- Custom mapping from Kubernetes event reason to StackState event category. Categories allowed: Alerts, Activities, Changes, Others + categories: {} + topology: + # clusterAgent.config.topology.collectionInterval -- Interval for running topology collection, in seconds + collectionInterval: 90 + configMap: + # clusterAgent.config.configMap.maxDataSize -- Maximum amount of characters for the data property of a ConfigMap collected by the kubernetes topology check + maxDataSize: + # clusterAgent.config.override -- A list of objects containing three keys `name`, `path` and `data`, specifying filenames at specific paths which need to be (potentially) overridden using a mounted configmap + override: [] + + service: + # clusterAgent.service.port -- Change the Cluster Agent service port + port: 5005 + # clusterAgent.service.targetPort -- Change the Cluster Agent service targetPort + targetPort: 5005 + + # clusterAgent.enabled -- Enable / disable the cluster agent. + enabled: true + + # clusterAgent.skipSslValidation -- If true, ignores the server certificate being signed by an unknown authority. + skipSslValidation: false + + image: + # clusterAgent.image.repository -- Base container image repository. + repository: stackstate/stackstate-k8s-cluster-agent + # clusterAgent.image.tag -- Default container image tag. + tag: "c4caacef" + # clusterAgent.image.pullPolicy -- Default container image pull policy. + pullPolicy: IfNotPresent + + livenessProbe: + # clusterAgent.livenessProbe.enabled -- Enable use of livenessProbe check. + enabled: true + # clusterAgent.livenessProbe.failureThreshold -- `failureThreshold` for the liveness probe. + failureThreshold: 3 + # clusterAgent.livenessProbe.initialDelaySeconds -- `initialDelaySeconds` for the liveness probe. + initialDelaySeconds: 15 + # clusterAgent.livenessProbe.periodSeconds -- `periodSeconds` for the liveness probe. + periodSeconds: 15 + # clusterAgent.livenessProbe.successThreshold -- `successThreshold` for the liveness probe. + successThreshold: 1 + # clusterAgent.livenessProbe.timeoutSeconds -- `timeoutSeconds` for the liveness probe. + timeoutSeconds: 5 + + # clusterAgent.logLevel -- Logging level for stackstate-k8s-agent processes. + logLevel: INFO + + # clusterAgent.priorityClassName -- Priority class for stackstate-k8s-agent pods. + priorityClassName: "" + + readinessProbe: + # clusterAgent.readinessProbe.enabled -- Enable use of readinessProbe check. + enabled: true + # clusterAgent.readinessProbe.failureThreshold -- `failureThreshold` for the readiness probe. + failureThreshold: 3 + # clusterAgent.readinessProbe.initialDelaySeconds -- `initialDelaySeconds` for the readiness probe. + initialDelaySeconds: 15 + # clusterAgent.readinessProbe.periodSeconds -- `periodSeconds` for the readiness probe. + periodSeconds: 15 + # clusterAgent.readinessProbe.successThreshold -- `successThreshold` for the readiness probe. + successThreshold: 1 + # clusterAgent.readinessProbe.timeoutSeconds -- `timeoutSeconds` for the readiness probe. + timeoutSeconds: 5 + + # clusterAgent.replicaCount -- Number of replicas of the cluster agent to deploy. + replicaCount: 1 + + serviceaccount: + # clusterAgent.serviceaccount.annotations -- Annotations for the service account for the cluster agent pods + annotations: {} + + # clusterAgent.strategy -- The strategy for the Deployment object. + strategy: + type: RollingUpdate + # rollingUpdate: + # maxUnavailable: 1 + + resources: + limits: + # clusterAgent.resources.limits.cpu -- CPU resource limits. + cpu: "400m" + # clusterAgent.resources.limits.memory -- Memory resource limits. + memory: "800Mi" + requests: + # clusterAgent.resources.requests.cpu -- CPU resource requests. + cpu: "70m" + # clusterAgent.resources.requests.memory -- Memory resource requests. + memory: "512Mi" + + # clusterAgent.nodeSelector -- Node labels for pod assignment. + nodeSelector: {} + + # clusterAgent.tolerations -- Toleration labels for pod assignment. + tolerations: [] + + # clusterAgent.affinity -- Affinity settings for pod assignment. + affinity: {} + +openShiftLogging: + # openShiftLogging.installSecret -- Install a secret for logging on openshift + installSecret: false + +logsAgent: + # logsAgent.enabled -- Enable / disable k8s pod log collection + enabled: true + + # logsAgent.skipSslValidation -- If true, ignores the server certificate being signed by an unknown authority. + skipSslValidation: false + + # logsAgent.priorityClassName -- Priority class for logsAgent pods. + priorityClassName: "" + + image: + # logsAgent.image.repository -- Base container image repository. + repository: stackstate/promtail + # logsAgent.image.tag -- Default container image tag. + tag: 2.9.8-5b179aee + # logsAgent.image.pullPolicy -- Default container image pull policy. + pullPolicy: IfNotPresent + + resources: + limits: + # logsAgent.resources.limits.cpu -- CPU resource limits. + cpu: "1300m" + # logsAgent.resources.limits.memory -- Memory resource limits. + memory: "192Mi" + requests: + # logsAgent.resources.requests.cpu -- CPU resource requests. + cpu: "20m" + # logsAgent.resources.requests.memory -- Memory resource requests. + memory: "100Mi" + + # logsAgent.updateStrategy -- The update strategy for the DaemonSet object. + updateStrategy: + type: RollingUpdate + rollingUpdate: + maxUnavailable: 100 + + # logsAgent.nodeSelector -- Node labels for pod assignment. + nodeSelector: {} + + # logsAgent.tolerations -- Toleration labels for pod assignment. + tolerations: [] + + # logsAgent.affinity -- Affinity settings for pod assignment. + affinity: {} + + serviceaccount: + # logsAgent.serviceaccount.annotations -- Annotations for the service account for the daemonset pods + annotations: {} + +checksAgent: + # checksAgent.enabled -- Enable / disable runnning cluster checks in a separately deployed pod + enabled: true + scc: + # checksAgent.scc.enabled -- Enable / disable the installation of the SecurityContextConfiguration needed for installation on OpenShift + enabled: false + apm: + # checksAgent.apm.enabled -- Enable / disable the agent APM module. + enabled: true + networkTracing: + # checksAgent.networkTracing.enabled -- Enable / disable the agent network tracing module. + enabled: true + processAgent: + # checksAgent.processAgent.enabled -- Enable / disable the agent process agent module. + enabled: true + # checksAgent.skipSslValidation -- Set to true if self signed certificates are used. + skipSslValidation: false + + # nodeAgent.checksTagCardinality -- low, orchestrator or high. Orchestrator level adds pod_name, high adds display_container_name + checksTagCardinality: orchestrator + + # checksAgent.config -- + config: + # checksAgent.config.override -- A list of objects containing three keys `name`, `path` and `data`, specifying filenames at specific paths which need to be (potentially) overridden using a mounted configmap + override: [] + + image: + # checksAgent.image.repository -- Base container image repository. + repository: stackstate/stackstate-k8s-agent + # checksAgent.image.tag -- Default container image tag. + tag: "c4caacef" + # checksAgent.image.pullPolicy -- Default container image pull policy. + pullPolicy: IfNotPresent + + livenessProbe: + # checksAgent.livenessProbe.enabled -- Enable use of livenessProbe check. + enabled: true + # checksAgent.livenessProbe.failureThreshold -- `failureThreshold` for the liveness probe. + failureThreshold: 3 + # checksAgent.livenessProbe.initialDelaySeconds -- `initialDelaySeconds` for the liveness probe. + initialDelaySeconds: 15 + # checksAgent.livenessProbe.periodSeconds -- `periodSeconds` for the liveness probe. + periodSeconds: 15 + # checksAgent.livenessProbe.successThreshold -- `successThreshold` for the liveness probe. + successThreshold: 1 + # checksAgent.livenessProbe.timeoutSeconds -- `timeoutSeconds` for the liveness probe. + timeoutSeconds: 5 + + # checksAgent.logLevel -- Logging level for clusterchecks agent processes. + logLevel: INFO + + # checksAgent.priorityClassName -- Priority class for clusterchecks agent pods. + priorityClassName: "" + + readinessProbe: + # checksAgent.readinessProbe.enabled -- Enable use of readinessProbe check. + enabled: true + # checksAgent.readinessProbe.failureThreshold -- `failureThreshold` for the readiness probe. + failureThreshold: 3 + # checksAgent.readinessProbe.initialDelaySeconds -- `initialDelaySeconds` for the readiness probe. + initialDelaySeconds: 15 + # checksAgent.readinessProbe.periodSeconds -- `periodSeconds` for the readiness probe. + periodSeconds: 15 + # checksAgent.readinessProbe.successThreshold -- `successThreshold` for the readiness probe. + successThreshold: 1 + # checksAgent.readinessProbe.timeoutSeconds -- `timeoutSeconds` for the readiness probe. + timeoutSeconds: 5 + + # checksAgent.replicas -- Number of clusterchecks agent pods to schedule + replicas: 1 + + resources: + limits: + # checksAgent.resources.limits.cpu -- CPU resource limits. + cpu: "400m" + # checksAgent.resources.limits.memory -- Memory resource limits. + memory: "600Mi" + requests: + # checksAgent.resources.requests.cpu -- CPU resource requests. + cpu: "20m" + # checksAgent.resources.requests.memory -- Memory resource requests. + memory: "512Mi" + + serviceaccount: + # checksAgent.serviceaccount.annotations -- Annotations for the service account for the cluster checks pods + annotations: {} + + # checksAgent.strategy -- The strategy for the Deployment object. + strategy: + type: RollingUpdate + # rollingUpdate: + # maxUnavailable: 1 + + # checksAgent.nodeSelector -- Node labels for pod assignment. + nodeSelector: {} + + # checksAgent.tolerations -- Toleration labels for pod assignment. + tolerations: [] + + # checksAgent.affinity -- Affinity settings for pod assignment. + affinity: {} + +################################## +# http-header-injector variables # +################################## + +httpHeaderInjectorWebhook: + # httpHeaderInjectorWebhook.enabled -- Enable the webhook for injection http header injection sidecar proxy + enabled: false + +######################## +# StackState variables # +######################## + +stackstate: + # stackstate.manageOwnSecrets -- Set to true if you don't want this helm chart to create secrets for you. + manageOwnSecrets: false + # stackstate.customSecretName -- Name of the secret containing the receiver API key. + customSecretName: "" + # stackstate.customApiKeySecretKey -- Key in the secret containing the receiver API key. + customApiKeySecretKey: "sts-api-key" + # stackstate.customClusterAuthTokenSecretKey -- Key in the secret containing the cluster auth token. + customClusterAuthTokenSecretKey: "sts-cluster-auth-token" + # stackstate.apiKey -- (string) **PROVIDE YOUR API KEY HERE** API key to be used by the StackState agent. + apiKey: + cluster: + # stackstate.cluster.name -- (string) **PROVIDE KUBERNETES CLUSTER NAME HERE** Name of the Kubernetes cluster where the agent will be installed. + name: + # stackstate.cluster.authToken -- Provide a token to enable secure communication between the agent and the cluster agent. + authToken: "" + # stackstate.url -- (string) **PROVIDE STACKSTATE URL HERE** URL of the StackState installation to receive data from the agent. + url: diff --git a/index.yaml b/index.yaml index 8120aeb953..48caaec708 100644 --- a/index.yaml +++ b/index.yaml @@ -241,6 +241,40 @@ entries: - assets/amd/amd-gpu-0.9.0.tgz version: 0.9.0 artifactory-ha: + - annotations: + artifactoryServiceVersion: 7.90.20 + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: JFrog Artifactory HA + catalog.cattle.io/kube-version: '>= 1.19.0-0' + catalog.cattle.io/release-name: artifactory-ha + apiVersion: v2 + appVersion: 7.90.14 + created: "2024-10-09T00:35:07.012228604Z" + dependencies: + - condition: postgresql.enabled + name: postgresql + repository: https://charts.jfrog.io/ + version: 10.3.18 + description: Universal Repository Manager supporting all major packaging formats, + build tools and CI servers. + digest: dc098ed79093e4ae894fdff48aaa382ca9764be14c0f2d6e98ad6953e4890ee5 + home: https://www.jfrog.com/artifactory/ + icon: file://assets/icons/artifactory-ha.png + keywords: + - artifactory + - jfrog + - devops + kubeVersion: '>= 1.19.0-0' + maintainers: + - email: installers@jfrog.com + name: Chart Maintainers at JFrog + name: artifactory-ha + sources: + - https://github.com/jfrog/charts + type: application + urls: + - assets/jfrog/artifactory-ha-107.90.14.tgz + version: 107.90.14 - annotations: artifactoryServiceVersion: 7.90.18 catalog.cattle.io/certified: partner @@ -1642,6 +1676,40 @@ entries: - assets/jfrog/artifactory-ha-107.55.14.tgz version: 107.55.14 artifactory-jcr: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: JFrog Container Registry + catalog.cattle.io/kube-version: '>= 1.19.0-0' + catalog.cattle.io/release-name: artifactory-jcr + apiVersion: v2 + appVersion: 7.90.14 + created: "2024-10-09T00:35:07.404179085Z" + dependencies: + - name: artifactory + repository: file://charts/artifactory + version: 107.90.14 + description: JFrog Container Registry + digest: b629ddb62d78b9ee327db9067b1a5fd5bb71ddf7c43850929fd63de16ac4e581 + home: https://jfrog.com/container-registry/ + icon: file://assets/icons/artifactory-jcr.png + keywords: + - artifactory + - jfrog + - container + - registry + - devops + - jfrog-container-registry + kubeVersion: '>= 1.19.0-0' + maintainers: + - email: helm@jfrog.com + name: Chart Maintainers at JFrog + name: artifactory-jcr + sources: + - https://github.com/jfrog/charts + type: application + urls: + - assets/jfrog/artifactory-jcr-107.90.14.tgz + version: 107.90.14 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: JFrog Container Registry @@ -20578,6 +20646,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.11 + created: "2024-10-09T00:35:07.82628804Z" + dependencies: + - condition: grafana.enabled + name: grafana + repository: "" + version: 8.5.0 + - condition: prometheus.server.enabled + name: prometheus + repository: "" + version: 25.24.1 + description: Kasten’s K10 Data Management Platform + digest: 18e3b1b58d07b8acdb8b9fb76b7b63b02b38ac8207872db83f76b12814f77692 + 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.1101.tgz + version: 7.0.1101 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: K10 @@ -28633,6 +28730,95 @@ entries: - assets/f5/nginx-ingress-1.0.2.tgz version: 1.0.2 nri-bundle: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: New Relic + catalog.cattle.io/release-name: nri-bundle + apiVersion: v2 + created: "2024-10-09T00:35:09.545137647Z" + dependencies: + - condition: infrastructure.enabled,newrelic-infrastructure.enabled + name: newrelic-infrastructure + repository: https://newrelic.github.io/nri-kubernetes + version: 3.34.6 + - condition: prometheus.enabled,nri-prometheus.enabled + name: nri-prometheus + repository: https://newrelic.github.io/nri-prometheus + version: 2.1.19 + - condition: newrelic-prometheus-agent.enabled + name: newrelic-prometheus-agent + repository: https://newrelic.github.io/newrelic-prometheus-configurator + version: 1.14.4 + - condition: webhook.enabled,nri-metadata-injection.enabled + name: nri-metadata-injection + repository: https://newrelic.github.io/k8s-metadata-injection + version: 4.21.2 + - condition: metrics-adapter.enabled,newrelic-k8s-metrics-adapter.enabled + name: newrelic-k8s-metrics-adapter + repository: https://newrelic.github.io/newrelic-k8s-metrics-adapter + version: 1.11.4 + - condition: ksm.enabled,kube-state-metrics.enabled + name: kube-state-metrics + repository: https://prometheus-community.github.io/helm-charts + version: 5.12.1 + - condition: kubeEvents.enabled,nri-kube-events.enabled + name: nri-kube-events + repository: https://newrelic.github.io/nri-kube-events + version: 3.10.8 + - condition: logging.enabled,newrelic-logging.enabled + name: newrelic-logging + repository: https://newrelic.github.io/helm-charts + version: 1.23.0 + - condition: newrelic-pixie.enabled + name: newrelic-pixie + repository: https://newrelic.github.io/helm-charts + version: 2.1.4 + - condition: k8s-agents-operator.enabled + name: k8s-agents-operator + repository: https://newrelic.github.io/k8s-agents-operator + version: 0.13.0 + - alias: pixie-chart + condition: pixie-chart.enabled + name: pixie-operator-chart + repository: https://pixie-operator-charts.storage.googleapis.com + version: 0.1.6 + - condition: newrelic-infra-operator.enabled + name: newrelic-infra-operator + repository: https://newrelic.github.io/newrelic-infra-operator + version: 2.11.4 + description: Groups together the individual charts for the New Relic Kubernetes + solution for a more comfortable deployment. + digest: a067cc71a21b7f8be82f3f12171751746741c898bbf09b4b5c5f054e4d4fc286 + home: https://github.com/newrelic/helm-charts + icon: file://assets/icons/nri-bundle.svg + keywords: + - infrastructure + - newrelic + - monitoring + maintainers: + - name: juanjjaramillo + url: https://github.com/juanjjaramillo + - name: csongnr + url: https://github.com/csongnr + - name: dbudziwojskiNR + url: https://github.com/dbudziwojskiNR + name: nri-bundle + sources: + - https://github.com/newrelic/nri-bundle/ + - https://github.com/newrelic/nri-bundle/tree/master/charts/nri-bundle + - https://github.com/newrelic/nri-kubernetes/tree/master/charts/newrelic-infrastructure + - https://github.com/newrelic/nri-prometheus/tree/master/charts/nri-prometheus + - https://github.com/newrelic/newrelic-prometheus-configurator/tree/master/charts/newrelic-prometheus-agent + - https://github.com/newrelic/k8s-metadata-injection/tree/master/charts/nri-metadata-injection + - https://github.com/newrelic/newrelic-k8s-metrics-adapter/tree/master/charts/newrelic-k8s-metrics-adapter + - https://github.com/newrelic/nri-kube-events/tree/master/charts/nri-kube-events + - https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-logging + - https://github.com/newrelic/helm-charts/tree/master/charts/newrelic-pixie + - https://github.com/newrelic/newrelic-infra-operator/tree/master/charts/newrelic-infra-operator + - https://github.com/newrelic/k8s-agents-operator/tree/master/charts/k8s-agents-operator + urls: + - assets/new-relic/nri-bundle-5.0.94.tgz + version: 5.0.94 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: New Relic @@ -40451,6 +40637,35 @@ entries: - assets/speedscale/speedscale-operator-1.3.10.tgz version: 1.3.10 stackstate-k8s-agent: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: StackState Agent + catalog.cattle.io/kube-version: '>=1.19.0-0' + catalog.cattle.io/release-name: stackstate-k8s-agent + apiVersion: v2 + appVersion: 3.0.0 + created: "2024-10-09T00:35:10.293415083Z" + dependencies: + - alias: httpHeaderInjectorWebhook + name: http-header-injector + repository: https://helm.stackstate.io + version: 0.0.11 + description: Helm chart for the StackState Agent. + digest: bb89a8c3005bac496a1a9076b936b314815ddc3fccbbd0767f99064fa6dc5955 + home: https://github.com/StackVista/stackstate-agent + icon: file://assets/icons/stackstate-k8s-agent.svg + keywords: + - monitoring + - observability + - stackstate + kubeVersion: '>=1.19.0-0' + maintainers: + - email: ops@stackstate.com + name: Stackstate + name: stackstate-k8s-agent + urls: + - assets/stackstate/stackstate-k8s-agent-1.0.98.tgz + version: 1.0.98 - annotations: catalog.cattle.io/certified: partner catalog.cattle.io/display-name: StackState Agent @@ -44399,4 +44614,4 @@ entries: urls: - assets/netfoundry/ziti-host-1.5.1.tgz version: 1.5.1 -generated: "2024-10-08T00:34:54.185236969Z" +generated: "2024-10-09T00:35:05.224730742Z"