From 00d13b5da8e40edac7abbd7312ae2c900200c17f Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" Date: Fri, 12 Jul 2024 16:38:01 +0000 Subject: [PATCH] Charts CI ``` Added: cerbos/cerbos: - 0.37.0 Updated: airlock/microgateway: - 4.3.0 airlock/microgateway-cni: - 4.3.0 jenkins/jenkins: - 5.4.2 ``` --- assets/airlock/microgateway-4.3.0.tgz | Bin 0 -> 61618 bytes assets/airlock/microgateway-cni-4.3.0.tgz | Bin 0 -> 10769 bytes assets/cerbos/cerbos-0.37.0.tgz | Bin 0 -> 7873 bytes assets/icons/cerbos.png | Bin 0 -> 14027 bytes assets/jenkins/jenkins-5.4.2.tgz | Bin 0 -> 76631 bytes .../microgateway-cni/4.3.0/.helmignore | 27 + .../airlock/microgateway-cni/4.3.0/Chart.yaml | 43 + .../airlock/microgateway-cni/4.3.0/README.md | 137 + .../microgateway-cni/4.3.0/gke-values.yaml | 4 + .../4.3.0/openshift-values.yaml | 15 + .../microgateway-cni/4.3.0/questions.yml | 18 + .../4.3.0/templates/NOTES.txt | 3 + .../4.3.0/templates/_helpers.tpl | 101 + .../4.3.0/templates/clusterrole.yaml | 22 + .../4.3.0/templates/clusterrolebinding.yaml | 20 + .../4.3.0/templates/configmap.yaml | 22 + .../4.3.0/templates/daemonset.yaml | 136 + .../network-attachment-definition.yaml | 13 + .../4.3.0/templates/scc-role.yaml | 22 + .../4.3.0/templates/scc-rolebinding.yaml | 20 + .../4.3.0/templates/serviceaccount.yaml | 13 + .../4.3.0/templates/tests/rbac.yaml | 64 + .../4.3.0/templates/tests/test-install.yaml | 103 + .../microgateway-cni/4.3.0/values.schema.json | 225 ++ .../microgateway-cni/4.3.0/values.yaml | 85 + charts/airlock/microgateway/4.3.0/.helmignore | 28 + charts/airlock/microgateway/4.3.0/Chart.yaml | 44 + charts/airlock/microgateway/4.3.0/README.md | 180 + .../airlock/microgateway/4.3.0/app-readme.md | 28 + ...cesscontrols.microgateway.airlock.com.yaml | 124 + ...ntsecurities.microgateway.airlock.com.yaml | 139 + .../denyrules.microgateway.airlock.com.yaml | 1804 ++++++++++ ...nvoyclusters.microgateway.airlock.com.yaml | 58 + ...nfigurations.microgateway.airlock.com.yaml | 185 + ...yhttpfilters.microgateway.airlock.com.yaml | 58 + .../graphqls.microgateway.airlock.com.yaml | 88 + ...aderrewrites.microgateway.airlock.com.yaml | 759 +++++ ...propagations.microgateway.airlock.com.yaml | 108 + .../crds/limits.microgateway.airlock.com.yaml | 651 ++++ ...idcproviders.microgateway.airlock.com.yaml | 305 ++ ...lyingparties.microgateway.airlock.com.yaml | 224 ++ .../openapis.microgateway.airlock.com.yaml | 167 + .../parsers.microgateway.airlock.com.yaml | 358 ++ ...disproviders.microgateway.airlock.com.yaml | 159 + ...ionhandlings.microgateway.airlock.com.yaml | 77 + ...ecargateways.microgateway.airlock.com.yaml | 758 +++++ .../telemetries.microgateway.airlock.com.yaml | 96 + .../4.3.0/dashboards/blockLogs.json | 510 +++ .../4.3.0/dashboards/blockMetrics.json | 758 +++++ .../4.3.0/dashboards/license.json | 521 +++ .../4.3.0/dashboards/overview.json | 1138 +++++++ .../microgateway/4.3.0/templates/NOTES.txt | 34 + .../microgateway/4.3.0/templates/_helpers.tpl | 153 + .../templates/operator/_operator_helpers.tpl | 42 + .../4.3.0/templates/operator/_rbac.gen.tpl | 237 ++ .../templates/operator/_webhooks.gen.tpl | 339 ++ .../4.3.0/templates/operator/configmap.yaml | 394 +++ .../operator/dashboard-configmap.yaml | 28 + .../4.3.0/templates/operator/deployment.yaml | 143 + .../templates/operator/manager-role.yaml | 33 + .../operator/manager-rolebinding.yaml | 45 + .../templates/operator/metrics-service.yaml | 47 + .../templates/operator/mutating-webhook.yaml | 28 + .../4.3.0/templates/operator/podmonitor.yaml | 27 + .../4.3.0/templates/operator/role.yaml | 45 + .../4.3.0/templates/operator/rolebinding.yaml | 20 + .../templates/operator/selfsigned-issuer.yaml | 13 + .../templates/operator/serviceaccount.yaml | 13 + .../templates/operator/servicemonitor.yaml | 60 + .../operator/serving-certificate.yaml | 19 + .../operator/validating-webhook.yaml | 28 + .../templates/operator/webhook-service.yaml | 23 + .../4.3.0/templates/operator/xds-service.yaml | 24 + .../4.3.0/templates/tests/rbac.yaml | 143 + .../4.3.0/templates/tests/service.yaml | 23 + .../4.3.0/templates/tests/statefulset.yaml | 56 + .../4.3.0/templates/tests/test-install.yaml | 227 ++ .../microgateway/4.3.0/values.schema.json | 540 +++ charts/airlock/microgateway/4.3.0/values.yaml | 213 ++ charts/cerbos/cerbos/0.37.0/.helmignore | 23 + charts/cerbos/cerbos/0.37.0/Chart.yaml | 27 + charts/cerbos/cerbos/0.37.0/README.md | 9 + charts/cerbos/cerbos/0.37.0/app-readme.md | 5 + .../cerbos/cerbos/0.37.0/templates/NOTES.txt | 17 + .../cerbos/0.37.0/templates/_helpers.tpl | 191 ++ .../cerbos/0.37.0/templates/certificate.yaml | 10 + .../cerbos/0.37.0/templates/configmap.yaml | 9 + .../cerbos/0.37.0/templates/deployment.yaml | 149 + .../cerbos/cerbos/0.37.0/templates/hpa.yaml | 32 + .../cerbos/0.37.0/templates/service.yaml | 40 + .../0.37.0/templates/serviceaccount.yaml | 12 + .../cerbos/0.37.0/values-audit-log.yaml | 40 + .../cerbos/0.37.0/values-cert-manager.yaml | 40 + .../cerbos/0.37.0/values-git-storage.yaml | 33 + .../cerbos/0.37.0/values-hub-storage.yaml | 22 + charts/cerbos/cerbos/0.37.0/values-otlp.yaml | 43 + .../cerbos/0.37.0/values-volume-storage.yaml | 19 + charts/cerbos/cerbos/0.37.0/values.yaml | 149 + charts/jenkins/jenkins/5.4.2/CHANGELOG.md | 3029 +++++++++++++++++ charts/jenkins/jenkins/5.4.2/Chart.yaml | 54 + charts/jenkins/jenkins/5.4.2/README.md | 706 ++++ charts/jenkins/jenkins/5.4.2/UPGRADING.md | 148 + charts/jenkins/jenkins/5.4.2/VALUES.md | 309 ++ charts/jenkins/jenkins/5.4.2/VALUES.md.gotmpl | 28 + .../jenkins/jenkins/5.4.2/templates/NOTES.txt | 68 + .../jenkins/5.4.2/templates/_helpers.tpl | 669 ++++ .../5.4.2/templates/auto-reload-config.yaml | 60 + .../5.4.2/templates/config-init-scripts.yaml | 18 + .../jenkins/5.4.2/templates/config.yaml | 92 + .../jenkins/5.4.2/templates/deprecation.yaml | 151 + .../jenkins/5.4.2/templates/home-pvc.yaml | 41 + .../jenkins/5.4.2/templates/jcasc-config.yaml | 53 + .../5.4.2/templates/jenkins-agent-svc.yaml | 43 + .../jenkins-aws-security-group-policies.yaml | 16 + .../jenkins-controller-alerting-rules.yaml | 26 + .../jenkins-controller-backendconfig.yaml | 24 + .../templates/jenkins-controller-ingress.yaml | 77 + .../jenkins-controller-networkpolicy.yaml | 76 + .../templates/jenkins-controller-pdb.yaml | 34 + .../jenkins-controller-podmonitor.yaml | 30 + .../templates/jenkins-controller-route.yaml | 34 + .../jenkins-controller-secondary-ingress.yaml | 56 + .../jenkins-controller-servicemonitor.yaml | 45 + .../jenkins-controller-statefulset.yaml | 424 +++ .../templates/jenkins-controller-svc.yaml | 56 + .../jenkins/jenkins/5.4.2/templates/rbac.yaml | 149 + .../5.4.2/templates/secret-additional.yaml | 21 + .../5.4.2/templates/secret-claims.yaml | 29 + .../5.4.2/templates/secret-https-jks.yaml | 20 + .../jenkins/5.4.2/templates/secret.yaml | 20 + .../templates/service-account-agent.yaml | 26 + .../5.4.2/templates/service-account.yaml | 26 + .../5.4.2/templates/tests/jenkins-test.yaml | 49 + .../5.4.2/templates/tests/test-config.yaml | 14 + charts/jenkins/jenkins/5.4.2/values.yaml | 1333 ++++++++ index.yaml | 187 +- 136 files changed, 22525 insertions(+), 1 deletion(-) create mode 100644 assets/airlock/microgateway-4.3.0.tgz create mode 100644 assets/airlock/microgateway-cni-4.3.0.tgz create mode 100644 assets/cerbos/cerbos-0.37.0.tgz create mode 100644 assets/icons/cerbos.png create mode 100644 assets/jenkins/jenkins-5.4.2.tgz create mode 100644 charts/airlock/microgateway-cni/4.3.0/.helmignore create mode 100644 charts/airlock/microgateway-cni/4.3.0/Chart.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/README.md create mode 100644 charts/airlock/microgateway-cni/4.3.0/gke-values.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/openshift-values.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/questions.yml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/NOTES.txt create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/_helpers.tpl create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/clusterrole.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/clusterrolebinding.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/configmap.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/daemonset.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/network-attachment-definition.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/scc-role.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/scc-rolebinding.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/serviceaccount.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/tests/rbac.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/templates/tests/test-install.yaml create mode 100644 charts/airlock/microgateway-cni/4.3.0/values.schema.json create mode 100644 charts/airlock/microgateway-cni/4.3.0/values.yaml create mode 100644 charts/airlock/microgateway/4.3.0/.helmignore create mode 100644 charts/airlock/microgateway/4.3.0/Chart.yaml create mode 100644 charts/airlock/microgateway/4.3.0/README.md create mode 100644 charts/airlock/microgateway/4.3.0/app-readme.md create mode 100644 charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml create mode 100644 charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json create mode 100644 charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json create mode 100644 charts/airlock/microgateway/4.3.0/dashboards/license.json create mode 100644 charts/airlock/microgateway/4.3.0/dashboards/overview.json create mode 100644 charts/airlock/microgateway/4.3.0/templates/NOTES.txt create mode 100644 charts/airlock/microgateway/4.3.0/templates/_helpers.tpl create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/dashboard-configmap.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/deployment.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/manager-role.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/manager-rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/mutating-webhook.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/podmonitor.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/role.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/validating-webhook.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/tests/service.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml create mode 100644 charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml create mode 100644 charts/airlock/microgateway/4.3.0/values.schema.json create mode 100644 charts/airlock/microgateway/4.3.0/values.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/.helmignore create mode 100644 charts/cerbos/cerbos/0.37.0/Chart.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/README.md create mode 100644 charts/cerbos/cerbos/0.37.0/app-readme.md create mode 100644 charts/cerbos/cerbos/0.37.0/templates/NOTES.txt create mode 100644 charts/cerbos/cerbos/0.37.0/templates/_helpers.tpl create mode 100644 charts/cerbos/cerbos/0.37.0/templates/certificate.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/templates/configmap.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/templates/deployment.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/templates/hpa.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/templates/service.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/templates/serviceaccount.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values-audit-log.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values-cert-manager.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values-git-storage.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values-hub-storage.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values-otlp.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values-volume-storage.yaml create mode 100644 charts/cerbos/cerbos/0.37.0/values.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/CHANGELOG.md create mode 100644 charts/jenkins/jenkins/5.4.2/Chart.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/README.md create mode 100644 charts/jenkins/jenkins/5.4.2/UPGRADING.md create mode 100644 charts/jenkins/jenkins/5.4.2/VALUES.md create mode 100644 charts/jenkins/jenkins/5.4.2/VALUES.md.gotmpl create mode 100644 charts/jenkins/jenkins/5.4.2/templates/NOTES.txt create mode 100644 charts/jenkins/jenkins/5.4.2/templates/_helpers.tpl create mode 100644 charts/jenkins/jenkins/5.4.2/templates/auto-reload-config.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/config-init-scripts.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/config.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/deprecation.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/home-pvc.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jcasc-config.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-agent-svc.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-aws-security-group-policies.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-alerting-rules.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-backendconfig.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-ingress.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-networkpolicy.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-pdb.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-podmonitor.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-route.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-secondary-ingress.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-servicemonitor.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-statefulset.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-svc.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/rbac.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/secret-additional.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/secret-claims.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/secret-https-jks.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/secret.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/service-account-agent.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/service-account.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/tests/jenkins-test.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/templates/tests/test-config.yaml create mode 100644 charts/jenkins/jenkins/5.4.2/values.yaml diff --git a/assets/airlock/microgateway-4.3.0.tgz b/assets/airlock/microgateway-4.3.0.tgz new file mode 100644 index 0000000000000000000000000000000000000000..3a6d73415952dc43f14c2af00f572171e6c764ce GIT binary patch literal 61618 zcmV)mK%T!JiwG0|00000|0w_~VMtOiV@ORlOnEsqVl!4SWK%V1T2nbTPgYhoO;>Dc zVQyr3R8em|NM&qo0POu~f7>{cIF9dQy$Xzy9@|-yI(=!!J>zlGJ>JRTwcXv{{im}I zM1m696u|~SIqD?;*Z%w9BEd_S6(^mT4-%OK3WY+Us!%8tCfK871Ub5ev)1VtQtr;+ zB>ay}KJ9k9z2EQ4|J&_$@&9hG-~W$Jf4|@F?(OaMy8qGc^bXpc{{ZccVo~!Xj6?b# z?K`(s?A)K^fl)*_4WicN! z(J7}Mlges0$q+eHL>VU00Bn7K+;O{mZrf>ZRTifxM35of%A^2Yh!f_L7)5N111>7e zK?lb8XVdVs+kkPL{@HhXZoAJg!*jIaKJAoKZt+P5S+wu zh&{>A;0H|6Eeu0h;N-lb1dM!ea()J4N;vZ5Byhm1UrsJXVcngElkboNF3r4jzz;a& zh~j9}aKKYEy+q!tm^B;#;EgdreQBZ5940chGR({$w|ex+)n0cbWGOM-+91W6dGcO(E5>ACMV#QSQn&*2b- zOn`vCFaHZeyQW7b6B4OX1MvQ%MkX_I7FD&1X7$e|aD>j2FuX(_MVy%~+>TKM;v@`3 z|F9^;+y*Wb9)Q9-7Lft*?-+INFwwuMxKB%L}NI`gi2M08A}iYej*24>gs6^M1%vF za58}$doT=VAVXHPrh5e`0x+Zq`ZLiG6Aj@=c1NQxu~PI$G$3>$AVAQRz16hPgg{Nv z2SG%w-%dqG1|WbTLk*%8U`qIx$z+JA;z+4dm^3d$+omjdDs2s6tqRb(+$l*SJ^)<< z=rTs0Fw|JSZW0TM1IDQ^u4e*(dUdJa45V?WwAuxcL;8~}AOYv0Rt}rd>53&q1M-K0 zn=!P&n=s4?oq>RoiKe|6TQM(J01qrs1JL|E5ymvB)YgunL$intha=Org2~}%0Gd*T zTZJ$Gb<*A2S8W3yj}YT2%G;Vp1V2CynrsZad;0_L;0U2!aJYB4-wlSn!+mt*`}^JD zQM)tTZy$8_JAU`zuze8hb-MmRF!Yg!+IvV;@(;ZIUUSFQ?Hhr_B({?Mm>v&bl~EWC z(zbnXKWGO>u+#C|z23gpX$SkmL%8Q11p9se=n#5GouPl!_WDNFs#{mWbmO6I{1YGbFLz2yK6od(iJfyU?H%5LU z#JFhBKukM)4k?@=cT>zD3ikYDe@t{Gvj}*QrDIUQGl$ZI_k#7vpB3<69rcWa9 z_9P6g_jye9)J~YNTMTLPpo}bhD}^^UQ?t~dpoE~xB<3^K_yrn4>W7Fi1G+?JW;`~+ zP@k27?t+Vdo}3DH*HcxoNJ<%c2@vE7MOn`wIHzcWlL^ocW>LG$l5fg0&7VbZqE{D8l2p~R2Uk-m zorj_$N>-CN9iCe>920WGzz{M?4*)^hI0%K+pc^1mvpyFDgu_rSdVIu1;{vwO|B;77 zl$i%|HKy0U8Y@YTiNsY-)?5qub<-{o%P*-hB6ORR9b#ce5-ueUq})g>(^s>g6GVp| z4l##Z(p%GQ{=AXAvjVtYf`MIcQ_E ziOruz*<>rtrp<9^$+{5&DoCi*mgpT6(ueH=hAg-AO)^9@LL5oE)+Zio3En|5XDu=n z3Ju-1ZVA1?(Z~^F?U3Pb$m6Uf`GvI}r7Y$M>>WMt9fr6gIoZ)&5Jrohl#1)LA)FeB zwwpWeDHklBv&I#GXYa¬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

*`(y(IAgiT9-0$4>?#Ftak=j7;UI|$FKtF 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

4je($)oh0XaDNqrq%TgiE>b32$&yr(3DY zNYa`7JkF~K1Pfq3sfj{=i*B@9;Uczm@*3j*@^KP{uzNu9bG&fe%Lex-g1s}hwul0; zB6)JDTc%Bc#FOH)gE)2 z)ozcgyKI_twTk!PoxrI_nH_qPO8dmq#R(2t6Ys}T_)dWtcPn*lknqo;{jk_rW|fvs zj-5$$NWaXE+uNF*Y8S}(Q&l=pr0{b}eI8 zW`CMAY2A;Pmx-o(%&4Y9S}N#8y|x?N%S4pxOpdhaX<9WBn%0{Jm;8+1yvUTG%kaOG zE_kQE()WL04tCniB2l9=`fee9>GjL8Q1dvS(QjHmiye>1aXdDCS#W!eOk}tjN$L2dbvNqy$AEla zUHhGj@pJZqEjn$wn&BY{Tf|gAF}(EKTie)j9wBg1T(zL1qeH0v3h1)gO~G#;IBiKu z2b~_SRAk~LvA{MfqUUxG#ooo2%9JKj;s?As;?2(w);D1Fv#{@i^*pK7PB&n>n=$fH4f z?}n#GZ6_TxA>OHRYD=WX1ZZ}u!qM2w|1hIB{R@bjd!xwC-4G@z>mD1Tv?mV!$uVgT zIVY#mki-@rw>MywaB&dE%X1Yirb0l8xcsG6lTU$wCs-gJY?=5>|B-xf#qA^q;{RAg zP9@Tn_sgn^FyuR3-sH`$bcljLA1r0MaDmEIS%L>2)cZtvoEi;s1waTAp{yTmu|bvf%)6QZI3%QdgH*>C?StyhcJ+*acbKB{G<6uZL0v&lXCA4fi*RLE+$8yfXn*#6?(V=nYqL6IF zFjKs<_0n;00F&i-5Did#O8*n%L$5YEdO<43jD+)W`d;_ArR9DQ*q0EG<;wfQvkI-Tw9WW zZ+K?$Zm%e@wB~)q)}QQOl3>yQ(7VPVXIJsOx1is@?GwPI6J#A;Azpaw#j^%^4`(J< z71+jnP0PPfg@2z9znx_BWRvo%>@*w<_@yt1uAxk%1?;9ezF>GWschZa*o~f#Sk{FT0h`^7eq;kr8(=MootJZ~02>>8fkC{rDZw2$d^jI*Hs74+ffR(`)~ zBOub{;_c&4Y=X0beMjG>d*lq>3I0Jd^7p9n-87<#$&*PF#%X>z)A8x7P6}=|0XCa> z!R3B&-oTzwTb3U`wmr+IvhjHL8f|-wbgR|r@~)~6m_pCom1S%=F9#gHTu zz~R7289r}mB49Z%3CQgXJj83YwUPWb;f>&=y8s-)nbAYVp6FzEW z=_DCAuBBYFoH2W^5gke!^M3Gyur$hl#sXZ}=A^MoX#9Xq{qdhmYb7&pLodzzQ{k2H1zibBB~k{@-3+pB=2ib2D&`<8lz^jT`BrZ z0hK7$L|_+!G~2KJ;q9cx!M5obS2iRrwQ=<{%0#acFLzRzVgn6sEt?}AJo+*KVFJ{? ziG(lZ)UlB&H$6*u#W=>LDMcev(~P?a6;!PM!<^oDX0Esk(|~1HFjh#qTpCF7C9={D zsIua?%?x}7G8ZbVqzqYe*f@7w!T;HV?Ej-~cFf!A+^07;*VQGBiG`_N`R#nI<FRAqBa><$T}|^dgbC$S z?WmlNBOJ6FJ^S|kbZ$(Ro7o!-?U5@-`%IKHt67=f9gY#cirR=QEUE@;z@Fmi4ZloF`wd z6o59*5?+7X*5cjC7dk#fmF+Opdtq^nqa<57T29&Q4i_m0p1?h!qWzzAKuKl7t%f0^ zr!0kvfKV2NY$6<`rrsFIXm%^lW=N)3PQQ z$+$*WG^ESQFL3rO15M9*kyXd_yB0~Tc58;(R$nlcWesA!J*eB!VsZiHekgANe}~@2 z#bAKxOjJ6em=kN`c{b`B0l@^T9`WIx?JTNatKlw-xBhz04D-IGWL1-M*1~TZdN;VJ zQj5jx=dbZ`vV-3WfCb+jkIZBJN4bpjHgZ?KRy>h@wv?B)`I$Xmk(9{AZ6F&O(aqGx zO^!vff-rX<&ReXIEh|Nm1fFmjF(Cg=66v^BaNnQUX+3;QAay1)RNAM}Y5RQ&vXZGm zY(g8t6P!quhpEpFDn2nR&04G)<6Eb?4)7i&`TjnJzVp$Q^F7 zbJAc;?n$21s}9}kC?F8I&|J2_+emFN%r{;OoLk7<-3)!DusdYRyM0Yg{gYYhsMZYH^5vKM_n= zZWmb*BYM=^2<@;d!e_~D8;3}W+BFE`=}aw0=?v}ZG_N>!jz~3s*)GZcrmpRi2ca|w z1G9-K7>4O|;gB>?+WvE+q|mV>D797s{|`dk4^h&@QEOsA{Z1(iPQ$#zqCo0 z!`MkAOskQ~kgWsD?80jkX8WuKTME{My>)C0_#TxfcG|RTampmG)c$<1I42_D7$@ph z>6&yjI^sK8ts*|CAuY^GXbnM$N?wTXsX1J4tHQs^2gF?bP@GV)(8|zhZkTG5%8|8b zMfeZM+?b)@#IU41h7I>waP4m4LlV%mr7ZD5qJ!CSUs3QymUUKda@en9{O5KA66uu3 zayNH=o7U`~O%J|Z;%8PxTiZ0Pu35ZcA7x8vHg|iz0oq9#Uj0vf!s~LsEww2JeQZ)A z@w3BSb7a<#HvIEhjg8=x%5!KovHR|Kj`G~dgxsy^UFrO&w3RrPB1)FuD&`ga_xpu56H8qSfthYE{-mU zA%nOKd;Jy`Qh#T z=;`rFe8(@}`T9vmtTOe}&8Js%&{4f(%ANb~-X1)&8(rQq*j9~T;7rqqZ@qcHVbkxR z_QlQmBv}94$kRXUcCg1+iKvWHX^$$LZ8*B5>dpsdBS*5jKh4wc3oORkAW7F6DEet7 z>17IA1GX<|9;zwF_K4(CF=pO>obTBpspfN^9xx}1K7UycrBg3iRs9v`IN_b5@^R{x z?l;x&r*g6ChRE$LR9h4=6-9^Q=y<2zm}o~_DAxM(+ZHO|@I1KDL=?jT#u8h}%9DElKNLGkv4kXT1$65UvAvx z-;{BD#@Mcag8`s-X&Q)=aFch?X3D&JD;G(Ng8cRMc|V50KK@gG!UoF&8sN6!;Qj|f CnT=Bb literal 0 HcmV?d00001 diff --git a/assets/cerbos/cerbos-0.37.0.tgz b/assets/cerbos/cerbos-0.37.0.tgz new file mode 100644 index 0000000000000000000000000000000000000000..3e68373979b7f5a73fab2583a15e203698671311 GIT binary patch literal 7873 zcmV;y9zNk8iwG0|00000|0w_~VMtOiV@ORlOnEsqVl!4SWK%V1T2nbTPgYhoO;>Dc zVQyr3R8em|NM&qo0PKBhcN;gdU_SS+sHJ>&tX#86JuGMYTxCa=ltd@8#FgaC&Y7Jf zgWVvB2;JxaXwrPA0ENr|G2XN3i>tsVe|j3oj0$)+S=XQ-Q9lu`t{ECSHrFC-Mzi9VE6~%(n^Jph<`Qw z-EB2H_k|>sWlSPTSSF$XAVv~VHVtDWCB-3OgBWv3CIlxDH;a-2cTIDQ2bWzzypF`#M2 zge38+06@fuqA|rF7%dcdkTWD9MK@SLAuxv=^OOjMjRKeeD&&lD@~a+X5G2hhR#@mC zDb8d7x?GAf)!29m3r-}-rVz7C;*YYA7Kmd&ag2q~nVeC`86`2n0w#>Zk`)|Ko>LO* zW&!|?1uJ-rMHtMO8oQa4xrhb>GlpS;=K+b;@JtBB(O`gtz)}oUo^XZoY#Q9)<%01< zL_r_W7{%&8ze_qr=1t)SDC<6GYbN*9AGKolrxo!-n zh|maTnCAaE!!-S?hdF>xXci$6)Tl%*a~#!1W!|8YKYQ{03na_`=ZF?q{6PZX4*vh< z^=^y*zuDP(=KoKTHsA=|yVL%J@f1mk6U{VKfg>DK z#hvDe6U7&WVr^p#xnd^?kW4beog_|kilpL4dYPWjG3O+~5%hY&q@c8w8En8=k&ZD3 zHgO9*XNhis7g@luUfGAN$Ycats{A3d3X#=2u>zvOpZ7fhEH$6iGfR|SZG;N2pqzd=KeXyF-~w6V_$WUARrl$ ztD^VX6I_2J8HSHF(WpCj7bn!YN2UxQ534Mc?HM- zA_As3!(3qug;2s&0d{0o5@RVx47siDNY0Wl$XRmGKtI@kQ#8hOReAH#6+L8W$}*r` zH9UD8>tgk)t5GxBj8Y`Rp!ywkjCHU9msScp-9j&S1!XZ~v_jhzJ2a)rnVN|2IVAaBzC6e{+nI^NcPp7?a--ip5e$ zoO*=Fi|jz0u}l?cX1^DhN3dn_zObVWl%zz85^X&I`%wwEM#Ten6p+?LUd#90GT~hI7mZ)V)aV#0H4_?L+ zd}+Tpsv9vytrVLsi7@c~F#O4sL6eDMUQ6HwRSUuW?ciLooKZGi!ey>#L9u%wxzZYi zt0HX?Q^G5t+fkkDgDdkRUPk2~q0&ovhLwaOZ15|~u$wIO@$v>Q%;2hw#SCZfGl3)# z6D{;2xQu5wDJVY3mJ760xcXCiFgKrE&rL~46|ZZSZ!aTL`JFE8ARK3N!da%) zeO(YYTgKVk)wwkEk2|YQI7=-ZLW*($j4K6R$rugfJI>M)P>y5Axz%Q|Z#FVoi%>hq zSTShH9II|H2%mdt?yhQSkVI%skJDT(6=U~ds~MVa4Gz5)+@ggAq0|#tupn{(z;9fE4vn(pZ8Z~yO&@a3<|M&@vLz%= zz%vdc`x(blXIn(HzC^sSPQ}gX-0X1mTor0?QAf-Go&o^it=9 zX-Pnl5V}v2X^GAX~W!kQPE>eliL`Z-GaQ z%LsN}|BvkeV9N8j%-FFd6mv6~TVAnx?oOvF6T7w|$O=jW05yDX5k*;C%T|I@>PaG* z#5_aviYsOp8@)4BlP|66U8s?KrL~>#C$mjcJ=0K#(Bmc@=hlU7kvAH_E3`>2}a5J;0Eli^#GD_J(fxp2^C%^k3sR(0* z&jtRaARH%QZT+j=kWqm$#UZ^)H%i7-l}4P*RfQY8EIDBdf9OMuOmpYjgoS7bhX+lD z>M?i8RX=jZsNNs={G!}z*r}y#3KY*V?O4f2$8X<%{|ZjdzB_*fe>=E1(^PkSadCd3 zX?c38Dn`11(fpGHMx3Q;`%qNNF(D#WY-s7Pk(wX^#S?9{QBAErcKS`JzEv_%iKC98 zzGSJ{G+CiHJ_brmvkY+FFj z$>v@uAuT7;k`iu}Y*qq?OE^0}I=&vAUtI0Kc&Rl0SW?}mzCK|PIf^lS`UK&HId=%p z%G}#q=rj0Puq;P%w%_{{YOPN}75^CiGt4X!w>`%Pg@V-VkF!DaAcWi7%?4&CqXu3I zs`yRV5s6ccn2<;oB1B2Tu@G42**|W38i9Dm6j(P6x3|$?@WSE~y(oulpueMmSfHs= z!c}W}GB6dmJ|m}|-m`9206;OE2u>uVXsNYXKrq1zwXZECuD0erNH`f;J0qGSq7P(3 z1I;8ytqUY$L!xF7)IMdbwwIaKApY`T3J~)c`ir~g^)J=DULIfkbaL1+wHnra8ojeM z==U81hu)`8y-%Njqil-RYD(}&a3tVsqan$riUeS|3AeYOK85NZz*z#fx4qkSgi&=U zQYbg(R`>b#_5rj~YUpd1uR^7F#1@&}k+iu=$$-?vHgVqCeiIJEVYtLMJV&fItp4Z|F?FAZ{9TI z|Lfu2&a?b~iuCEz;OpS{W3Ci^*|5WFO*ag_9#|4IW`~l74gL13k9w~e8dnm%t#>r- zAML#y{a%CC1C89Fu^XYX1*4+;QXs02RRHC=Mmr4th7D|8 zUaA&U7X*xPj0%B4sEvYW;Lx3&5KI%Z|Iub%Yez)P$0ADvyQvWl}PnZFqZClj!q z2?j{X&CKq=974!)8&Rmu0uoBH6_i?RSB-1M8$4yl4Q2DB%njMf@AnyQ#ggss7aH|w z|NisYp^zCMHSSpAS<8GNayciNoIvkCME^fTuK~+6dVlKb{#m0rZ!*>Tu6k{(= z`r~C(6WEv5Qeo@sos6Sy|M<=JPqM$GlvdqXMK|u?Fb)}$D|_efN8M_EA*nbrR>nkc zV(~BSjcj9B!cE(W+Mc!sNiRhy^NJ6;98q?=-!%lDNzOCCt?W|HS?XEfOus7ooFXa_ zbClxk1dh_iAi}tk@`5op{Efg1=$cm8bAmC*}VhO!?hT|JiqyvLM zuT3rPLkOQi%+U7hJyE2an_Vh_H;Zcwsul8$>F%_;v`8Y{{H#~K=7XK%&^ZjX60Mef z;?*p$?u^=M;(ox+H=l^THG38ZwI)! z3RLDZZBGw!KzsH_YvbGrpe?f;!!*!peg~?98O!u>qqA+V94L#%p`}?hMnW=feXbHt z=9ov&ONhAXX(uRg6KQizLb#TL%LoPw#&2|{Hrjm0_~8s?Q!JdPLmTnK8o~)y?Ih+b zw`rBPs#+m1U(*W>!0*K)F?VyY5ZM#4>c}C2UUUc7G-qLVRdMHKwu&XtjLACo+L9uv zFoUn;Y6RNBTMX<0m;Rv7ti1diyfhWipT=%BwWL6GhuW z=y9AfseVYD=11f{Sl7RIVQ`4z8U778#H!x!LxN1;rP>U)2D3gs4P15oB^JIWo3XV@ z?9~igFJ6u(m}}n_Bd~#F#yw;eUF&LwihZqAXNQ#waOdvN6tWJ!3c_C%gqjpqxls$Q zL{s?OG@I0Sxqg3p9X)pLZNj*>#h!bc#!>GoE5AGAKD0(IuXKj%)MHkbh6anZTVq9j zCF^T3%H|gL#pY%woN~*oz!~kB`uRPVn<}-SgjPev1MQDlv=5Ys(t>V|L|()y|K5F8 zn?RyALl>o~xOyWxt3wI}L94sQNN(NRzmG46AWyi{sUfk)DC0*6L?i zp=1An#}UuziBs$R$M;5hr0YO;_0wQUJ4yPLI6R#z_Dp18pc z$&$!-i3?IJQGz6j$~zW!9)kv8?WSgbqETy9prt{%G6C$Wsv^g+vl%#R?Ycg#upspR zf8%NG^{@5F6y=Y18St+4zq7O3T>m?J!{_z?6sdNVx?IPa2Hv6ksaLUU3ZUn9$BGJj z5q$2igyx${9kg$rm6Q9W*82A@WIPrFxWoTrYiqdIT>r1%>^`smr$~=@eDz|5IcX)x zmF6b}e2ImH?XxJ9=R)ZlR&Bw*)b2z;ucBUkIB3|}QFZ&o_dsh`c6TWJx;pO?%htwi zf(V7)-1niMFkT0VxPLzr7>9ZrR_(aGeQHK)iq{Mu-op3KTy){a!AxDv>V|DMh`wHC zP13GG?_y(UZ#7oc;Xc!=?{nLWj-EZnzZmXAv?$f^gVLO0I_6m!BI{AwHLz_5hiVY^ ztexGmR14ReUzKRBiHA4xP3b=i9dp@qVekufOj`?BIXElemp7Y#H}I~0H;-LA1{c5) zRwsVVi}6)^tt;~nK);22`+%@>*>;^HhIO#b(QRD7_M5EVqW(STw}-73IlvpVdPITW z+oreQY+dlHN_O1+^BvvR)NC~xci`Ty*sSk;<MF!Vy(Z#QQOCOvJ#%{5Yi9TRM)BUaTgE|6V{ezHY}(fs&-UBr zm+oqrJ->8G->qokW2Q~}#Xv2Uvuth-u02@oggNDT{7{(Q$*WhDk~z+>5F^gUxZVMU z#P_&C9RS83rgVpyVMOJuSw`c~(erzI$$W+)&oQqf%P~sG7lUnUO_tFbn!R#C93E{Q zwwlJ-yj}ul^FOw^?>I|aP3qhDkJfB0z(D6Zv8T%QS=HeSMusjr3%KYV<;_kX3o==~ zeOR$t&FQ90o8;vq()?Plt4~AM)DD{k{4GEi*wTfi8(!n{%1K}8y3sKem`ky4COW>a zort@s(5H_kTECU`YU2(!Bqc2_8mq>!?lNFh3&P-u~~Z(4?S_DqTdi zf}mE_?ohDS5zs2)F}|?%ffe3G@{ZwcS_K}WZQ4nH*cRe$uzmoVs#q;7&7-zg2CXTJ zlwoTgwH+^JPF2ber1QJg^5)#G_8zG1$et~&UvO&Ke`h&*>@bjbod0aSY2E+Z-5x&2 ze|(Z;#f2YTrpqq(a-Atod!5v*v=xNGeA`-5e_)*as`Qoc~Rz=27Fb*fb!b&%?@+S(G+_hS9bY+m^b9xog=X(sSE25W#k;0SeP zi#ixxvp=4-w{NY7cf`f;dF!ZYZp=B8EM_!9|v(o@9L+i4?We~eRkV<;J+t2I&Ns^7y zs-xCxH>hC0nS!8tAx+M3*Ar>|@`<$SfwacEXOH$=+M_>~=0Cj>d=3Awy zi;PHHS=D8(J;DY;723Q91FF6M)kj4_WS_F>qaFd>W&a!QZ8gt-_J+^v|4GsYoKRW_ z$#q<>8C$d-NHp>pWpMewPKm@jT3#ys*R0lrNT6)0)=u?6fQ@~w_3909aav3m5C_J_ zDX{S?bWty841gCnM*Kom73Ugg|E#R(iO3@8nRg7;{=+R*Q}BPLEW!JH5&!q=$<4kA zVAVvd?8lU3#@$5cxF&ETD7`MASF!zaiXBV7g+Ijp(EGG5)R2A=PW5=BMcbi9BT(t^Olu@ zJ;ri@u@3Q|ShGZ2D(dolgggh9F%$jru}NtA$TW)T0ibN!E*dob-qtnZyI3il?`X&` zu0N>xb5A=&<>ODd`(YgWfZ5^by%Bx-tOttfuxIrfQI_ED_59O4AAkC@8P;=p)Kr)M zwVT~fb^mwk&Fj|r-}c^{XZimW>E80c83YuF5QT}XR->9PN)1@~7?#1q^L?~zy>gi} zAFpbcyxP~g+TmGamgX$OnT-6=TvH)OT=vy_-y~h z=-}$U0(MH|Lxk&;NGDjQ#Pd|E91Tq1X_vnN7d9?^`3%(ii&J0|V_)-(T-D1w1$SUN zZCyfh?2?VCNnY37n4>J7l>vTrv-S2&i3n0evKpk6&9UP1)O#tAOsoX3rb5KAd1_iR zu3s_-qy%aivbJhQ6l2YzXWVNY1L*e2Kdr|Wj z)swRraY``vhQkoO=i*ri%#HAV@yTD}{xzu_39(>2DIe#w{AmSZB|^4Oiv84xSK}E% z;9VK)+uXLY*5*vlTmOHU+c(jik3(3tMNqA3z$CM}r7FVqLfqIxCSFiSm$Z2Bt(z+V8;s3bWYEN8O z!VZrw-kx7xD=vL_dUAYrb$xQQ|6f|yEC2rXVE`_V4=;|d>Of51eF6RL{Nl&U(ZS(y z9ab~@+xsev=EmwnbjDdRb~@v%7+2!W$W~w68Q9w8GlHJR^Yq~D@o5h>;DBl$VM*qg zE??>BkFR_ zMc?R1!)%c$%J5o{7^QxRB3C=&2ofY_V}^JVT0^jBXt)z4OZa*}T@Oa4kz0$)v*G%& z4cC7*i~P4vb@^}R>9Mzfclm$5ZoU8idhhxD|EEcv^50ndkTdPsAZXeb+*wDh0`7Cl z`?7`^6J~w8t8CR;A2iGVX3kOj(1>`nLStE_hPY?TbY}T<88n_Rt2GE+&zR}9|Lpm0 zCcFMop7+%#rT7o<0l#zoZ#C^dJHzMuA5W7uHtZS>f|Jt2nlq&nqb#YNz>%!&I!!W2 zNmeM{p|=SJRtS~HID(uOQv-~mCB%$UG-lk~5ovgP+xHoj9+u!arYy(&l~YtS4Cj0R zO4$-&5FEg?Ac?8dxqXRI24nTT$n;^QVq4Ca`siNH+wOcyUyzjMfn7>|k*9 literal 0 HcmV?d00001 diff --git a/assets/icons/cerbos.png b/assets/icons/cerbos.png new file mode 100644 index 0000000000000000000000000000000000000000..e4ea978454906c2fc97128e926e0b33cfdb79488 GIT binary patch literal 14027 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 gV+BX)Dc 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�HZ@Y?XNg0HCcuKw+ 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 N|Nq?)`Rf1*1puy=<+K0* literal 0 HcmV?d00001 diff --git a/charts/airlock/microgateway-cni/4.3.0/.helmignore b/charts/airlock/microgateway-cni/4.3.0/.helmignore new file mode 100644 index 000000000..8561d2892 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/.helmignore @@ -0,0 +1,27 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ + +# Helm unit tests +/tests +/validation diff --git a/charts/airlock/microgateway-cni/4.3.0/Chart.yaml b/charts/airlock/microgateway-cni/4.3.0/Chart.yaml new file mode 100644 index 000000000..f36cc383b --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/Chart.yaml @@ -0,0 +1,43 @@ +annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.3/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway CNI + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI +apiVersion: v2 +appVersion: 4.3.0 +description: A Helm chart for deploying the Airlock Microgateway CNI plugin +home: https://www.airlock.com/en/microgateway +icon: file://assets/icons/microgateway-cni.svg +keywords: +- WAF +- Web Application Firewall +- WAAP +- Web Application and API protection +- OWASP +- Airlock +- Microgateway +- Security +- Filtering +- DevSecOps +- shift left +- CNI +kubeVersion: '>=1.25.0-0' +maintainers: +- email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ +name: microgateway-cni +sources: +- https://github.com/airlock/microgateway +type: application +version: 4.3.0 diff --git a/charts/airlock/microgateway-cni/4.3.0/README.md b/charts/airlock/microgateway-cni/4.3.0/README.md new file mode 100644 index 000000000..2fa8977de --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/README.md @@ -0,0 +1,137 @@ +# Airlock Microgateway CNI + +![Version: 4.3.0](https://img.shields.io/badge/Version-4.3.0-informational?style=flat-square) ![AppVersion: 4.3.0](https://img.shields.io/badge/AppVersion-4.3.0-informational?style=flat-square) + +*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* + + + + + Microgateway + + +Modern application security is embedded in the development workflow and follows DevSecOps paradigms. Airlock Microgateway is the perfect fit for these requirements. It is a lightweight alternative to the Airlock Gateway appliance, optimized for Kubernetes environments. Airlock Microgateway protects your applications and microservices with the tried-and-tested Airlock security features against attacks, while also providing a high degree of scalability. +__This Helm chart is part of Airlock Microgateway. See our [GitHub repo](https://github.com/airlock/microgateway/tree/4.3.0).__ + +### Features +* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. +* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction +* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication +* Content security filters for protecting against known attacks (OWASP Top 10) +* Access control to allow only authenticated users to access the protected services +* API security features like JSON parsing or OpenAPI specification enforcement + +For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. + +## Documentation and links + +Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. + +* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) +* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) +* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) +* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) +* [GitHub](https://github.com/airlock/microgateway) + +# Quick start guide + +The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. + +## Prerequisites +* [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) + +## Deploy Airlock Microgateway CNI +1. Install the CNI Plugin with Helm. + > **Note**: Certain environments such as OpenShift or GKE require non-default configurations when installing the CNI plugin. For the most common setups, values files are provided in the [chart folder](/deploy/charts/airlock-microgateway-cni). + ```bash + # Standard setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # GKE setup + helm install airlock-microgateway-cni -n kube-system oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.0/deploy/charts/airlock-microgateway-cni/gke-values.yaml + kubectl -n kube-system rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + ```bash + # OpenShift setup + helm install airlock-microgateway-cni -n openshift-operators oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' -f https://raw.githubusercontent.com/airlock/microgateway/4.3.0/deploy/charts/airlock-microgateway-cni/openshift-values.yaml + kubectl -n openshift-operators rollout status daemonset -l app.kubernetes.io/instance=airlock-microgateway-cni + ``` + **Important:** On OpenShift, all pods which should be protected by Airlock Microgateway must explicitly reference the Airlock Microgateway CNI NetworkAttachmentDefinition via the annotation `k8s.v1.cni.cncf.io/networks` (see [documentation](https://docs.airlock.com/microgateway/latest/#data/1658483168033.html) for details). + +2. (Recommended) You can verify the correctness of the installation with `helm test`. + ```bash + # Standard and GKE setup + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + helm test airlock-microgateway-cni -n kube-system --logs + helm upgrade airlock-microgateway-cni -n kube-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + ``` + ```bash + # OpenShift setup + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + helm test airlock-microgateway-cni -n openshift-operators --logs + helm upgrade airlock-microgateway-cni -n openshift-operators --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway-cni --version '4.3.0' + ``` + + Consult our [documentation](https://docs.airlock.com/microgateway/latest/#data/1699611533587.html) in case of any installation error. + +## Support + +### Premium support +If you have a paid license, please follow the [premium support process](https://techzone.ergon.ch/support-process). + +### Community support +For the community edition, check our **[Airlock community forum](https://forum.airlock.com/)** for FAQs or register to post your question. +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| affinity | object | `{}` | Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. | +| commonAnnotations | object | `{}` | Annotations to add to all resources. | +| commonLabels | object | `{}` | Labels to add to all resources. | +| config.cniBinDir | string | `"/opt/cni/bin"` | Directory where the CNI plugin binaries reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. | +| config.cniNetDir | string | `"/etc/cni/net.d"` | Directory where the CNI config files reside on the host. This path can either be found in the documentation of your Kubernetes distribution or CNI provider. It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. | +| config.excludeNamespaces | list | `["kube-system"]` | Namespaces for which this CNI plugin should not apply any modifications. | +| config.installMode | string | `"chained"` | Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. | +| config.logLevel | string | `"info"` | Log level for the CNI installer and plugin. | +| fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| image.digest | string | `"sha256:cb165e34a1ab1a903a9f38b741a7d78946470a118640310a41d2af8153d6e409"` | SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). Overrides tag when specified. | +| image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| image.repository | string | `"quay.io/airlock/microgateway-cni"` | Image repository from which to pull the Airlock Microgateway CNI image. | +| image.tag | string | `"4.3.0"` | Image tag to pull. | +| imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | +| multusNetworkAttachmentDefinition.create | bool | `false` | Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. | +| multusNetworkAttachmentDefinition.namespace | string | `"default"` | Namespace in which the NetworkAttachmentDefinition is deployed. Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway-cni". | +| nodeSelector | object | `{"kubernetes.io/os":"linux"}` | NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. | +| podAnnotations | object | `{}` | Annotations to add to all Pods. | +| podLabels | object | `{}` | Labels to add to all Pods. | +| privileged | bool | `false` | Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). | +| rbac.create | bool | `true` | Whether to create RBAC resources which are required for the CNI plugin to function. | +| rbac.createSCCRole | OpenShift | `false` | Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. | +| resources | object | `{"requests":{"cpu":"10m","memory":"100Mi"}}` | Resource restrictions to apply to the CNI installer container. | +| serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | +| tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | + +## License +View the [detailed license terms](https://www.airlock.com/en/airlock-license) for the software contained in this image. +* Decompiling or reverse engineering is not permitted. +* Using any of the deny rules or parts of these filter patterns outside of the image is not permitted. + +Airlock® is a security innovation by [ergon](https://www.ergon.ch/en) + + + + + + + Airlock Secure Access Hub + + diff --git a/charts/airlock/microgateway-cni/4.3.0/gke-values.yaml b/charts/airlock/microgateway-cni/4.3.0/gke-values.yaml new file mode 100644 index 000000000..d6d5c21d1 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/gke-values.yaml @@ -0,0 +1,4 @@ +# values for deploying on GKE + +config: + cniBinDir: "/home/kubernetes/bin" diff --git a/charts/airlock/microgateway-cni/4.3.0/openshift-values.yaml b/charts/airlock/microgateway-cni/4.3.0/openshift-values.yaml new file mode 100644 index 000000000..3b1d6cccd --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/openshift-values.yaml @@ -0,0 +1,15 @@ +# values for deploying on OpenShift + +rbac: + createSCCRole: true + +privileged: true + +multusNetworkAttachmentDefinition: + create: true + namespace: default + +config: + installMode: "standalone" + cniNetDir: "/etc/cni/multus/net.d" + cniBinDir: "/var/lib/cni/bin" diff --git a/charts/airlock/microgateway-cni/4.3.0/questions.yml b/charts/airlock/microgateway-cni/4.3.0/questions.yml new file mode 100644 index 000000000..73ed44d64 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/questions.yml @@ -0,0 +1,18 @@ +questions: + - variable: config.cniNetDir + required: true + type: string + label: CNI Network Configuration Directory + group: "CNI Settings" + description: "Directory where the CNI config files reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your kubernetes host." + - variable: config.cniBinDir + required: true + type: string + label: CNI Plugin Binaries Directory + group: "CNI Settings" + description: "Directory where the CNI plugin binaries reside on the host. This value depends on the kubernetes distribution and interface CNI Provider used. It can be fetched by running `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your kubernetes host." + - variable: config.installMode + required: true + label: CNI Plugin Installation Mode + group: "CNI Settings" + description: "Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers) as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) or in `manual` mode, where no CNI network configuration is written. Please refer to the CNI installation documentation (https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-cni) to correctly setup the CNI Plugin for your environment." diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/NOTES.txt b/charts/airlock/microgateway-cni/4.3.0/templates/NOTES.txt new file mode 100644 index 000000000..e8aa45888 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/NOTES.txt @@ -0,0 +1,3 @@ +Thank you for installing Airlock Microgateway CNI. + +For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway-cni.docsVersion" .}}. diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/_helpers.tpl b/charts/airlock/microgateway-cni/4.3.0/templates/_helpers.tpl new file mode 100644 index 000000000..996491a87 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/_helpers.tpl @@ -0,0 +1,101 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "airlock-microgateway-cni.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Convert an image configuration object into an image ref string. +*/}} +{{- define "airlock-microgateway-cni.image" -}} + {{- if .digest -}} + {{- printf "%s@%s" .repository .digest -}} + {{- else if .tag -}} + {{- printf "%s:%s" .repository .tag -}} + {{- else -}} + {{- printf "%s" .repository -}} + {{- end -}} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 50 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest suffix is 13 characters. +If release name contains chart name it will be used as a full name. +*/}} +{{- define "airlock-microgateway-cni.fullname" -}} +{{- if .Values.fullnameOverride }} +{{- .Values.fullnameOverride | trunc 50 | trimSuffix "-" }} +{{- else }} +{{- $name := default .Chart.Name .Values.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 50 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 50 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "airlock-microgateway-cni.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "airlock-microgateway-cni.labels" -}} +helm.sh/chart: {{ include "airlock-microgateway-cni.chart" . }} +{{ include "airlock-microgateway-cni.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- with .Values.commonLabels }} +{{ toYaml .}} +{{- end }} +{{- end }} + +{{/* +Common labels without component +*/}} +{{- define "airlock-microgateway-cni.labelsWithoutComponent" -}} +{{- $labels := fromYaml (include "airlock-microgateway-cni.labels" .) -}} +{{ unset $labels "app.kubernetes.io/component" | toYaml }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "airlock-microgateway-cni.selectorLabels" -}} +app.kubernetes.io/component: cni-plugin-installer +app.kubernetes.io/instance: {{ .Release.Name }} +app.kubernetes.io/name: {{ include "airlock-microgateway-cni.name" . }} +{{- end }} + +{{/* +Create the name of the service account to use for the CNI Plugin +*/}} +{{- define "airlock-microgateway-cni.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "airlock-microgateway-cni.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + +{{- define "airlock-microgateway-cni.isSemver" -}} +{{- regexMatch `^(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$` . -}} +{{- end -}} + +{{- define "airlock-microgateway-cni.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway-cni.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} + {{- $version := (semver .Chart.AppVersion) -}} + {{- $version.Major }}.{{ $version.Minor -}} +{{- else -}} + {{- print "latest" -}} +{{- end -}} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/clusterrole.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/clusterrole.yaml new file mode 100644 index 000000000..ef88ac783 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/clusterrole.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - pods + verbs: + - get + - list + - watch + - patch +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/clusterrolebinding.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/clusterrolebinding.yaml new file mode 100644 index 000000000..04f87cb0f --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/clusterrolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway-cni.fullname" . }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/configmap.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/configmap.yaml new file mode 100644 index 000000000..b880116ef --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/configmap.yaml @@ -0,0 +1,22 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + plugin-conf.json: |- + { + "type": "{{ include "airlock-microgateway-cni.fullname" . }}", + "debug": {{ eq .Values.config.logLevel "debug" }}, + "logFilePath": "/var/log/{{ include "airlock-microgateway-cni.fullname" . }}.log", + "kubernetes": { + "kubeconfig": "{{ .Values.config.cniNetDir }}/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig", + "excludeNamespaces": {{ toJson .Values.config.excludeNamespaces }} + } + } diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/daemonset.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/daemonset.yaml new file mode 100644 index 000000000..4ba9f2669 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/daemonset.yaml @@ -0,0 +1,136 @@ +apiVersion: apps/v1 +kind: DaemonSet +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway-cni.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: cni-installer + {{- with mustMerge .Values.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 8 }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + containers: + - args: + - --log-level + - "{{ .Values.config.logLevel }}" + env: + - name: CNI_NETWORK_CONFIG + valueFrom: + configMapKeyRef: + key: plugin-conf.json + name: {{ include "airlock-microgateway-cni.fullname" . }} + - name: CNI_BIN_DIR + value: /host/opt/cni/bin + - name: CNI_NET_DIR + value: /host/etc/cni/net.d + - name: KUBECONFIG_FILE_NAME + value: "{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" + - name: INSTALL_MODE + value: {{ .Values.config.installMode }} + - name: KUBERNETES_NODE_NAME + valueFrom: + fieldRef: + fieldPath: spec.nodeName + image: {{ include "airlock-microgateway-cni.image" .Values.image }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + name: cni-installer + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + startupProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 5 + initialDelaySeconds: 3 + periodSeconds: 3 + timeoutSeconds: 3 + readinessProbe: + exec: + command: + - /cni-installer + - probe + failureThreshold: 1 + periodSeconds: 60 + timeoutSeconds: 3 + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: File + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + - mountPath: /run/cni-installer + name: cni-installer-status + hostNetwork: true + priorityClassName: system-node-critical + restartPolicy: Always + securityContext: + fsGroup: 0 + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + serviceAccountName: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + terminationGracePeriodSeconds: 5 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + tolerations: + - effect: NoSchedule + operator: Exists + - key: CriticalAddonsOnly + operator: Exists + - effect: NoExecute + operator: Exists + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir + - emptyDir: {} + name: cni-installer-status diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/network-attachment-definition.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/network-attachment-definition.yaml new file mode 100644 index 000000000..5d657e309 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/network-attachment-definition.yaml @@ -0,0 +1,13 @@ +{{- if .Values.multusNetworkAttachmentDefinition.create -}} +apiVersion: "k8s.cni.cncf.io/v1" +kind: NetworkAttachmentDefinition +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }} + namespace: {{ .Values.multusNetworkAttachmentDefinition.namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/scc-role.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/scc-role.yaml new file mode 100644 index 000000000..862748692 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/scc-role.yaml @@ -0,0 +1,22 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/scc-rolebinding.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/scc-rolebinding.yaml new file mode 100644 index 000000000..ebd02982c --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/scc-rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.rbac.createSCCRole -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway-cni.fullname" . }}-privileged +subjects: +- kind: ServiceAccount + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/serviceaccount.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/serviceaccount.yaml new file mode 100644 index 000000000..3dc8d58ea --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway-cni.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labels" . | nindent 4 }} + {{- with mustMerge .Values.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/tests/rbac.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/tests/rbac.yaml new file mode 100644 index 000000000..744799333 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/tests/rbac.yaml @@ -0,0 +1,64 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" +subjects: +- kind: ServiceAccount + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: tests +rules: +- apiGroups: + - "apps" + resources: + - daemonsets + resourceNames: + - {{ include "airlock-microgateway-cni.fullname" . }} + verbs: + - get + - watch + - list +- apiGroups: + - "" + resources: + - pods + - pods/log + verbs: + - get + - list +{{- if .Values.rbac.createSCCRole }} +- apiGroups: + - security.openshift.io + resourceNames: + - privileged + resources: + - securitycontextconstraints + verbs: + - use +{{- end -}} +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/templates/tests/test-install.yaml b/charts/airlock/microgateway-cni/4.3.0/templates/tests/test-install.yaml new file mode 100644 index 000000000..12d8c8de7 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/templates/tests/test-install.yaml @@ -0,0 +1,103 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "airlock-microgateway-cni.fullname" . }}-test-install" + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway-cni.labelsWithoutComponent" . | nindent 4 }} + app.kubernetes.io/component: test-install + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +spec: + restartPolicy: Never + containers: + - name: test + image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" + securityContext: + allowPrivilegeEscalation: {{ .Values.privileged }} + capabilities: + drop: + - ALL + privileged: {{ .Values.privileged }} + readOnlyRootFilesystem: true + runAsGroup: 0 + runAsNonRoot: false + runAsUser: 0 + seccompProfile: + type: RuntimeDefault + volumeMounts: + - mountPath: /host/opt/cni/bin + name: cni-bin-dir + readOnly: true + - mountPath: /host/etc/cni/net.d + name: cni-net-dir + readOnly: true + command: + - sh + - -c + - | + set -eu + + fail() { + echo "Error: ${1}" + echo "" + echo 'CNI installer logs:' + kubectl logs -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}} -c cni-installer + exit 1 + } + + containsMGWCNIConf() { + cat "${1}" | grep -qe '"type":.*"{{ include "airlock-microgateway-cni.fullname" . }}"' + } + + if ! kubectl rollout status --timeout=60s -n {{ .Release.Namespace }} daemonsets/{{ include "airlock-microgateway-cni.fullname" .}}; then + fail 'CNI DaemonSet rollout did not complete within timeout' + fi + + echo "Checking whether CNI binary was installed" + if ! [ -f "/host/opt/cni/bin/{{ include "airlock-microgateway-cni.fullname" . }}" ]; then + fail 'CNI binary was not installed' + fi + + echo "Checking whether CNI kubeconfig was installed" + if ! [ -f "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}-kubeconfig" ]; then + fail 'CNI kubeconfig was not created' + fi + + echo "Checking whether CNI configuration was written" + case {{ .Values.config.installMode }} in + "chained") + for file in "/host/etc/cni/net.d/"*.conflist; do + if containsMGWCNIConf "${file}"; then + echo "Success" + exit 0 + fi + done + ;; + "standalone") + if containsMGWCNIConf "/host/etc/cni/net.d/{{ include "airlock-microgateway-cni.fullname" . }}.conflist"; then + echo "Success" + exit 0 + fi + ;; + "manual") + echo "- Skipping because we are in 'manual' install mode" + echo "Success" + exit 0 + ;; + esac + + fail 'Configuration for plugin "{{ include "airlock-microgateway-cni.fullname" . }}" was not found' + serviceAccountName: "{{ include "airlock-microgateway-cni.fullname" . }}-tests" + volumes: + - hostPath: + path: "{{ .Values.config.cniBinDir }}" + type: Directory + name: cni-bin-dir + - hostPath: + path: "{{ .Values.config.cniNetDir }}" + type: Directory + name: cni-net-dir +{{- end -}} diff --git a/charts/airlock/microgateway-cni/4.3.0/values.schema.json b/charts/airlock/microgateway-cni/4.3.0/values.schema.json new file mode 100644 index 000000000..e087bd700 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/values.schema.json @@ -0,0 +1,225 @@ +{ + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "nameOverride": { + "type": "string" + }, + "fullnameOverride": { + "type": "string" + }, + "commonLabels": { + "$ref": "#/definitions/StringMap" + }, + "commonAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "imagePullSecrets": { + "type": "array", + "items": { + "type": "object", + "properties": { + "name": { + "type": "string", + "minLength": 1 + } + }, + "required": [ + "name" + ], + "additionalProperties": true + } + }, + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "affinity": { + "type": "object" + }, + "rbac": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "createSCCRole": { + "type": "boolean" + } + }, + "required": [ + "create", + "createSCCRole" + ], + "additionalProperties": false + }, + "privileged": { + "type": "boolean" + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "multusNetworkAttachmentDefinition": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "namespace": { + "type": "string" + } + }, + "required": [ + "create", + "namespace" + ], + "additionalProperties": false + }, + "config": { + "type": "object", + "properties": { + "installMode": { + "type": "string", + "enum": [ + "chained", + "standalone", + "manual" + ] + }, + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + }, + "cniNetDir": { + "type": "string", + "minLength": 1 + }, + "cniBinDir": { + "type": "string", + "minLength": 1 + }, + "excludeNamespaces": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "required": [ + "cniBinDir", + "cniNetDir", + "excludeNamespaces", + "installMode", + "logLevel" + ], + "additionalProperties": false + }, + "tests": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "additionalProperties": false + }, + "global": { + "type": "object" + } + }, + "required": [ + "affinity", + "commonAnnotations", + "commonLabels", + "config", + "fullnameOverride", + "image", + "imagePullSecrets", + "multusNetworkAttachmentDefinition", + "nameOverride", + "nodeSelector", + "podAnnotations", + "podLabels", + "privileged", + "rbac", + "resources", + "serviceAccount", + "tests" + ], + "additionalProperties": false, + "definitions": { + "StringMap": { + "type": "object", + "additionalProperties": { + "type": "string" + } + }, + "Image": { + "type": "object", + "properties": { + "repository": { + "type": "string", + "minLength": 1 + }, + "tag": { + "type": "string" + }, + "digest": { + "type": "string", + "pattern": "^$|^sha256:[a-f0-9]{64}$" + }, + "pullPolicy": { + "type": "string", + "enum": [ + "Always", + "IfNotPresent", + "Never" + ] + } + }, + "required": [ + "digest", + "pullPolicy", + "repository", + "tag" + ], + "additionalProperties": false + } + } +} diff --git a/charts/airlock/microgateway-cni/4.3.0/values.yaml b/charts/airlock/microgateway-cni/4.3.0/values.yaml new file mode 100644 index 000000000..a5bf5dac5 --- /dev/null +++ b/charts/airlock/microgateway-cni/4.3.0/values.yaml @@ -0,0 +1,85 @@ +# -- Allows overriding the name to use instead of "microgateway-cni". +nameOverride: "" +# -- Allows overriding the name to use as full name of resources. +fullnameOverride: "" +# -- Labels to add to all resources. +commonLabels: {} +# -- Annotations to add to all resources. +commonAnnotations: {} +# -- ImagePullSecrets to use when pulling images. +imagePullSecrets: [] +# - name: myRegistryKeySecretName + +# Specifies the Airlock Microgateway CNI image. +image: + # -- Image repository from which to pull the Airlock Microgateway CNI image. + repository: "quay.io/airlock/microgateway-cni" + # -- Image tag to pull. + tag: "4.3.0" + # -- SHA256 image digest to pull (in the format "sha256:7144f7bab3d4c2648d7e59409f15ec52a18006a128c733fcff20d3a4a54ba44a"). + # Overrides tag when specified. + digest: "sha256:cb165e34a1ab1a903a9f38b741a7d78946470a118640310a41d2af8153d6e409" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +# -- Annotations to add to all Pods. +podAnnotations: {} +# -- Labels to add to all Pods. +podLabels: {} +# -- Resource restrictions to apply to the CNI installer container. +resources: + requests: + cpu: 10m + memory: 100Mi +# -- NodeSelector to apply to the CNI DaemonSet in order to only deploy the CNI plugin on specific nodes. +nodeSelector: + kubernetes.io/os: linux +# -- Custom affinity for the DaemonSet to only deploy the CNI plugin on specific nodes. +affinity: {} +# Configures the generation of RBAC Roles and RoleBindings. +rbac: + # -- Whether to create RBAC resources which are required for the CNI plugin to function. + create: true + # -- (OpenShift) Whether to create RBAC resources which allow the CNI installer to use the "privileged" security context constraint. + createSCCRole: false +# -- Whether the DaemonSet should run in privileged mode. Must be enabled for environments which require it for writing files to the host (e.g. OpenShift). +privileged: false +# Configures the generation of the ServiceAccount. +serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" +# Configures the generation of a NetworkAttachmentDefinition for use with Multus CNI (OpenShift) +multusNetworkAttachmentDefinition: + # -- Whether a NetworkAttachmentDefinition CR should be created, which can be used for applying the CNI plugin to Pods. + create: false + # -- Namespace in which the NetworkAttachmentDefinition is deployed. + # Note: If namespace is set to a custom value, referencing the created NetworkAttachmentDefinition from other namespaces + # may not work if Multus namespace isolation is enabled. https://github.com/k8snetworkplumbingwg/multus-cni/blob/v4.0.2/docs/configuration.md#namespace-isolation + namespace: default +# Parameters for the CNI installer configuration. +config: + # -- Whether to install the CNI plugin as a `chained` plugin (default, required with most interface CNI providers), + # as a `standalone` plugin (required for use with Multus CNI, e.g. on OpenShift) + # or in `manual` mode, where no CNI network configuration is written. + installMode: "chained" + # -- Log level for the CNI installer and plugin. + logLevel: info + # -- Directory where the CNI config files reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.confDir}}'` on your Kubernetes node. + cniNetDir: "/etc/cni/net.d" + # -- Directory where the CNI plugin binaries reside on the host. + # This path can either be found in the documentation of your Kubernetes distribution or CNI provider. + # It can also be queried by running the command `crictl info -o go-template --template '{{.config.cni.binDir}}'` on your Kubernetes node. + cniBinDir: "/opt/cni/bin" + # -- Namespaces for which this CNI plugin should not apply any modifications. + excludeNamespaces: + - kube-system +tests: + # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). + # If set to false, `helm test` will not run any tests. + enabled: false diff --git a/charts/airlock/microgateway/4.3.0/.helmignore b/charts/airlock/microgateway/4.3.0/.helmignore new file mode 100644 index 000000000..101ff5ac5 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/.helmignore @@ -0,0 +1,28 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ +# CRDs kustomization.yaml +/crds/kustomization.yaml +# Helm unit tests +/tests +/validation diff --git a/charts/airlock/microgateway/4.3.0/Chart.yaml b/charts/airlock/microgateway/4.3.0/Chart.yaml new file mode 100644 index 000000000..41659f600 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/Chart.yaml @@ -0,0 +1,44 @@ +annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.3/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: microgateway + charts.openshift.io/name: Airlock Microgateway +apiVersion: v2 +appVersion: 4.3.0 +description: A Helm chart for deploying the Airlock Microgateway +home: https://www.airlock.com/en/microgateway +icon: file://assets/icons/microgateway.svg +keywords: +- WAF +- Web Application Firewall +- WAAP +- Web Application and API protection +- OWASP +- Airlock +- Microgateway +- Security +- Filtering +- DevSecOps +- shift left +- control plane +- Operator +kubeVersion: '>=1.25.0-0' +maintainers: +- email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ +name: microgateway +sources: +- https://github.com/airlock/microgateway +type: application +version: 4.3.0 diff --git a/charts/airlock/microgateway/4.3.0/README.md b/charts/airlock/microgateway/4.3.0/README.md new file mode 100644 index 000000000..15ea0012e --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/README.md @@ -0,0 +1,180 @@ +# Airlock Microgateway + +![Version: 4.3.0](https://img.shields.io/badge/Version-4.3.0-informational?style=flat-square) ![AppVersion: 4.3.0](https://img.shields.io/badge/AppVersion-4.3.0-informational?style=flat-square) + +*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* + + + + + Microgateway + + +Modern application security is embedded in the development workflow and follows DevSecOps paradigms. Airlock Microgateway is the perfect fit for these requirements. It is a lightweight alternative to the Airlock Gateway appliance, optimized for Kubernetes environments. Airlock Microgateway protects your applications and microservices with the tried-and-tested Airlock security features against attacks, while also providing a high degree of scalability. +__This Helm chart is part of Airlock Microgateway. See our [GitHub repo](https://github.com/airlock/microgateway/tree/4.3.0).__ + +### Features +* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. +* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction +* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication +* Content security filters for protecting against known attacks (OWASP Top 10) +* Access control to allow only authenticated users to access the protected services +* API security features like JSON parsing or OpenAPI specification enforcement + +For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. + +## Documentation and links + +Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. + +* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) +* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) +* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) +* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) +* [GitHub](https://github.com/airlock/microgateway) + +# Quick start guide + +The instructions below provide a quick start guide. Detailed information are provided in the **[manual](https://docs.airlock.com/microgateway/latest/)**. + +## Prerequisites +* [Airlock Microgateway CNI](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) +* [Airlock Microgateway License](#obtain-airlock-microgateway-license) +* [cert-manager](https://cert-manager.io/) +* [helm](https://helm.sh/docs/intro/install/) (>= v3.8.0) + +In order to use Airlock Microgateway you need a license and the cert-manager. You may either request a community license free of charge or purchase a premium license. +For an easy start in non-production environments, you may deploy the same cert-manager we are using internally for testing. +### Obtain Airlock Microgateway License +1. Either request a community or premium license + * Community license: [airlock.com/microgateway-community](https://airlock.com/en/microgateway-community) + * Premium license: [airlock.com/microgateway-premium](https://airlock.com/en/microgateway-premium) +2. Check your inbox and save the license file microgateway-license.txt locally. + +> See [Community vs. Premium editions in detail](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html) to choose the right license type. +### Deploy cert-manager +```bash +helm repo add jetstack https://charts.jetstack.io +helm install cert-manager jetstack/cert-manager --version '1.15.1' -n cert-manager --create-namespace --set crds.enabled=true --wait +``` + +## Deploy Airlock Microgateway Operator + +> This guide assumes a microgateway-license.txt file is present in the working directory. + +1. Install CRDs and Operator. + ```bash + # Create namespace + kubectl create namespace airlock-microgateway-system + + # Install License + kubectl -n airlock-microgateway-system create secret generic airlock-microgateway-license --from-file=microgateway-license.txt + + # Install Operator (CRDs are included via the standard Helm 3 mechanism, i.e. Helm will handle initial installation but not upgrades) + helm install airlock-microgateway -n airlock-microgateway-system oci://quay.io/airlockcharts/microgateway --version '4.3.0' --wait + ``` + +2. (Recommended) You can verify the correctness of the installation with `helm test`. + ```bash + helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=true --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.0' + helm test airlock-microgateway -n airlock-microgateway-system --logs + helm upgrade airlock-microgateway -n airlock-microgateway-system --set tests.enabled=false --reuse-values oci://quay.io/airlockcharts/microgateway --version '4.3.0' + ``` + +### Upgrading CRDs + +The `helm install/upgrade` command currently does not support upgrading CRDs that already exist in the cluster. +CRDs should instead be manually upgraded before upgrading the Operator itself via the following command: +```bash +kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=4.3.0 --server-side --force-conflicts +``` + +**Note**: Certain GitOps solutions such as e.g. Argo CD or Flux CD have their own mechanisms for automatically upgrading CRDs included with Helm charts. + +## Support + +### Premium support +If you have a paid license, please follow the [premium support process](https://techzone.ergon.ch/support-process). + +### Community support +For the community edition, check our **[Airlock community forum](https://forum.airlock.com/)** for FAQs or register to post your question. +## Values + +| Key | Type | Default | Description | +|-----|------|---------|-------------| +| commonAnnotations | object | `{}` | Annotations to add to all resources. | +| commonLabels | object | `{}` | Labels to add to all resources. | +| crds.skipVersionCheck | bool | `false` | Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster when performing a "helm install/upgrade". | +| dashboards.config.grafana.dashboardLabel.name | string | `"grafana_dashboard"` | Name of the label that lets Grafana identify ConfigMaps that represent dashboards. | +| dashboards.config.grafana.dashboardLabel.value | string | `"1"` | Value of the label that lets Grafana identify ConfigMaps that represent dashboards. | +| dashboards.config.grafana.folderAnnotation.name | string | `"grafana_folder"` | Name of the annotation containing the folder name to file dashboards into. | +| dashboards.config.grafana.folderAnnotation.value | string | `"Airlock Microgateway"` | Name of the folder dashboards are filed into within the Grafana UI. | +| dashboards.create | bool | `false` | Whether to create any ConfigMaps containing Grafana dashboards to import. | +| dashboards.instances.blockLogs.create | bool | `true` | Whether to create the block logs dashboard. | +| dashboards.instances.blockMetrics.create | bool | `true` | Whether to create the block metrics dashboard. | +| dashboards.instances.license.create | bool | `true` | Whether to create the license dashboard. | +| dashboards.instances.overview.create | bool | `true` | Whether to create the overview dashboard. | +| engine.image.digest | string | `"sha256:f442143294f3138965c9fa2734cafd39ebebe8e289600332b12f8a59c23dd9ef"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | +| engine.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| engine.image.repository | string | `"quay.io/airlock/microgateway-engine"` | Image repository from which to pull the Airlock Microgateway Engine image. | +| engine.image.tag | string | `"4.3.0"` | Image tag to pull. | +| engine.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Engine container. | +| engine.sidecar.podMonitor.create | bool | `false` | Whether to create a PodMonitor resource for monitoring. | +| engine.sidecar.podMonitor.labels | object | `{}` | Labels to add to the PodMonitor. | +| fullnameOverride | string | `""` | Allows overriding the name to use as full name of resources. | +| imagePullSecrets | list | `[]` | ImagePullSecrets to use when pulling images. | +| license.secretName | string | `"airlock-microgateway-license"` | Name of the secret containing the "microgateway-license.txt" key. | +| nameOverride | string | `""` | Allows overriding the name to use instead of "microgateway". | +| networkValidator.image.digest | string | `"sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6"` | SHA256 image digest to pull (in the format "sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6"). Overrides tag when specified. | +| networkValidator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| networkValidator.image.repository | string | `"cgr.dev/chainguard/busybox"` | Image repository from which to pull the busybox image for the Airlock Microgateway Network Validator init-container. | +| networkValidator.image.tag | string | `""` | Image tag to pull. | +| operator.affinity | object | `{}` | Custom affinity to apply to the operator Deployment. Used to influence the scheduling. | +| operator.config.logLevel | string | `"info"` | Operator application log level. | +| operator.image.digest | string | `"sha256:dc6f0f9a11d0336c10f6b8a5c7f64d98ac91bd90c49aa1dc4fe7b68cfdea8217"` | SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). Overrides tag when specified. | +| operator.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| operator.image.repository | string | `"quay.io/airlock/microgateway-operator"` | Image repository from which to pull the Airlock Microgateway Operator image. | +| operator.image.tag | string | `"4.3.0"` | Image tag to pull. | +| operator.nodeSelector | object | `{}` | Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. | +| operator.podAnnotations | object | `{}` | Annotations to add to all Pods. | +| operator.podLabels | object | `{}` | Labels to add to all Pods. | +| operator.rbac.create | bool | `true` | Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. | +| operator.replicaCount | int | `2` | Number of replicas for the operator Deployment. | +| operator.resources | object | `{}` | Resource restrictions to apply to the operator container. | +| operator.serviceAccount.annotations | object | `{}` | Annotations to add to the ServiceAccount. | +| operator.serviceAccount.create | bool | `true` | Whether a ServiceAccount should be created. | +| operator.serviceAccount.name | string | `""` | Name of the ServiceAccount to use. If not set and create is true, a name is generated using the fullname template. | +| operator.serviceAnnotations | object | `{}` | Annotations to add to the Service. | +| operator.serviceLabels | object | `{}` | Labels to add to the Service. | +| operator.serviceMonitor.create | bool | `false` | Whether to create a ServiceMonitor resource for monitoring. | +| operator.serviceMonitor.labels | object | `{}` | Labels to add to the ServiceMonitor. | +| operator.tolerations | list | `[]` | Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. | +| operator.updateStrategy | object | `{"type":"RollingUpdate"}` | Specifies the operator update strategy. | +| operator.watchNamespaceSelector | object | `{}` | Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. Please note that this feature requires a Premium license. | +| operator.watchNamespaces | list | `[]` | Allows to restrict the operator to specific namespaces, depending on your needs. For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. Please note that this feature requires a Premium license. | +| sessionAgent.image.digest | string | `"sha256:579dfded99145f9c2c1491ff1aeccb08721d63239a8b7f61bb9f455e17e968b2"` | SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). Overrides tag when specified. | +| sessionAgent.image.pullPolicy | string | `"IfNotPresent"` | Pull policy for this image. | +| sessionAgent.image.repository | string | `"quay.io/airlock/microgateway-session-agent"` | Image repository from which to pull the Airlock Microgateway Session Agent image. | +| sessionAgent.image.tag | string | `"4.3.0"` | Image tag to pull. | +| sessionAgent.resources | object | `{}` | Resource restrictions to apply to the Airlock Microgateway Session Agent container. | +| tests.enabled | bool | `false` | Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). If set to false, `helm test` will not run any tests. | + +## License +View the [detailed license terms](https://www.airlock.com/en/airlock-license) for the software contained in this image. +* Decompiling or reverse engineering is not permitted. +* Using any of the deny rules or parts of these filter patterns outside of the image is not permitted. + +Airlock® is a security innovation by [ergon](https://www.ergon.ch/en) + + + + + + + Airlock Secure Access Hub + + diff --git a/charts/airlock/microgateway/4.3.0/app-readme.md b/charts/airlock/microgateway/4.3.0/app-readme.md new file mode 100644 index 000000000..e32cac025 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/app-readme.md @@ -0,0 +1,28 @@ +# Airlock Microgateway + +*Airlock Microgateway is a Kubernetes native WAAP (Web Application and API Protection) solution to protect microservices.* + +## Features +* Kubernetes native integration with its Operator, Custom Resource Definitions, hot-reload, automatic sidecar injection. +* Reverse proxy functionality with request routing rules, TLS termination and remote IP extraction +* Using native Envoy HTTP filters like Lua scripting, RBAC, ext_authz, JWT authentication +* Content security filters for protecting against known attacks (OWASP Top 10) +* Access control to allow only authenticated users to access the protected services +* API security features like JSON parsing or OpenAPI specification enforcement + +For a list of all features, view the **[comparison of the community and premium edition](https://docs.airlock.com/microgateway/latest/#data/1675772882054.html)**. + +## Requirements +* [Airlock Microgateway CNI Helm Chart](https://artifacthub.io/packages/helm/airlock-microgateway-cni/microgateway-cni) (Also available as Rancher Chart) +* [Airlock Microgateway License](https://github.com/airlock/microgateway?tab=readme-ov-file#obtain-airlock-microgateway-license) (After obtaining the license install it according to the [documentation](https://github.com/airlock/microgateway?tab=readme-ov-file#deploy-airlock-microgateway-operator)) +* [cert-manager](https://cert-manager.io/docs/installation/) + +## Documentation and links + +Check the official documentation at **[docs.airlock.com](https://docs.airlock.com/microgateway/latest/)** or the product website at **[airlock.com/microgateway](https://www.airlock.com/en/microgateway)**. The links below point out the most interesting documentation sites when starting with Airlock Microgateway. + +* [Getting Started](https://docs.airlock.com/microgateway/latest/#data/1660804708742.html) +* [System Architecture](https://docs.airlock.com/microgateway/latest/#data/1660804709650.html) +* [Installation](https://docs.airlock.com/microgateway/latest/#data/1660804708637.html) +* [Troubleshooting](https://docs.airlock.com/microgateway/latest/#data/1659430054787.html) +* [GitHub](https://github.com/airlock/microgateway) \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml new file mode 100644 index 000000000..5c6215c90 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/accesscontrols.microgateway.airlock.com.yaml @@ -0,0 +1,124 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: accesscontrols.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: AccessControl + listKind: AccessControlList + plural: accesscontrols + singular: accesscontrol + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: AccessControl specifies the options to perform access control with a Microgateway Engine container. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specifies how the Airlock Microgateway Engine performs access control. + properties: + policies: + description: Policies configures access control policies. + items: + properties: + authorization: + description: Authorization configures how requests are authorized. An empty object value {} disables authorization. + properties: + authentication: + description: Authentication specifies that clients need to be authenticated with the provided method. + properties: + oidc: + description: OIDC configures client authentication using OpenID Connect. + properties: + oidcRelyingPartyRef: + description: OIDCRelyingPartyRef configures how the Airlock Microgateway Engine interacts with the OpenID provider. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - oidcRelyingPartyRef + type: object + type: object + type: object + identityPropagation: + description: IdentityPropagation configures how the authenticated user's identity is communicated to the protected application. + properties: + actions: + description: Actions specifies the propagation actions. + items: + properties: + identityPropagationRef: + description: IdentityPropagationRef selects an IdentityPropagation to apply. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - identityPropagationRef + type: object + type: array + onFailure: + description: |- + OnFailure configures what should happen, if an identity propagation fails. Meaning of the possible values: + _Pass_: The request should be forwarded to the upstream, without including the information from the failed identity propagations. + enum: + - Pass + type: string + required: + - actions + - onFailure + type: object + required: + - authorization + type: object + maxItems: 1 + minItems: 1 + type: array + required: + - policies + type: object + required: + - spec + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml new file mode 100644 index 000000000..05214f023 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/contentsecurities.microgateway.airlock.com.yaml @@ -0,0 +1,139 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: contentsecurities.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: ContentSecurity + listKind: ContentSecurityList + plural: contentsecurities + singular: contentsecurity + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: ContentSecurity specifies the options to secure an upstream web application with a Microgateway Engine container. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specifies the options to secure an upstream web application with a Microgateway Engine container. + properties: + apiProtection: + description: |- + APIProtection defines the relevant configurations to protect APIs. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + graphQLRef: + description: |- + GraphQLRef selects the relevant GraphQL configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + openAPIRef: + description: |- + OpenAPIRef selects the relevant OpenAPI configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + filter: + description: |- + Filter defines the set of filters, e.g. Airlock Deny Rules, to be applied to incoming requests + to protect against various attack patterns. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + denyRulesRef: + description: |- + DenyRulesRef selects the relevant DenyRules configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + headerRewritesRef: + description: |- + HeaderRewritesRef selects the relevant HeaderRewrites. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + limitsRef: + description: |- + LimitsRef selects the relevant Limits configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + parserRef: + description: |- + ParserRef selects the relevant Parser configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml new file mode 100644 index 000000000..906353c0a --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/denyrules.microgateway.airlock.com.yaml @@ -0,0 +1,1804 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: denyrules.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: DenyRules + listKind: DenyRulesList + plural: denyrules + singular: denyrules + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: |- + DenyRules configures request filtering using Airlock built-in and custom deny rules. + Deny rules establish a negative security model. They define prohibited patterns which, when a match is found in a request, lead to it being blocked from reaching the upstream web application. + To handle possible false positives, lower the security level or define fine-granular deny rule exceptions + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired deny rules behavior. + properties: + request: + description: Request configures deny rules for downstream requests. + properties: + builtIn: + description: BuiltIn configures the built-in deny rules. + properties: + exceptions: + description: Exceptions allows to define exceptions for specific requests and deny rules. + items: + description: |- + DenyRulesException defines an exception for deny rules. Exceptions may be defined by any or a combination of the following elements: blockedData (the request data causing a block) or requestConditions (properties of a request without taking into consideration the reason why a request has been blocked). + At least one of blockedData and requestConditions must be set. + properties: + blockedData: + description: BlockedData defines an exception based on the request data causing the block. + properties: + graphQL: + description: |- + GraphQL defines an exception based on a blocked GraphQL query. + Only one of parameter, header, path, pathSegment, json or graphQL can be set. + properties: + argument: + description: |- + Argument defines an argument of a field of the GraphQL query. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + field: + description: |- + Field defines a field of the GraphQL query. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: |- + Value defines the value of an argument of the GraphQL query. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + header: + description: |- + Header defines an exception based on a blocked header. + Only one of parameter, header, path, pathSegment, json or graphQL can be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + json: + description: |- + JSON defines an exception based on a blocked JSON property. + Only one of parameter, header, path, pathSegment, json or graphQL can be set. + properties: + jsonPath: + description: |- + JSONPath defines the JSONPath pattern to match the path within the JSON. + Expressions in JSONPath i.e. `?(expr)` are not supported. + minLength: 1 + type: string + key: + description: |- + Key defines the key of the JSON property. + At most one of key and value can be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: |- + Value defines the value of the JSON property. + At most one of key and value can be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + parameter: + description: |- + Parameter defines an exception based on a blocked parameter. + Only one of parameter, header, path, pathSegment, json or graphQL can be set. + properties: + name: + description: Name defines the name of a parameter. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + source: + default: Any + description: Source defines the source of the parameter. + enum: + - Query + - Post + - Any + type: string + value: + description: Value defines the value of a parameter. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + path: + description: |- + Path defines an exception based on the blocked path. + Only one of parameter, header, path, pathSegment, json or graphQL can be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + pathSegment: + description: |- + PathSegment defines an exception based on a blocked path segment. + Only one of parameter, header, path, pathSegment, json or graphQL can be set. + properties: + segments: + description: Segments defines the position of a segment within the path. + properties: + index: + description: Index specifies an exact path segment position by index (0-based). + minimum: 0 + type: integer + type: object + value: + description: Value defines the value of a path segment. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + type: object + requestConditions: + description: RequestConditions defines an exception based on a property of a request without taking into consideration the reason why a request has been blocked. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + ruleKeys: + description: RuleKeys restricts the exception to a set of deny rules. + items: + description: |- + A deny rule name can be any of the following values: + ENCODING | + EXPLOIT | + HPP | + HTML | + IDOR | + LDAP | + NOSQL | + OGNL | + PHP | + PROTOCOL | + SANITY | + SCANNING | + SQL | + TEMPLATE | + UNIXCMD | + WINCMD | + XSS + enum: + - ENCODING + - EXPLOIT + - HPP + - HTML + - IDOR + - LDAP + - NOSQL + - OGNL + - PHP + - PROTOCOL + - SANITY + - SCANNING + - SQL + - TEMPLATE + - UNIXCMD + - WINCMD + - XSS + type: string + minItems: 1 + type: array + type: object + type: array + overrides: + description: Overrides allows to override the builtIn settings for specific deny rules. + items: + description: DenyRulesOverride allows to override the builtIn settings for specific deny rules. + properties: + conditions: + description: Conditions select which built-in deny rules' settings will be adjusted. + properties: + ruleKeys: + description: RuleKeys is a list of built-in deny rule names. + items: + description: |- + A deny rule name can be any of the following values: + ENCODING | + EXPLOIT | + HPP | + HTML | + IDOR | + LDAP | + NOSQL | + OGNL | + PHP | + PROTOCOL | + SANITY | + SCANNING | + SQL | + TEMPLATE | + UNIXCMD | + WINCMD | + XSS + enum: + - ENCODING + - EXPLOIT + - HPP + - HTML + - IDOR + - LDAP + - NOSQL + - OGNL + - PHP + - PROTOCOL + - SANITY + - SCANNING + - SQL + - TEMPLATE + - UNIXCMD + - WINCMD + - XSS + type: string + minItems: 1 + type: array + types: + description: Types defines the type of attributes the override should be applied on. If Types are defined without any RuleKeys the override is applied to all deny rules. + items: + description: |- + A deny rule override type name can be any of the following values: + Header | + Parameter | + Path | + JSON | + GraphQL + enum: + - Header + - Parameter + - Path + - PathSegment + - JSON + - GraphQL + type: string + minItems: 0 + type: array + type: object + settings: + description: Settings override the corresponding properties for the selected rules. + properties: + level: + description: Level specifies the filter strength. + enum: + - Unfiltered + - Basic + - Standard + - Strict + type: string + threatHandlingMode: + description: ThreatHandlingMode specifies how threats should be handled. + enum: + - Block + - LogOnly + type: string + type: object + type: object + type: array + settings: + description: Settings contains the keys which will be adjusted. + properties: + level: + default: Standard + description: Level represents a set of deny rules with different filter strengths. + enum: + - Unfiltered + - Basic + - Standard + - Strict + type: string + threatHandlingMode: + default: Block + description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. + enum: + - Block + - LogOnly + type: string + type: object + type: object + custom: + description: Custom allows configuring additional deny rules. + properties: + rules: + description: Rules defines list of additional deny rules. + items: + properties: + blockData: + description: BlockData specifies the request data which should cause a block. + properties: + graphQL: + description: |- + GraphQL specifies to block requests containing a matching GraphQL property. + At least one of field, argument and value must be set. + properties: + argument: + description: |- + Argument defines an argument of a field of the GraphQL query. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + field: + description: |- + Field defines a field of the GraphQL query. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: |- + Value defines the value of an argument of the GraphQL query. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + header: + description: |- + Header specifies to block requests containing a matching header. + Only one of parameter, header, path, pathSegment or json can be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + json: + description: |- + JSON specifies to block requests containing a matching JSON property in the body. + Only one of parameter, header, path, pathSegment or json can be set. + properties: + key: + description: Key defines the key of a JSON object. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a JSON object. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + parameter: + description: |- + Parameter specifies to block requests containing a matching parameter. + Only one of parameter, header, path, pathSegment or json can be set. + properties: + name: + description: Name defines the name of a parameter. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a parameter. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + path: + description: |- + Path specifies to block requests with a matching path. + Only one of parameter, header, path, pathSegment or json can be set. + properties: + matcher: + description: Matcher specifies which path to block. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + pathSegment: + description: |- + PathSegment specifies to block requests containing a matching path segment. + Only one of parameter, header, path, pathSegment or json can be set. + properties: + segments: + description: |- + Segments restricts which path segments are filtered by this rule. + If not specified, all segments of a path are filtered. + properties: + index: + description: Index restricts the rule to the path segment at this index (0-based). + minimum: 0 + type: integer + type: object + value: + description: Value specifies which path segment values to block. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + required: + - value + type: object + type: object + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + ruleKey: + description: RuleKey defines a technical key for the deny rule. Must be unique. + minLength: 1 + pattern: ^[A-Z][A-Z0-9_]*$ + type: string + threatHandlingMode: + default: Block + description: ThreatHandlingMode specifies how threats should be handled when a deny rule matches. + enum: + - Block + - LogOnly + type: string + required: + - blockData + - ruleKey + type: object + type: array + x-kubernetes-list-map-keys: + - ruleKey + x-kubernetes-list-type: map + type: object + type: object + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml new file mode 100644 index 000000000..07ba25df5 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/envoyclusters.microgateway.airlock.com.yaml @@ -0,0 +1,58 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: envoyclusters.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: EnvoyCluster + listKind: EnvoyClusterList + plural: envoyclusters + singular: envoycluster + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: EnvoyCluster is an additional Envoy Cluster resource which is added to those defined by the Airlock Microgateway. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired additional Envoy cluster. + properties: + value: + description: Value defines the Envoy Cluster which is added to those configured by the Airlock Microgateway. + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml new file mode 100644 index 000000000..cc09fbbb1 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/envoyconfigurations.microgateway.airlock.com.yaml @@ -0,0 +1,185 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: envoyconfigurations.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: EnvoyConfiguration + listKind: EnvoyConfigurationList + plural: envoyconfigurations + singular: envoyconfiguration + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.status + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: |- + EnvoyConfiguration is the Schema for the envoyconfigurations API + {{% notice warning %}} EnvoyConfiguration resources may contain sensitive information and thus RBAC permissions should be granted with care. {{% /notice %}} + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: EnvoyConfigurationSpec defines the desired state of EnvoyConfiguration + properties: + envoyResources: + properties: + clusters: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + endpoints: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + extensions: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + listeners: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + routes: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + runtimes: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + scopedRoutes: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + secrets: + items: + type: object + x-kubernetes-preserve-unknown-fields: true + type: array + type: object + envoyResourcesRaw: + description: |- + EnvoyResourcesRaw defines the desired state for each resource type. The resources are stored as zstd compressed JSON bytes. + For debugging purposes, the resources can be inspected with the following command: `kubectl get envoyconfiguration -ojsonpath='{.spec.envoyResourcesRaw}' | base64 -d | zstd -d | jq` + format: byte + type: string + nodeID: + description: '**Deprecated:** This field is now ignored as NodeID is always derived from the resource name.' + type: string + type: object + status: + description: EnvoyConfigurationStatus defines the observed state of EnvoyConfiguration + properties: + conditions: + items: + properties: + lastTransitionTime: + description: Last time the condition transitioned from one status to another. + format: date-time + type: string + message: + description: A human-readable message indicating details about the transition. + type: string + reason: + description: The reason for the condition's last transition. + type: string + status: + description: Status of the condition, one of True, False, Unknown. + type: string + type: + description: Type of EnvoyConfiguration condition. + type: string + required: + - status + - type + type: object + type: array + status: + type: string + xds: + properties: + resourceTypes: + additionalProperties: + description: XdsResourceTypeSyncStatus defines the sync status of xDS for a specific resource type + properties: + errorMessage: + description: ErrorMessage defines an optional message why the currently served resources of this resource type are rejected by the client. + type: string + resources: + additionalProperties: + description: XdsResourceStatus defines the status of xDS for a specific resource + properties: + version: + description: Version defines the version which is currently served for this resource. + type: string + required: + - version + type: object + description: Resources defines the resources which are currently served for this resource type. + type: object + status: + description: Status defines the current sync status of this resource type. + type: string + version: + description: Version defines the version which is currently served for this resource type. + type: string + required: + - resources + - status + - version + type: object + description: ResourceTypes defines the sync statuses for each resource type. + type: object + version: + description: Version defines the version of the underlying xDS snapshot. + type: integer + required: + - version + type: object + required: + - status + - xds + type: object + type: object + served: true + storage: true + subresources: + status: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml new file mode 100644 index 000000000..d6eb787ab --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/envoyhttpfilters.microgateway.airlock.com.yaml @@ -0,0 +1,58 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: envoyhttpfilters.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: EnvoyHTTPFilter + listKind: EnvoyHTTPFilterList + plural: envoyhttpfilters + singular: envoyhttpfilter + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: EnvoyHTTPFilter is an additional Envoy HTTP Filter resource which is added to those defined by the Airlock Microgateway. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired additional Envoy HTTP filter. + properties: + value: + description: Value defines the HTTP filter which is added to those configured by the Airlock Microgateway. + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml new file mode 100644 index 000000000..77f8991e6 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/graphqls.microgateway.airlock.com.yaml @@ -0,0 +1,88 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: graphqls.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: GraphQL + listKind: GraphQLList + plural: graphqls + singular: graphql + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: GraphQL contains the configuration for the GraphQL specification. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired GraphQL specification. + properties: + settings: + description: Settings defines the settings to configure GraphQL. + properties: + allowIntrospection: + default: true + description: AllowIntrospection specifies if the introspection system is exposed. + type: boolean + allowMutations: + default: true + description: AllowMutations specifies if mutations are allowed. + type: boolean + schema: + description: Specifies the GraphQL schema. + properties: + source: + description: Source specifies the GraphQL schema to be enforced. + properties: + configMapRef: + description: ConfigMapRef references the configmap by its name containing the well-known key 'schema.graphql'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + required: + - source + type: object + threatHandlingMode: + default: Block + description: ThreatHandlingMode specifies how threats should be handled. + enum: + - Block + - LogOnly + type: string + type: object + type: object + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml new file mode 100644 index 000000000..8fd43dc3a --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/headerrewrites.microgateway.airlock.com.yaml @@ -0,0 +1,759 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: headerrewrites.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: HeaderRewrites + listKind: HeaderRewritesList + plural: headerrewrites + singular: headerrewrites + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: HeaderRewrites is the Schema for the headerrewrites API + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired header rewriting behavior. + properties: + request: + description: Request defines manipulations on upstream request headers. + properties: + add: + description: Add defines which request headers will be added before forwarding to the upstream. + properties: + custom: + description: |- + Custom allows configuring additional upstream request headers. + Add selected headers. + items: + properties: + headers: + description: Headers to add. + items: + description: HeaderRewritesHeader specifies a header with a particular value + properties: + name: + description: Name defines the name of a header. + minLength: 1 + type: string + value: + description: Value defines the value of a header. + type: string + required: + - name + - value + type: object + minItems: 1 + type: array + mode: + default: AddIfAbsent + description: Mode defines the header addition strategy. + enum: + - AddIfAbsent + - OverwriteOrAdd + type: string + name: + description: Name describing the configured operation. + minLength: 1 + type: string + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + allow: + description: |- + Allow defines which request headers will be forwarded to the upstream. + This can either be allHeaders or matchingHeaders. + Default: matchingHeaders: {...} + properties: + allHeaders: + description: AllHeaders specifies that all request headers should be forwarded. + type: object + matchingHeaders: + description: MatchingHeaders specifies which request headers should be forwarded. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream request headers. + properties: + standardHeaders: + default: true + description: StandardHeaders defines whether the request headers which are forwarded to the upstream will be restricted to a set of common request headers. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream request headers. + items: + properties: + headers: + description: Headers to allow. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured operation. Must be unique. + minLength: 1 + type: string + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + remove: + description: Remove defines which request headers will be removed before forwarding to the upstream. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream request headers. + properties: + alternativeForwardedHeaders: + default: true + description: |- + AlternativeForwardedHeaders removes downstream request headers which could potentially + be abused to alter the upstream's view of the remote connection. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream request headers. + items: + properties: + headers: + description: Headers to remove. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured operation. Must be unique. + minLength: 1 + type: string + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + response: + description: Response defines manipulations on upstream response headers. + properties: + add: + description: Add defines which response headers will be added before forwarding to the downstream. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream response headers. + properties: + csp: + default: true + description: |- + CSP sets a content security policy which allows only same-origin requests except for images + if the 'Content-Security-Policy' header is not set by the upstream. + type: boolean + featurePolicy: + default: false + description: |- + FeaturePolicy sets a feature policy which prevents cross-origin use of several browser features + if the 'Feature-Policy' header is not set by the upstream. + **Deprecated:** Use permissionsPolicy instead. + type: boolean + hsts: + default: true + description: HSTS enforces the use of HTTPS if the 'Strict-Transport-Security' header is not already set by the upstream. + type: boolean + hstsPreload: + default: false + description: HSTSPreload enforces the use of HTTPS including for subdomains and enables HSTS preload. + type: boolean + permissionsPolicy: + default: true + description: |- + PermissionsPolicy sets a permissions policy which prevents cross-origin use of several browser features + if the 'Permissions-Policy' header is not set by the upstream. + type: boolean + referrerPolicy: + default: true + description: |- + ReferrerPolicy ensures that no 'Referer' header is sent for cross-origin requests + if the 'Referrer-Policy' header is not set by the upstream. + type: boolean + xContentTypeOptions: + default: true + description: XContentTypeOptions sets 'X-Content-Type-Options' to 'nosniff' if it is not set by the upstream. + type: boolean + xFrameOptions: + default: true + description: XFrameOptions sets 'X-Frame-Options' to SAMEORIGIN if it is not set by the upstream. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream response headers. + items: + properties: + headers: + description: Headers to add. + items: + description: HeaderRewritesHeader specifies a header with a particular value + properties: + name: + description: Name defines the name of a header. + minLength: 1 + type: string + value: + description: Value defines the value of a header. + type: string + required: + - name + - value + type: object + minItems: 1 + type: array + mode: + default: AddIfAbsent + description: Mode defines the header addition strategy. + enum: + - AddIfAbsent + - OverwriteOrAdd + type: string + name: + description: Name describing the configured operation. + minLength: 1 + type: string + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + allow: + description: |- + Allow defines which response headers will be forwarded to the downstream. + This can either be allHeaders or matchingHeaders. + Default: allHeaders: {} + properties: + allHeaders: + description: AllHeaders specifies that all response headers should be forwarded. + type: object + matchingHeaders: + description: MatchingHeaders specifies which response headers should be forwarded. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream response header. + properties: + standardHeaders: + default: false + description: StandardHeaders defines whether the response headers which are forwarded to the downstream will be restricted to a set of common response headers. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream response headers. + items: + properties: + headers: + description: Headers to allow. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured operation. Must be unique. + minLength: 1 + type: string + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + remove: + description: Remove defines which response headers will be removed before forwarding to the downstream. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined upstream response headers. + properties: + auth: + description: Auth defines the categories of headers concerning authentication. + properties: + basic: + default: false + description: Basic removes upstream response headers that advise clients to authenticate with Basic Authentication. + type: boolean + negotiate: + default: true + description: Negotiate removes upstream response headers that advise clients to authenticate with Negotiate. + type: boolean + ntlm: + default: true + description: |- + NTLM removes upstream response headers that advise clients to authenticate with NTLM. + By default, these headers are removed, because NTLM pass-through is not supported. + type: boolean + type: object + informationLeakage: + description: InformationLeakage defines the categories of headers concerning information leakage. + properties: + application: + default: true + description: Application removes upstream response headers that leak information about the deployed software. + type: boolean + server: + default: true + description: Server removes upstream response headers that leak information about the server. + type: boolean + type: object + permissiveCors: + default: true + description: PermissiveCORS removes upstream response headers for CORS (Cross-Origin Resource Sharing) which have no restrictions and therefore reduce client-side security. + type: boolean + type: object + custom: + description: Custom allows configuring additional upstream response headers. + items: + properties: + headers: + description: Headers to remove. + items: + description: |- + HeaderMatcher defines a matcher for an HTTP header. + At least one of name and value must be set. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + minItems: 1 + type: array + name: + description: Name describing the configured remove operation. Must be unique. + minLength: 1 + type: string + required: + - headers + - name + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + type: object + type: object + settings: + description: Settings configures the HeaderRewrites filter. + properties: + operationalMode: + default: Production + description: OperationalMode defines the behavior of the filter. In integration mode more information is logged about the requests and responses. + enum: + - Production + - Integration + type: string + type: object + type: object + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml new file mode 100644 index 000000000..8ff36ad33 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/identitypropagations.microgateway.airlock.com.yaml @@ -0,0 +1,108 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: identitypropagations.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: IdentityPropagation + listKind: IdentityPropagationList + plural: identitypropagations + singular: identitypropagation + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: IdentityPropagation specifies the desired identity propagation. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired identity propagation. + properties: + header: + description: Header configures identity propagation via a request header. + properties: + name: + description: Name of the header to set. + minLength: 1 + type: string + value: + description: Value to propagate to the application. + properties: + source: + description: Source from which to extract the value. + properties: + metadata: + description: Metadata specifies to extract a value from an Envoy dynamic filter metadata key. + properties: + key: + description: Key specifies the metadata key from which to load the value, e.g. `some_payload.aud`. + minLength: 1 + type: string + namespace: + description: Namespace specifies the metadata namespace within which the lookup should be performed, e.g. `envoy.filters.http.jwt_authn`. + minLength: 1 + type: string + required: + - key + - namespace + type: object + oidc: + description: OIDC specifies to extract a value from the result of an OpenID Connect flow. + properties: + idToken: + description: IDToken specifies to extract the value from the OpenID Connect ID Token. + properties: + claim: + description: Claim selects the JWT claim from which to extract the value. + minLength: 1 + type: string + required: + - claim + type: object + required: + - idToken + type: object + type: object + required: + - source + type: object + required: + - name + - value + type: object + type: object + required: + - spec + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml new file mode 100644 index 000000000..f807994db --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/limits.microgateway.airlock.com.yaml @@ -0,0 +1,651 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: limits.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: Limits + listKind: LimitsList + plural: limits + singular: limits + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: Limits contains the configuration for limits. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired limits behavior. + properties: + request: + description: Request defines the limits for requests. + properties: + limited: + description: Limited enables limits on request scope. + properties: + exceptions: + description: Exceptions defines limit exceptions. + items: + description: LimitsException defines an exception for limits. + properties: + length: + description: Length defines an exception for length limits based on the data element exceeding the limit. + properties: + graphQL: + description: GraphQL defines a field, argument or value length limit exception for a GraphQL query. + properties: + argument: + description: |- + Argument restricts the exception to GraphQL queries with a matching argument of a field. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + field: + description: |- + Field restricts the exception to GraphQL queries with a matching field. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: |- + Value restricts the exception to GraphQL queries with a matching argument value. + At least one of field, argument and value must be set. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + json: + description: JSON defines a key and value length limit exception for a JSON property. + properties: + jsonPath: + description: |- + JSONPath restricts the exception to JSON properties with a matching JSONPath. + Expressions in JSONPath i.e. `?(expr)` are not supported. + minLength: 1 + type: string + required: + - jsonPath + type: object + parameter: + description: Parameter defines a name and value length limit exception for a parameter. + properties: + name: + description: Name restricts the exception to parameters with a matching name. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + source: + default: Any + description: Source restricts the exception to parameters of this kind. + enum: + - Query + - Post + - Any + type: string + required: + - name + type: object + type: object + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this exception to apply. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + type: object + type: array + general: + description: General defines general request limits. + properties: + bodySize: + anyOf: + - type: integer + - type: string + default: 100Mi + description: BodySize limits the total size of the request body. It specifies the number of bytes (0 = unlimited). This limit is effective for any request not processed by one of the content parsers (e.g. json) as configured in the Parser CRD. **Note** This limit does not apply to WebSocket or gRPC traffic. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + pathLength: + anyOf: + - type: integer + - type: string + default: 1Ki + description: PathLength defines the maximum path length for all requests (parsed and unparsed). + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + graphQL: + description: GraphQL defines the limits for GraphQL requests. + properties: + nestingDepth: + default: 10 + description: NestingDepth defines the maximum depth of nesting for GraphQL objects. + format: int64 + type: integer + querySize: + anyOf: + - type: integer + - type: string + default: 1Ki + description: QuerySize defines the maximum size for GraphQL queries. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + valueLength: + anyOf: + - type: integer + - type: string + default: "256" + description: ValueLength defines the maximum length for GraphQL values. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + json: + description: JSON defines the limits for JSON requests. + properties: + bodySize: + anyOf: + - type: integer + - type: string + default: 100Ki + description: BodySize limits the total size of the JSON request body. It specifies the number of bytes (0 = unlimited). + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + elementCount: + default: 10000 + description: ElementCount defines the maximum number of keys and array items in the whole JSON document (recursive). + format: int64 + type: integer + keyCount: + default: 250 + description: KeyCount defines the maximum number of keys of a single JSON object (non-recursive). + format: int64 + type: integer + keyLength: + anyOf: + - type: integer + - type: string + default: "128" + description: KeyLength defines the maximum length for JSON keys. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + nestingDepth: + default: 100 + description: NestingDepth defines the maximum depth of nesting for JSON objects and JSON arrays. + format: int64 + type: integer + valueLength: + anyOf: + - type: integer + - type: string + default: 8Ki + description: ValueLength defines the maximum length for JSON values. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + multipart: + description: Multipart defines the limits for Multipart requests. + properties: + bodySize: + anyOf: + - type: integer + - type: string + default: 100Mi + description: BodySize limits the total size of the Multipart request body. It specifies the number of bytes (0 = unlimited). + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + parameter: + description: Parameter defines the limits for request parameters. + properties: + bodySize: + anyOf: + - type: integer + - type: string + default: 100Ki + description: BodySize limits the total size of the form data body. It specifies the number of bytes (0 = unlimited). + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + count: + default: 128 + description: Count defines the maximum number of request parameters. + format: int64 + type: integer + nameLength: + anyOf: + - type: integer + - type: string + default: "128" + description: NameLength defines the maximum length for parameter names. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + valueLength: + anyOf: + - type: integer + - type: string + default: 8Ki + description: ValueLength defines the maximum length for parameter values. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + unlimited: + description: Unlimited disables all limits on request scope. + type: object + type: object + settings: + description: Settings configures the limits filter. + properties: + threatHandlingMode: + default: Block + description: ThreatHandlingMode specifies how threats should be handled when a limit hits. + enum: + - Block + - LogOnly + type: string + type: object + type: object + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml new file mode 100644 index 000000000..56dad4130 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/oidcproviders.microgateway.airlock.com.yaml @@ -0,0 +1,305 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: oidcproviders.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: OIDCProvider + listKind: OIDCProviderList + plural: oidcproviders + singular: oidcprovider + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: |- + OIDCProvider specifies an OpenID Provider (OP). + + + {{% notice warning %}} The OIDC feature is currently in an experimental state. + + + We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. + In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: + - The state parameter is guessable. + - Sessions are always shared across all Microgateway Engines using the same Redis instance. + I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A + may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. + + + {{% /notice %}} + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of an OpenID Provider. + properties: + static: + description: Static configures an OpenID Provider by explicitly specifying all endpoints. + properties: + endpoints: + description: Endpoints specifies the OpenID Provider endpoints. + properties: + authorization: + description: Authorization specifies the endpoint to which the authorization request is sent. + properties: + uri: + description: URI specifies the endpoint address. + format: uri + minLength: 1 + pattern: ^(http|https)://.*$ + type: string + required: + - uri + type: object + token: + description: Token configures the endpoint from which the access, ID and refresh tokens are obtained. + properties: + tls: + description: TLS defines TLS settings. + properties: + certificateVerification: + description: CertificateVerification specifies how the certificate presented by the server is verified. + properties: + custom: + description: |- + Custom explicitly specifies how the server certificate should be verified. + Typical use cases include specifying a custom CA and SAN match when working with self-signed certificates or pinning a specific public key. + properties: + allowedSANs: + description: |- + AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the + Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, + that is to say, the SAN is verified if at least one matcher is matched. + AllowedSANs requires trustedCA to be set. + items: + description: |- + TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the + Subject Alternative Name of the presented certificate matches one of the specified matchers. + properties: + matcher: + description: Matcher defines the string matcher for the SAN value. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + sanType: + description: SanType defines the type of SAN matcher. + enum: + - DNS + - Email + - URI + - IPAddress + type: string + required: + - matcher + - sanType + type: object + minItems: 1 + type: array + certificatePinning: + description: |- + CertificatePinning defines constraints the presented certificate must fulfill. + If more than one constraint is configured only one must be satisfied. + At least one of allowedSPKIs and allowedHashes must be set. + properties: + allowedHashes: + description: |- + AllowedHashes is a list of hex-encoded SHA-256 hashes. + If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. + items: + type: string + minItems: 1 + type: array + allowedSPKIs: + description: |- + AllowedSPKIs is a list of base64-encoded SHA-256 hashes. + If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. + items: + type: string + minItems: 1 + type: array + type: object + crl: + description: CRL defines the Certificate Revocation List (CRL) settings. + properties: + lists: + description: Lists defines the list of secretRefs containing Certificate Revocation Lists. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + validationMode: + default: VerifyChain + description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. + enum: + - VerifyLeafCertOnly + - VerifyChain + type: string + type: object + trustedCA: + description: TrustedCA defines which CA certificates are trusted. + properties: + certificates: + description: Certificates defines the list of secretRefs containing trusted CA certificates. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + verificationDepth: + default: 1 + description: |- + VerificationDepth specifies the hops in the certificate chain at which validation is performed. + 1 means that either the leaf or the signing CA must be in the set of trusted certificates. + format: int32 + type: integer + required: + - certificates + type: object + type: object + disabled: + description: |- + Disabled specifies to trust any certificate without verification. + THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. + type: object + publicCAs: + description: PublicCAs specifies to only accept certificates with a SAN matching "uri" and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Engine's base image. + type: object + type: object + ciphers: + description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. + items: + type: string + minItems: 1 + type: array + protocol: + description: Protocol defines the supported TLS protocol versions. + properties: + maximum: + description: Maximum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + minimum: + description: Minimum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + type: object + type: object + uri: + description: URI specifies the endpoint address. + format: uri + minLength: 1 + pattern: ^(http|https)://.*$ + type: string + required: + - uri + type: object + required: + - authorization + - token + type: object + required: + - endpoints + type: object + type: object + required: + - spec + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml new file mode 100644 index 000000000..578ac39df --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/oidcrelyingparties.microgateway.airlock.com.yaml @@ -0,0 +1,224 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: oidcrelyingparties.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: OIDCRelyingParty + listKind: OIDCRelyingPartyList + plural: oidcrelyingparties + singular: oidcrelyingparty + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: |- + OIDCRelyingParty specifies how the Airlock Microgateway Engine interacts with an OpenID Provider (OP). + + + {{% notice warning %}} The OIDC feature is currently in an experimental state. + + + We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as security has not yet been hardened. + In particular, the current implementation has the following limitations, which we intend to address in future Microgateway releases: + - The state parameter is guessable. + - Sessions are always shared across all Microgateway Engines using the same Redis instance. + I.e. if application A and B (with different SidecarGateways) have the same Redis instance configured in their SessionHandling CR, users which are logged into application A + may be able to access authenticated routes on application B, even if their OIDCRelyingParty configuration differs. + + + {{% /notice %}} + {{% notice info %}} The OIDC feature requires SessionHandling to be configured in the SidecarGateway. {{% /notice %}} + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the OIDC Relying Party configuration. + properties: + clientID: + description: ClientID specifies the OIDCRelyingParty "client_id". + minLength: 1 + type: string + credentials: + description: Credentials used for client authentication on the back-channel with the authorization server. + properties: + clientSecret: + description: ClientSecret authenticates with the client password issued by the OpenID Provider (OP). + properties: + method: + default: BasicAuth + description: Method specifies in which format the client secret is sent with the authorization request. + enum: + - BasicAuth + - FormURLEncoded + type: string + secretRef: + description: SecretRef specifies the kubernetes secret containing the client password with key "client.secret". + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + required: + - clientSecret + type: object + oidcProviderRef: + description: OIDCProviderRef selects the OpenID Provider (OP) used to authenticate users. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + pathMapping: + description: PathMapping configures the action matching. + properties: + logoutPath: + description: LogoutPath specifies which request paths should initiate a logout. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + redirectPath: + description: RedirectPath specifies which request paths should be interpreted as a response from the authorization endpoint. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + required: + - logoutPath + - redirectPath + type: object + redirectURI: + description: |- + RedirectURI configures the "redirect_uri" parameter included in the authorization request. + May contain envoy command operators, e.g. '%REQ(:x-forwarded-proto)%://%REQ(:authority)%/callback'. + minLength: 1 + type: string + required: + - clientID + - credentials + - oidcProviderRef + - pathMapping + - redirectURI + type: object + required: + - spec + type: object + served: true + storage: true + subresources: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml new file mode 100644 index 000000000..451dc8b88 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/openapis.microgateway.airlock.com.yaml @@ -0,0 +1,167 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: openapis.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: OpenAPI + listKind: OpenAPIList + plural: openapis + singular: openapi + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: OpenAPI contains the configuration for the OpenAPI specification. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired OpenAPI specification. + properties: + response: + description: Response defines the validation behaviour for responses. + properties: + secured: + description: Secured enables response checking. + properties: + validation: + default: Lax + description: Validation defines the validation mode for responses. + enum: + - Lax + - Strict + type: string + type: object + unsecured: + description: Unsecured disables response checking. + type: object + type: object + settings: + description: Settings defines the settings to configure OpenAPI specification enforcement. + properties: + logging: + description: Logging specifies the access log behavior. + properties: + maxFailedSubvalidations: + default: 10 + description: MaxFailedSubvalidations defines the maximum number of failed subvalidations being logged. + format: int64 + type: integer + type: object + schema: + description: Schema configures the OpenAPI specification. + properties: + source: + description: Source specifies the OpenAPI specification to be enforced. + properties: + configMapRef: + description: ConfigMapRef references the configmap by its name containing the well-known key 'openapi.json'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + required: + - source + type: object + threatHandlingMode: + default: Block + description: ThreatHandlingMode specifies how threats should be handled. + enum: + - Block + - LogOnly + type: string + validation: + description: Validation specifies the patterns for the validation behavior. + properties: + authentication: + description: Authentication defines the settings for the authentication scheme. + properties: + oAuth2: + description: OAuth2 specifies the OAuth2 parameters. + properties: + allowedParameters: + description: AllowedParameters specifies the allowed parameters for the authentication scheme. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined allowed parameters. + properties: + standardParameters: + default: true + description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. + type: boolean + type: object + custom: + description: Custom allows configuring additional allowed parameters. + items: + minLength: 1 + type: string + minItems: 1 + type: array + type: object + type: object + oidc: + description: Oidc specifies the OIDC parameters. + properties: + allowedParameters: + description: AllowedParameters specifies the allowed parameters for the authentication scheme. + properties: + builtIn: + description: BuiltIn allows configuring a set of predefined allowed parameters. + properties: + standardParameters: + default: true + description: StandardParameters defines whether the allowed parameters should be expanded by the set of common parameters. + type: boolean + type: object + custom: + description: Custom allows configuring additional allowed parameters. + items: + minLength: 1 + type: string + minItems: 1 + type: array + type: object + type: object + type: object + type: object + required: + - schema + type: object + required: + - settings + type: object + required: + - spec + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml new file mode 100644 index 000000000..4d37c5adb --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/parsers.microgateway.airlock.com.yaml @@ -0,0 +1,358 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: parsers.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: Parser + listKind: ParserList + plural: parsers + singular: parser + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: Parser contains the configuration for content parsers (default and custom). + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired parser behavior. + properties: + request: + description: Request defines the parsing for downstream requests. + properties: + custom: + description: Custom allows configuring additional rules for parser selection. + properties: + rules: + description: |- + Rules defines a custom set prepended before built-in rules of enabled request parsers. + Disable all built-in parsers to overrule them completely. + items: + properties: + action: + description: |- + Action specifies what should happen when a request condition matches. + Only one of parse or skip can be set. + properties: + parse: + description: Parse activates the configured parser. + properties: + form: + description: Form activates the Form parser. + type: object + json: + description: JSON activates the JSON parser. + type: object + multipart: + description: Multipart activates the multipart parser. + type: object + type: object + skip: + description: Skip disables any content parsing + type: object + type: object + requestConditions: + description: RequestConditions defines additional request properties which must be matched in order for this rule to apply. + properties: + header: + description: Header defines the matching headers of a request. + properties: + name: + description: Name defines the name of a header. + properties: + matcher: + description: Matcher defines the way to match a string. In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + value: + description: Value defines the value of a header. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + type: object + invert: + default: false + description: Invert indicates whether the request condition should be inverted. + type: boolean + mediaType: + description: MediaType defines the matching media type from the content-type header of a request. + properties: + matcher: + description: |- + NonInvertableCaseInsensitiveStringMatcher defines the way to match a string. + In comparison to a normal StringMatcher, a value is always matched ignoring the case and can't be inverted. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + method: + description: Method defines the matching methods of a request. + items: + description: Method defines common HTTP methods. + enum: + - GET + - HEAD + - POST + - PUT + - PATCH + - DELETE + - CONNECT + - OPTIONS + - TRACE + type: string + type: array + path: + description: Path defines the matching path of a request. + properties: + matcher: + description: StringMatcher defines the way to match a string. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + required: + - matcher + type: object + remoteIP: + description: RemoteIP defines the matching remote IPs of a request. + properties: + cidrRanges: + description: CIDRRanges defines the IPv4 or IPv6 CIDR ranges, e.g. ``196.148.3.128/26`` or ``2001:db8::/28``. + items: + description: CIDRRange defines an IPv4 or IPv6 CIDR range, e.g. “196.148.3.128/26“ or “2001:db8::/28“. + format: cidr + type: string + minItems: 1 + type: array + invert: + default: false + description: Invert indicates whether the match should be inverted. + type: boolean + required: + - cidrRanges + type: object + type: object + required: + - action + - requestConditions + type: object + type: array + type: object + defaultContentType: + default: application/x-www-form-urlencoded + description: DefaultContentType specifies the content-type header which should be injected into the request before parser selection if it is not already present and the request has a body. + minLength: 1 + type: string + parsers: + description: Parsers defines the configuration for the available content parsers. + properties: + form: + description: Form defines the configuration for the form parser. + properties: + enable: + default: true + description: Enable defines whether form payloads are inspected. + type: boolean + mediaTypePattern: + default: .*urlencoded.* + description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as form arguments. + minLength: 1 + type: string + type: object + json: + description: JSON defines the configuration for the JSON parser. + properties: + enable: + default: true + description: Enable defines whether json payloads are inspected. + type: boolean + mediaTypePattern: + default: .*json.* + description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as JSON. + minLength: 1 + type: string + type: object + multipart: + description: Multipart defines the configuration for the multipart parser. + properties: + enable: + default: true + description: Enable defines whether multipart payloads are inspected. + type: boolean + mediaTypePattern: + default: .*multipart.* + description: MediaTypePattern is a regex specifying the media types for which the request body should be treated as a multipart payload. + minLength: 1 + type: string + type: object + type: object + type: object + type: object + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml new file mode 100644 index 000000000..59c94b966 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/redisproviders.microgateway.airlock.com.yaml @@ -0,0 +1,159 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: redisproviders.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: RedisProvider + listKind: RedisProviderList + plural: redisproviders + singular: redisprovider + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: RedisProvider contains a client configuration for connecting to a Redis database. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of a Redis database client configuration. + properties: + auth: + description: Auth specifies the Redis credentials. + properties: + password: + description: Password specifies the Redis password. + properties: + secretRef: + description: SecretRef selects the secret containing the Redis password under the key 'redis.password'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + username: + default: default + description: Username specifies the Redis username to authenticate with. + minLength: 1 + pattern: ^[^\s]+$ + type: string + required: + - password + type: object + mode: + description: Mode configures the redis deployment mode. + properties: + standalone: + description: Standalone specifies the standalone Redis instance to connect to. + properties: + host: + description: Host specifies the IP or hostname. + minLength: 1 + pattern: ^(\d{1,3}(\.\d{1,3}){3}|([0-9a-fA-F]{1,4}|:)+(:\d{1,3}(\.\d{1,3}){3})?|[a-z0-9\-]+(\.[a-z0-9\-]+)*)$ + type: string + port: + default: 6379 + description: Port specifies the port. + maximum: 65535 + minimum: 1 + type: integer + required: + - host + type: object + type: object + timeouts: + description: Timeouts specifies the timeouts when interacting with the Redis endpoint. + properties: + connect: + default: 5s + description: Connect specifies the timeout for establishing a connection. + type: string + maxDuration: + default: 2s + description: MaxDuration specifies the response timeout. + type: string + type: object + tls: + description: TLS defines TLS settings. If not specified, TLS is disabled i.e. unencrypted TCP is used when connecting to the Redis instance. + properties: + certificateVerification: + description: CertificateVerification specifies how the certificate presented by the server is verified. + properties: + custom: + description: Custom explicitly specifies how the server certificate should be verified. + properties: + trustedCA: + description: TrustedCA defines which CA certificates are trusted. + properties: + certificates: + description: Certificates defines the list of secretRefs containing trusted CA certificates. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + required: + - certificates + type: object + required: + - trustedCA + type: object + disabled: + description: 'Disabled specifies to trust any certificate without verification. THIS IS INSECURE AND SHOULD ONLY BE USED FOR TESTING. Note: This setting currently also disables TLS SNI.' + type: object + publicCAs: + description: PublicCAs specifies to only accept certificates with a SAN matching the host and which are signed by a CA which is either directly or indirectly trusted by any of the root CA certificates shipped with the Airlock Microgateway Session Agent’s base image. + type: object + type: object + type: object + required: + - mode + type: object + required: + - spec + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml new file mode 100644 index 000000000..5275aa7b4 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/sessionhandlings.microgateway.airlock.com.yaml @@ -0,0 +1,77 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: sessionhandlings.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: SessionHandling + listKind: SessionHandlingList + plural: sessionhandlings + singular: sessionhandling + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: |- + SessionHandling contains the configuration for session handling. + + + {{% notice warning %}} The Session Handling feature (required for OIDC) is currently in an experimental state. + + + We encourage you to try it out and give feedback, but be aware that we do not recommend using it in a production environment yet, as high-availability Redis configurations (e.g. Sentinel/Cluster) are not yet supported. + {{% /notice %}} + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired session handling behavior. + properties: + persistence: + description: Persistence configures where to store the session state. + properties: + redisProviderRef: + description: RedisProviderRef specifies to cache session information in the provided Redis instance. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - redisProviderRef + type: object + required: + - persistence + type: object + required: + - spec + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml new file mode 100644 index 000000000..ead724a75 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/sidecargateways.microgateway.airlock.com.yaml @@ -0,0 +1,758 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: sidecargateways.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: SidecarGateway + listKind: SidecarGatewayList + plural: sidecargateways + singular: sidecargateway + scope: Namespaced + versions: + - additionalPrinterColumns: + - jsonPath: .status.status + name: Status + type: string + - jsonPath: .metadata.creationTimestamp + name: Age + type: date + name: v1alpha1 + schema: + openAPIV3Schema: + description: SidecarGateway contains the configuration how to configure the Airlock Microgateway Engine when used as Sidecar Container within the Pod of an application. + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired sidecar gateway behavior. + properties: + applications: + description: Applications defines applications which run on different ports. + items: + properties: + containerPort: + default: 8080 + description: |- + ContainerPort refers to the container port. + This must be a valid port number, 0 < x < 65536. + format: int32 + maximum: 65535 + minimum: 1 + type: integer + downstream: + description: Downstream defines the downstream configuration for this application + properties: + protocol: + description: |- + Protocol defines the exposed HTTP protocol version. At most one of http1, http2 and auto can be set. + Default: auto: {} + properties: + auto: + description: Auto specifies that the protocol should be inferred. + properties: + http2: + description: HTTP2 specifies the settings for when HTTP/2 is inferred. + properties: + allowConnect: + default: false + description: Allows proxying Websocket and other upgrades over H2 connect. + type: boolean + type: object + type: object + http1: + description: HTTP1 specifies that the client is assumed to speak HTTP/1.1. + type: object + http2: + description: HTTP2 specifies that the client is assumed to speak HTTP/2. + properties: + allowConnect: + default: false + description: Allows proxying Websocket and other upgrades over H2 connect. + type: boolean + type: object + type: object + remoteIP: + description: |- + RemoteIP defines how the remote IP of a client is propagated. + Default: xff: {...} + properties: + connectionIP: + description: ConnectionIP configures to use the source IP address of the direct downstream connection. + type: object + customHeader: + description: CustomHeader specifies to use a custom header for remote IP extraction. + properties: + headerName: + description: HeaderName specifies the name of the custom header containing the remote IP. + minLength: 1 + type: string + required: + default: true + description: Required specifies if the custom header is required. If true and not available the request will be rejected with 403. + type: boolean + required: + - headerName + type: object + xff: + description: XFF configures to use the standard 'X-Forwarded-For' header for IP extraction. + properties: + numTrustedHops: + default: 1 + description: NumTrustedHops specifies to extract the client's originating IP from the nth rightmost entry in the X-Forwarded-For header. With the default value of 1, the IP is extracted from the rightmost entry. + format: int32 + minimum: 1 + type: integer + type: object + type: object + requestNormalizations: + description: RequestNormalizations defines a set of normalization actions which are applied to the request before route matching. + properties: + mergeSlashes: + default: true + description: MergeSlashes ensures that adjacent slashes in the path are merged into one. + type: boolean + normalizePath: + default: true + description: NormalizePath ensures normalization according to RFC 3986 without case normalization. + type: boolean + type: object + restrictions: + description: Restrictions defines restrictions for downstream. + properties: + http: + description: HTTP defines limits for the HTTP protocol. + properties: + headersLength: + anyOf: + - type: integer + - type: string + default: 60Ki + description: HeadersLength defines maximum size of all request headers combined. Requests that exceed this limit will receive a 431 response. + pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$ + x-kubernetes-int-or-string: true + type: object + type: object + timeouts: + description: Timeouts defines timeouts for downstream + properties: + http: + description: HTTP defines the settings for HTTP timeouts. + properties: + idle: + default: 5m + description: |- + Idle defines the settings for the idle timeout when no data is sent or received. + A value of 0 will completely disable the timeout. + Default: 5m + type: string + maxDuration: + default: 5m + description: |- + MaxDuration defines the total duration for a HTTP request/response stream. + A value of 0 will completely disable the timeout. + Default: 5m + type: string + requestHeaders: + default: 10s + description: |- + RequestHeaders defines the duration before all request headers must be received. + A value of 0 will completely disable the timeout. + Default: 10s + type: string + type: object + type: object + tls: + description: TLS defines the TLS settings. + properties: + ciphers: + description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. + items: + type: string + minItems: 1 + type: array + clientCertificate: + description: |- + ClientCertificate defines the TLS settings for verification of client certificates. + At most one of ignored, optional and required can be set. + Default: ignored: {} + properties: + ignored: + description: Ignored disables verification of the client certificate. + type: object + optional: + description: |- + Optional enables verification of the client certificate if one is presented. + In this mode only trustedCA and crl settings can be configured since certificatePinning and allowedSANs require a client certificate. + properties: + crl: + description: CRL defines the Certificate Revocation List (CRL) settings. + properties: + lists: + description: Lists defines the list of secretRefs containing Certificate Revocation Lists. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + validationMode: + default: VerifyChain + description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. + enum: + - VerifyLeafCertOnly + - VerifyChain + type: string + type: object + trustedCA: + description: TrustedCA defines which CA certificates are trusted. + properties: + certificates: + description: Certificates defines the list of secretRefs containing trusted CA certificates. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + verificationDepth: + default: 1 + description: |- + VerificationDepth specifies the hops in the certificate chain at which validation is performed. + 1 means that either the leaf or the signing CA must be in the set of trusted certificates. + format: int32 + type: integer + required: + - certificates + type: object + required: + - trustedCA + type: object + required: + description: |- + Required contains settings for client certificate verification. A client must present a valid certificate. + At least one of trustedCA and certificatePinning must be set. + properties: + allowedSANs: + description: |- + AllowedSANs is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the + Subject Alternative Name of the presented certificate matches one of the specified matchers. The matching uses “any” semantics, + that is to say, the SAN is verified if at least one matcher is matched. + AllowedSANs requires trustedCA to be set. + items: + description: |- + TLSValidationContextSANMatcher is a list of matchers to verify the Subject Alternative name. If specified, it will verify that the + Subject Alternative Name of the presented certificate matches one of the specified matchers. + properties: + matcher: + description: Matcher defines the string matcher for the SAN value. + properties: + contains: + description: |- + Contains defines a substring match on the substring specified here. Empty contains match is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + exact: + description: |- + Exact defines an explicit match on the string specified here. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + ignoreCase: + default: false + description: IgnoreCase indicates whether the matching should be case-insensitive. In case of a regex match, the regex gets wrapped with a group `(?i:...)`. + type: boolean + prefix: + description: |- + Prefix defines a prefix match on the prefix specified here. Empty prefix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + regex: + description: |- + Regex defines a regex match on the regular expression specified here. Google's [RE2 regex engine](https://github.com/google/re2/wiki/Syntax) is used. + The regex matches only single-line by default, even with ".*". To match a multi-line string prepend (?s) to your regex. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + suffix: + description: |- + Suffix defines a suffix match on the suffix specified here. Empty suffix is not allowed, please use regex instead. + Only one of exact, prefix, suffix, regex or contains can be set. + minLength: 1 + type: string + type: object + sanType: + description: SanType defines the type of SAN matcher. + enum: + - DNS + - Email + - URI + - IPAddress + type: string + required: + - matcher + - sanType + type: object + minItems: 1 + type: array + certificatePinning: + description: |- + CertificatePinning defines the constraints a client certificate must fulfill. + If more than one constraint is configured only one must be satisfied. + At least one of allowedSPKIs and allowedHashes must be set. + properties: + allowedHashes: + description: |- + AllowedHashes is a list of hex-encoded SHA-256 hashes. + If specified, it will verify that the SHA-256 of the DER-encoded presented certificate matches one of the specified values. + items: + type: string + minItems: 1 + type: array + allowedSPKIs: + description: |- + AllowedSPKIs is a list of base64-encoded SHA-256 hashes. + If specified, it will verify that the SHA-256 of the DER-encoded Subject Public Key Information (SPKI) of the presented certificate matches one of the specified values. + items: + type: string + minItems: 1 + type: array + type: object + crl: + description: CRL defines the Certificate Revocation List (CRL) settings. + properties: + lists: + description: Lists defines the list of secretRefs containing Certificate Revocation Lists. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CRL's (in PEM format) under the key 'ca.crl'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + validationMode: + default: VerifyChain + description: ValidationMode defines whether only the leaf certificate or also the CA certs should be checked. + enum: + - VerifyLeafCertOnly + - VerifyChain + type: string + type: object + trustedCA: + description: TrustedCA defines which CA certificates are trusted. + properties: + certificates: + description: Certificates defines the list of secretRefs containing trusted CA certificates. + items: + properties: + secretRef: + description: SecretRef defines the reference to a secret containing one or more CA certificates under the key 'ca.crt'. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - secretRef + type: object + minItems: 1 + type: array + verificationDepth: + default: 1 + description: |- + VerificationDepth specifies the hops in the certificate chain at which validation is performed. + 1 means that either the leaf or the signing CA must be in the set of trusted certificates. + format: int32 + type: integer + required: + - certificates + type: object + type: object + type: object + enable: + default: false + description: Enable defines if the downstream connection is encrypted. + type: boolean + protocol: + description: Protocol defines the supported TLS protocol versions. + properties: + maximum: + description: Maximum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + minimum: + description: Minimum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + type: object + secretRef: + description: SecretRef defines the reference to the TLS server certificate (secret of type kubernetes.io/tls). + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + xfcc: + description: |- + XFCC defines the handling of X-Forwarded-Client-Cert header. Meaning of the possible values: + _Sanitize_: Do not send the XFCC header to the next hop. This is the default value. + _ForwardOnly_: When the client connection is mTLS (Mutual TLS), forward the XFCC header in the request. + _AppendAndForward_: When the client connection is mTLS, append the client certificate information to the request’s XFCC header and forward it. + _SanitizeAndSet_: When the client connection is mTLS, reset the XFCC header with the client certificate information and send it to the next hop. + _AlwaysForwardOnly_: Always forward the XFCC header in the request, regardless of whether the client connection is mTLS. + Note: When forwarding the XFCC header in the request you might have to adjust the header length restrictions (See sidecargateway.spec.applications.downstream.restrictions.http) + enum: + - Sanitize + - ForwardOnly + - AppendAndForward + - SanitizeAndSet + - AlwaysForwardOnly + type: string + type: object + type: object + envoyHTTPFilterRefs: + description: EnvoyHTTPFilterRefs selects the relevant EnvoyHTTPFilters. + properties: + prepend: + description: Prepend selects the relevant EnvoyHTTPFilters which are added before those configured by the Airlock Microgateway. + items: + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: array + type: object + routes: + description: Routes defines the security configurations for different paths. The first matching route (from top to bottom) applies. + items: + description: |- + SidecarGatewayApplicationRoute defines the security configurations for different paths. + At most one of secured and unsecured can be set. + Default: secured: {...} + properties: + pathPrefix: + default: / + description: PathPrefix defines the path prefix used during route selection. + minLength: 1 + type: string + secured: + description: Secured enables WAF processing for this route. + properties: + accessControlRef: + description: |- + AccessControlRef selects the relevant AccessControl configuration resource. + If undefined, Airlock Microgateway does not perform any access control. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + contentSecurityRef: + description: |- + ContentSecurityRef selects the relevant ContentSecurity configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: object + unsecured: + description: |- + Unsecured disables all WAF functionality and therefore protection for this route. + WARNING: Using this setting when the application is exposed to untrusted downstream traffic is highly discouraged. + type: object + type: object + type: array + x-kubernetes-list-map-keys: + - pathPrefix + x-kubernetes-list-type: map + telemetryRef: + description: |- + TelemetryRef selects the relevant Telemetry configuration resource. + If undefined, default settings are applied, designed to work with most upstream web application services. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + upstream: + description: Upstream defines the upstream configuration for this application + properties: + protocol: + description: |- + Protocol defines HTTP protocol version used to communicate with the upstream. At most one of http1, http2 and auto can be set. + Default: auto: {} + properties: + auto: + description: Auto specifies to negotiate the protocol with TLS ALPN (if TLS is enabled) or, as a fallback, use the same protocol that is used by the downstream connection. + properties: + http2: + description: HTTP2 specifies the settings for when HTTP/2 is inferred. + properties: + allowConnect: + default: false + description: Allows proxying Websocket and other upgrades over H2 connect. + type: boolean + type: object + type: object + http1: + description: HTTP1 specifies to use HTTP/1.1. + type: object + http2: + description: HTTP2 specifies to use HTTP/2. + properties: + allowConnect: + default: false + description: Allows proxying Websocket and other upgrades over H2 connect. + type: boolean + type: object + type: object + timeouts: + description: Timeouts defines the timeout settings. + properties: + http: + description: HTTP defines the settings for HTTP timeouts. + properties: + idle: + description: |- + Timeout defines the settings for http timeouts. If this setting is not specified, the value of applications[].downstream.timeouts.http.idle is inherited. + A value of 0 will completely disable the timeout. + type: string + maxDuration: + default: 15s + description: |- + MaxDuration defines the total duration for a HTTP request/response stream. + Default: 15s + type: string + type: object + type: object + tls: + description: TLS defines the TLS settings. + properties: + ciphers: + description: Ciphers defines a list of the supported TLS cipher suites. For details on cipher list refer to the envoy documentation on cipher_suites in common tls configuration. + items: + type: string + minItems: 1 + type: array + enable: + default: false + description: Enable defines if the upstream connection is encrypted. + type: boolean + protocol: + description: Protocol defines the supported TLS protocol versions. + properties: + maximum: + description: Maximum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + minimum: + description: Minimum supported TLS version. + enum: + - TLSv1_0 + - TLSv1_1 + - TLSv1_2 + - TLSv1_3 + type: string + type: object + type: object + type: object + type: object + minItems: 1 + type: array + x-kubernetes-list-map-keys: + - containerPort + x-kubernetes-list-type: map + envoyClusterRefs: + description: EnvoyClusterRefs selects the relevant EnvoyClusters. + items: + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + type: array + x-kubernetes-list-map-keys: + - name + x-kubernetes-list-type: map + podSelector: + description: PodSelector defines to which Pods the configuration will be applied to. + properties: + matchLabels: + additionalProperties: + type: string + description: MatchLabels is a map of {key,value} pairs. A single {key,value} in the matchLabels. + type: object + type: object + sessionHandlingRef: + description: SessionHandlingRef selects the SessionHandling configuration to apply. + properties: + name: + description: Name of the resource + minLength: 1 + type: string + required: + - name + type: object + required: + - applications + type: object + status: + description: Most recently observed status of the SidecarGateway which is populated by the system. This data is read-only and may not be up to date. + properties: + conditions: + items: + properties: + lastTransitionTime: + description: Last time the condition transitioned from one status to another. + format: date-time + type: string + message: + description: A human-readable message indicating details about the transition. + type: string + reason: + description: The reason for the condition's last transition. + type: string + status: + description: Status of the condition, one of True, False, Unknown. + type: string + type: + description: Type of SidecarGateway condition. + type: string + required: + - status + - type + type: object + type: array + pods: + items: + properties: + envoyConfig: + description: EnvoyConfig indicates the name of the EnvoyConfig CR for the Pod. + type: string + name: + description: Name indicates the name of a Pod selected by the SidecarGateway. + type: string + sessionAgentSecret: + type: string + required: + - name + type: object + type: array + status: + type: string + unmanagedPods: + items: + properties: + managedBy: + description: ManagedBy indicates the Airlock Microgateway Operator instance which manages this Pod. + type: string + name: + description: Name indicates the name of a Pod selected by the SidecarGateway. + type: string + sessionAgentSecret: + type: string + required: + - name + type: object + type: array + required: + - status + type: object + type: object + served: true + storage: true + subresources: + status: {} diff --git a/charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml b/charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml new file mode 100644 index 000000000..80a7cba97 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/crds/telemetries.microgateway.airlock.com.yaml @@ -0,0 +1,96 @@ +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + annotations: + controller-gen.kubebuilder.io/version: v0.15.0 + labels: + app.kubernetes.io/name: airlock-microgateway-operator + app.kubernetes.io/version: 4.3.0 + name: telemetries.microgateway.airlock.com +spec: + group: microgateway.airlock.com + names: + categories: + - airlock-microgateway + kind: Telemetry + listKind: TelemetryList + plural: telemetries + singular: telemetry + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: Telemetry contains the configuration for telemetry (logging, metrics & tracing). + properties: + apiVersion: + description: |- + APIVersion defines the versioned schema of this representation of an object. + Servers should convert recognized schemas to the latest internal value, and + may reject unrecognized values. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources + type: string + kind: + description: |- + Kind is a string value representing the REST resource this object represents. + Servers may infer this from the endpoint the client submits requests to. + Cannot be updated. + In CamelCase. + More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds + type: string + metadata: + type: object + spec: + description: Specification of the desired telemetry behavior. + properties: + correlation: + description: Correlation defines the correlation aspects of Telemetry. + properties: + idSource: + description: IDSource specifies how an external correlation ID should be obtained for a request. If not specified, no correlation ID will be logged. + properties: + header: + description: Header specifies to extract the correlation ID from a request header. If the header is absent from a request, no correlation ID will be logged. + properties: + name: + default: X-Correlation-Id + description: Name of the header (case-insensitive) from which to extract the correlation ID. + minLength: 1 + type: string + type: object + required: + - header + type: object + request: + description: Request defines the request related correlation settings of Telemetry. + properties: + allowDownstreamRequestID: + default: true + description: AllowDownstreamRequestID defines whether trace sampling will consider a provided x-request-id. + type: boolean + alterRequestID: + default: true + description: AlterRequestID defines whether to alter the UUID to reflect the trace sampling decision. If disabled no modification to the UUID will be performed, this may break tracing in the upstream. + type: boolean + type: object + type: object + logging: + description: Logging defines the logging aspects of Telemetry. + properties: + accessLog: + description: AccessLog defines the access log settings of Telemetry. + properties: + format: + description: Format defines the Access Log format of the sidecar. + properties: + json: + description: JSON defines the Access Log format as JSON. + type: object + x-kubernetes-preserve-unknown-fields: true + type: object + type: object + type: object + type: object + type: object + served: true + storage: true diff --git a/charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json b/charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json new file mode 100644 index 000000000..ef0ce6d62 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/dashboards/blockLogs.json @@ -0,0 +1,510 @@ +{ + "__inputs": [ + { + "name": "DS_LOKI", + "label": "Loki", + "description": "", + "type": "datasource", + "pluginId": "loki", + "pluginName": "Loki" + }, + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": {}, + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "10.2.0" + }, + { + "type": "datasource", + "id": "loki", + "name": "Loki", + "version": "1.0.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + } + ], + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": { + "type": "grafana", + "uid": "-- Grafana --" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "type": "dashboard" + } + ] + }, + "description": "Blocked requests by Airlock Microgateway retrieved from corresponding access logs.\n\nThe dashboard can be filtered by namespace and block type. Column filters on the table allow for even a more granular filtering of the logs.", + "editable": true, + "fiscalYearStartMonth": 0, + "graphTooltip": 0, + "id": null, + "links": [ + { + "asDropdown": true, + "icon": "external link", + "includeVars": true, + "keepTime": true, + "tags": [ + "airlock-microgateway" + ], + "targetBlank": true, + "title": "Airlock Microgateway", + "tooltip": "", + "type": "dashboards", + "url": "" + } + ], + "panels": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "custom": { + "align": "auto", + "cellOptions": { + "type": "auto" + }, + "filterable": true, + "inspect": true + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "Namespace" + }, + "properties": [ + { + "id": "custom.width", + "value": 221 + }, + { + "id": "custom.filterable" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Timestamp" + }, + "properties": [ + { + "id": "custom.width", + "value": 214 + }, + { + "id": "unit", + "value": "dateTimeAsIso" + }, + { + "id": "custom.filterable" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Method" + }, + "properties": [ + { + "id": "custom.width", + "value": 89 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Client IP" + }, + "properties": [ + { + "id": "custom.width", + "value": 138 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Request ID" + }, + "properties": [ + { + "id": "custom.width", + "value": 328 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Block Type" + }, + "properties": [ + { + "id": "custom.width", + "value": 116 + }, + { + "id": "custom.filterable", + "value": false + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Request Size" + }, + "properties": [ + { + "id": "custom.width", + "value": 126 + }, + { + "id": "unit", + "value": "bytes" + }, + { + "id": "custom.align", + "value": "right" + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Attack Type" + }, + "properties": [ + { + "id": "custom.width", + "value": 217 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Application" + }, + "properties": [ + { + "id": "custom.width", + "value": 207 + } + ] + } + ] + }, + "gridPos": { + "h": 27, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 2, + "options": { + "cellHeight": "sm", + "footer": { + "countRows": false, + "enablePagination": true, + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true, + "sortBy": [] + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "editorMode": "code", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_deny_rule\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.deny_rules.matches\"\n| label_format block_type=\"deny_rules\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule_key }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", + "hide": false, + "queryType": "range", + "refId": "Deny Rule Blocks" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "editorMode": "code", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_limit\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", details=\"airlock.limits.matches\"\n| label_format block_type=\"limits\", attack_type=`{{ range $q := fromJson .details }} {{ if eq $q.threat_handling_mode \"block\" }} {{ $q.rule }} {{ end }} {{ end }}` | block_type=~\"${blockType:regex}\"", + "hide": false, + "queryType": "range", + "refId": "Limit Blocks" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "editorMode": "code", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_openapi\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.openapi.reference\", constraint=\"airlock.openapi.request.failed_validation.constraint\", position=\"airlock.openapi.request.failed_validation.position\", message=\"airlock.openapi.request.failed_validation.message\"\n| label_format block_type=\"openapi\", attack_type=\"openapi\", details=`{{.reference }}: {{.constraint }} at {{ .position }} ({{ .message }})` | block_type=~\"${blockType:regex}\"", + "hide": false, + "queryType": "range", + "refId": "OpenAPI Blocks" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "editorMode": "code", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_parser\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", attack_type=\"airlock.parser\", failed_check=\"airlock.parser.matches[0].failed_check\", message=\"airlock.parser.matches[0].message\"\n| label_format block_type=\"parsing\", attack_type=\"parsing\", details=`{{.failed_check}}: {{.message}}` | block_type=~\"${blockType:regex}\"", + "hide": false, + "queryType": "range", + "refId": "Parser Blocks" + }, + { + "datasource": { + "type": "loki", + "uid": "${DS_LOKI}" + }, + "editorMode": "code", + "expr": "{container=\"airlock-microgateway-engine\", namespace=~\"${namespace:regex}\"} |= \"airlock_request_blocked_graphql\" |= \"envoy.access\"\n| json http_method=\"http.request.method\", url=\"url.path\", request_size=\"http.request.bytes\", client_ip=\"network.forwarded_ip\", request_id=\"http.request.id\", reference=\"airlock.graphql.reference\", message=\"airlock.graphql.request.failed_validation.message\"\n| label_format block_type=\"graphql\", attack_type=\"graphql\", details=`{{ .reference }}: {{ .message }}` | block_type=~\"${blockType:regex}\"", + "hide": false, + "queryType": "range", + "refId": "GraphQL Blocks" + } + ], + "title": "Blocked Request logs", + "transformations": [ + { + "id": "merge", + "options": {} + }, + { + "id": "extractFields", + "options": { + "format": "json", + "source": "labels" + } + }, + { + "id": "filterFieldsByName", + "options": { + "byVariable": false, + "include": { + "names": [ + "Time", + "attack_type", + "block_type", + "client_ip", + "details", + "http_method", + "namespace", + "request_id", + "request_size", + "url", + "pod" + ] + } + } + }, + { + "id": "organize", + "options": { + "excludeByName": { + "Line": true, + "id": true, + "labelTypes": true, + "labels": true, + "tsNs": false + }, + "includeByName": {}, + "indexByName": { + "Time": 0, + "attack_type": 7, + "block_type": 6, + "client_ip": 9, + "details": 8, + "http_method": 3, + "namespace": 1, + "pod": 2, + "request_id": 10, + "request_size": 5, + "url": 4 + }, + "renameByName": { + "Time": "Timestamp", + "attack_type": "Attack Type", + "block_type": "Block Type", + "client_ip": "Client IP", + "details": "Details", + "http_method": "Method", + "namespace": "Namespace", + "pod": "Pod", + "request_id": "Request ID", + "request_size": "Request Size", + "tsNs": "", + "url": "Path" + } + } + } + ], + "type": "table" + } + ], + "schemaVersion": 39, + "tags": [ + "airlock-microgateway" + ], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Loki", + "value": "P8E80F9AEF21F6940" + }, + "hide": 2, + "includeAll": false, + "label": "DS_LOKI", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_license_http_rq_total,namespace)", + "hide": 0, + "includeAll": true, + "label": "Application Namespace", + "multi": true, + "name": "namespace", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_license_http_rq_total,namespace)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 5, + "type": "query" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", + "hide": 0, + "includeAll": true, + "label": "Block Type", + "multi": true, + "name": "blockType", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 5, + "type": "query" + }, + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "PBFA97CFB590B2093" + }, + "hide": 2, + "includeAll": false, + "label": "DS_PROMETHEUS", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + } + ] + }, + "time": { + "from": "now-15m", + "to": "now" + }, + "timeRangeUpdatedDuringEditOrView": false, + "timepicker": {}, + "timezone": "browser", + "title": "Airlock Microgateway Blocked Request Logs", + "uid": "adnyzcvwnyadcc", + "version": 3, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json b/charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json new file mode 100644 index 000000000..ba383d22e --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/dashboards/blockMetrics.json @@ -0,0 +1,758 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": {}, + "__requires": [ + { + "type": "panel", + "id": "barchart", + "name": "Bar chart", + "version": "" + }, + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "10.2.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "timeseries", + "name": "Time series", + "version": "" + } + ], + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": { + "type": "grafana", + "uid": "-- Grafana --" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "type": "dashboard" + } + ] + }, + "description": "Metrics on requests blocked by Airlock Microgateway.\n\nDashboard can be filtered by namespaces as well as block types.", + "editable": true, + "fiscalYearStartMonth": 0, + "graphTooltip": 0, + "id": null, + "links": [ + { + "asDropdown": true, + "icon": "external link", + "includeVars": true, + "keepTime": true, + "tags": [ + "airlock-microgateway" + ], + "targetBlank": true, + "title": "Airlock Microgateway", + "tooltip": "", + "type": "dashboards", + "url": "" + } + ], + "panels": [ + { + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 6, + "title": "Airlock Microgateway Block Metrics", + "type": "row" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Total number of requests processed by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 4, + "x": 0, + "y": 1 + }, + "id": 1, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "disableTextWrap": false, + "editorMode": "code", + "exemplar": false, + "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "format": "time_series", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": true, + "legendFormat": "Processed Requests", + "range": false, + "refId": "A", + "useBackend": false + } + ], + "title": "Requests", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [ + { + "options": { + "match": "nan", + "result": { + "index": 0, + "text": "n/a" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "percentunit" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 4, + "x": 4, + "y": 1 + }, + "id": 2, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "disableTextWrap": false, + "editorMode": "code", + "exemplar": false, + "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", + "fullMetaSearch": false, + "includeNullMetadata": true, + "instant": true, + "legendFormat": "Blocked Requests (%)", + "range": false, + "refId": "A", + "useBackend": false + } + ], + "title": "% Blocked Requests", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "blue", + "mode": "fixed" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "left", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "blue", + "value": null + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "% Blocks" + }, + "properties": [ + { + "id": "custom.axisPlacement", + "value": "right" + }, + { + "id": "unit", + "value": "percentunit" + }, + { + "id": "color", + "value": { + "fixedColor": "orange", + "mode": "fixed" + } + }, + { + "id": "max", + "value": 1 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Requests per second" + }, + "properties": [ + { + "id": "unit", + "value": "short" + }, + { + "id": "custom.fillOpacity", + "value": 25 + } + ] + } + ] + }, + "gridPos": { + "h": 10, + "w": 20, + "x": 0, + "y": 5 + }, + "id": 3, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "timezone": [ + "" + ], + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", + "instant": false, + "legendFormat": "Requests per second", + "range": true, + "refId": "Requests per Second" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", + "hide": false, + "instant": false, + "legendFormat": "% Blocks", + "range": true, + "refId": "Blocks" + } + ], + "title": "Requests vs. % Blocks", + "type": "timeseries" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Blocked requests by block type.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "super-light-orange", + "mode": "fixed" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisGridShow": true, + "axisLabel": "", + "axisPlacement": "auto", + "fillOpacity": 80, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineWidth": 0, + "scaleDistribution": { + "type": "linear" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "fieldMinMax": false, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "none" + }, + "overrides": [] + }, + "gridPos": { + "h": 11, + "w": 10, + "x": 0, + "y": 15 + }, + "id": 4, + "options": { + "barRadius": 0, + "barWidth": 0.8, + "fullHighlight": false, + "groupWidth": 0.7, + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": false + }, + "orientation": "horizontal", + "showValue": "never", + "stacking": "none", + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "asc" + }, + "xField": "block_type", + "xTickLabelRotation": 0, + "xTickLabelSpacing": 0 + }, + "pluginVersion": "10.4.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "round(sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "format": "time_series", + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "A" + } + ], + "title": "Block Type", + "transformations": [ + { + "id": "reduce", + "options": { + "includeTimeField": false, + "labelsToFields": true, + "mode": "seriesToRows", + "reducers": [ + "sum" + ] + } + } + ], + "type": "barchart" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Blocked requests by attack type, which are subsets of the various block types.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "light-orange", + "mode": "fixed" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "fillOpacity": 80, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "lineWidth": 1, + "scaleDistribution": { + "type": "linear" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 11, + "w": 10, + "x": 10, + "y": 15 + }, + "id": 5, + "options": { + "barRadius": 0, + "barWidth": 0.8, + "fullHighlight": false, + "groupWidth": 0.7, + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": false + }, + "orientation": "horizontal", + "showValue": "never", + "stacking": "none", + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "none" + }, + "xField": "attack_type", + "xTickLabelRotation": 0, + "xTickLabelSpacing": 0 + }, + "pluginVersion": "10.4.3", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "round(sum by (attack_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{block_type=~\"${blockType:regex}\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "instant": true, + "legendFormat": "__auto", + "range": false, + "refId": "A" + } + ], + "title": "Attack Type", + "transformations": [ + { + "id": "reduce", + "options": { + "labelsToFields": true, + "reducers": [ + "sum" + ] + } + } + ], + "type": "barchart" + } + ], + "refresh": "", + "schemaVersion": 39, + "tags": [ + "airlock-microgateway" + ], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "PBFA97CFB590B2093" + }, + "hide": 2, + "includeAll": false, + "label": "Datasource Prometheus", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "current": { + "selected": false, + "text": "Loki", + "value": "P8E80F9AEF21F6940" + }, + "hide": 2, + "includeAll": false, + "label": "DS_LOKI", + "multi": false, + "name": "DS_LOKI", + "options": [], + "query": "loki", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_license_valid,namespace)", + "hide": 0, + "includeAll": true, + "label": "Operator Namespace", + "multi": true, + "name": "operator_namespace", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_license_valid,namespace)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": ".*", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_license_http_rq_total,namespace)", + "hide": 0, + "includeAll": true, + "label": "Application Namespace", + "multi": true, + "name": "namespace", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_license_http_rq_total,namespace)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 5, + "type": "query" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", + "hide": 0, + "includeAll": true, + "label": "Block Type", + "multi": true, + "name": "blockType", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_http_downstream_rq_threats_blocked_total,block_type)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 5, + "type": "query" + } + ] + }, + "time": { + "from": "now-24h", + "to": "now" + }, + "timeRangeUpdatedDuringEditOrView": false, + "timepicker": { + "hidden": false + }, + "timezone": "browser", + "title": "Airlock Microgateway Block Metrics", + "uid": "ddnqoczu7qvb4cdd3dd", + "version": 3, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/dashboards/license.json b/charts/airlock/microgateway/4.3.0/dashboards/license.json new file mode 100644 index 000000000..b9d5777e2 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/dashboards/license.json @@ -0,0 +1,521 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": {}, + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "10.2.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "timeseries", + "name": "Time series", + "version": "" + } + ], + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": { + "type": "grafana", + "uid": "-- Grafana --" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "graphTooltip": 0, + "id": null, + "links": [ + { + "asDropdown": true, + "icon": "external link", + "includeVars": true, + "keepTime": true, + "tags": [ + "airlock-microgateway" + ], + "targetBlank": true, + "title": "Airlock Microgateway", + "tooltip": "", + "type": "dashboards", + "url": "" + } + ], + "panels": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "License status of Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "color": "red", + "index": 1, + "text": "Invalid" + }, + "1": { + "color": "green", + "index": 0, + "text": "Valid" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 3, + "x": 0, + "y": 0 + }, + "id": 1, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", + "instant": true, + "legendFormat": "License Status", + "range": false, + "refId": "Licenses" + } + ], + "title": "License Status", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Expiry date of the Airlock Microgateway license associated with the selected operator.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "time: L" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 4, + "x": 3, + "y": 0 + }, + "id": 4, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "min(microgateway_license_expiry_timestamp_seconds{namespace=~\"${operator_namespace.regex}\"})*1000", + "instant": true, + "legendFormat": "Expiry Date (MM/DD/YYYY)", + "range": false, + "refId": "A" + } + ], + "title": "License Expiry Date", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Number of licensed requests for applications protected by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 4, + "x": 7, + "y": 0 + }, + "id": 6, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(microgateway_license_max_rq_count_per_month{namespace=~\"${operator_namespace.regex}\"})", + "instant": true, + "legendFormat": "Licensed Requests", + "range": false, + "refId": "A" + } + ], + "title": "Licensed Requests", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Estimated number of requests protected by Airlock Microgateway over 30 days based on the last 7 days.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 5, + "x": 11, + "y": 0 + }, + "id": 2, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d]))/7*30", + "instant": true, + "legendFormat": "Estimated Requests", + "range": false, + "refId": "A" + } + ], + "title": "Requests over 30 days (estimated)", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Number of requests per week processed by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "blue", + "mode": "fixed" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 12, + "w": 16, + "x": 0, + "y": 4 + }, + "id": 5, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "sum(avg_over_time(increase(microgateway_license_http_rq_total{job=~\"${operator_namespace.regex}/.*-engine\"}[7d])[2m:30s]))", + "instant": false, + "legendFormat": "# Requests per week", + "range": true, + "refId": "A" + } + ], + "title": "Processed Requests per week", + "type": "timeseries" + } + ], + "schemaVersion": 39, + "tags": [ + "airlock-microgateway" + ], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "PBFA97CFB590B2093" + }, + "hide": 2, + "includeAll": false, + "label": "DS_PROMETHEUS", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_license_valid,namespace)", + "description": "", + "hide": 0, + "includeAll": false, + "label": "Operator Namespace", + "multi": false, + "name": "operator_namespace", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_license_valid,namespace)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-7d", + "to": "now" + }, + "timeRangeUpdatedDuringEditOrView": false, + "timepicker": {}, + "timezone": "browser", + "title": "Airlock Microgateway License", + "uid": "cdpq79bzrr01se", + "version": 2, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/dashboards/overview.json b/charts/airlock/microgateway/4.3.0/dashboards/overview.json new file mode 100644 index 000000000..094276621 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/dashboards/overview.json @@ -0,0 +1,1138 @@ +{ + "__inputs": [ + { + "name": "DS_PROMETHEUS", + "label": "Prometheus", + "description": "", + "type": "datasource", + "pluginId": "prometheus", + "pluginName": "Prometheus" + } + ], + "__elements": {}, + "__requires": [ + { + "type": "grafana", + "id": "grafana", + "name": "Grafana", + "version": "10.2.0" + }, + { + "type": "datasource", + "id": "prometheus", + "name": "Prometheus", + "version": "1.0.0" + }, + { + "type": "panel", + "id": "stat", + "name": "Stat", + "version": "" + }, + { + "type": "panel", + "id": "table", + "name": "Table", + "version": "" + }, + { + "type": "panel", + "id": "timeseries", + "name": "Time series", + "version": "" + } + ], + "annotations": { + "list": [ + { + "builtIn": 1, + "datasource": { + "type": "grafana", + "uid": "-- Grafana --" + }, + "enable": true, + "hide": true, + "iconColor": "rgba(0, 211, 255, 1)", + "name": "Annotations & Alerts", + "type": "dashboard" + } + ] + }, + "editable": true, + "fiscalYearStartMonth": 0, + "graphTooltip": 0, + "id": null, + "links": [ + { + "asDropdown": true, + "icon": "external link", + "includeVars": true, + "keepTime": true, + "tags": [ + "airlock-microgateway" + ], + "targetBlank": true, + "title": "Airlock Microgateway", + "tooltip": "", + "type": "dashboards", + "url": "" + } + ], + "panels": [ + { + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 0 + }, + "id": 3, + "title": "Overview", + "type": "row" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Number of pods that are protected by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "text", + "value": null + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 3, + "x": 0, + "y": 1 + }, + "id": 11, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(microgateway_sidecars{namespace=~\"${operator_namespace.regex}\"})", + "instant": true, + "legendFormat": "Protected Pods", + "range": false, + "refId": "A" + } + ], + "title": "Protected Pods", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Total number of requests processed by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 3, + "x": 3, + "y": 1 + }, + "id": 4, + "options": { + "colorMode": "value", + "graphMode": "none", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "disableTextWrap": false, + "editorMode": "code", + "exemplar": false, + "expr": "round(sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])))", + "format": "time_series", + "fullMetaSearch": false, + "hide": false, + "includeNullMetadata": true, + "instant": true, + "legendFormat": "Processed Requests", + "range": false, + "refId": "A", + "useBackend": false + } + ], + "title": "Requests", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Ratio of blocked requests vs. processed requests by Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "text", + "mode": "fixed" + }, + "mappings": [ + { + "options": { + "match": "nan", + "result": { + "index": 0, + "text": "n/a" + } + }, + "type": "special" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "percentunit" + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 3, + "x": 6, + "y": 1 + }, + "id": 5, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "last" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "disableTextWrap": false, + "editorMode": "code", + "exemplar": false, + "expr": "sum(increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range])) / sum(increase(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[$__range]))", + "fullMetaSearch": false, + "includeNullMetadata": true, + "instant": true, + "legendFormat": "Blocked Requests (%)", + "range": false, + "refId": "A", + "useBackend": false + } + ], + "title": "% Blocked Requests", + "type": "stat" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "License status of Airlock Microgateway.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "mappings": [ + { + "options": { + "0": { + "color": "red", + "index": 1, + "text": "Invalid" + }, + "1": { + "color": "green", + "index": 0, + "text": "Valid" + } + }, + "type": "value" + } + ], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [] + }, + "gridPos": { + "h": 4, + "w": 3, + "x": 9, + "y": 1 + }, + "id": 10, + "options": { + "colorMode": "value", + "graphMode": "area", + "justifyMode": "auto", + "orientation": "auto", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "showPercentChange": false, + "textMode": "auto", + "wideLayout": true + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "min(microgateway_license_valid{namespace=~\"${operator_namespace.regex}\"})", + "instant": true, + "legendFormat": "License Status", + "range": false, + "refId": "Licenses" + } + ], + "title": "License", + "type": "stat" + }, + { + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 5 + }, + "id": 2, + "title": "Blocks", + "type": "row" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Requests per second processed by Airlock Microgateway along with the corresponding block rate.", + "fieldConfig": { + "defaults": { + "color": { + "fixedColor": "blue", + "mode": "fixed" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "left", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "blue", + "value": null + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "% Blocks" + }, + "properties": [ + { + "id": "custom.axisPlacement", + "value": "right" + }, + { + "id": "unit", + "value": "percentunit" + }, + { + "id": "color", + "value": { + "fixedColor": "orange", + "mode": "fixed" + } + }, + { + "id": "max", + "value": 1 + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Requests per second" + }, + "properties": [ + { + "id": "unit", + "value": "short" + }, + { + "id": "custom.fillOpacity", + "value": 25 + } + ] + } + ] + }, + "gridPos": { + "h": 10, + "w": 12, + "x": 0, + "y": 6 + }, + "id": 6, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "timezone": [ + "" + ], + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "exemplar": false, + "expr": "sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", + "instant": false, + "legendFormat": "Requests per second", + "range": true, + "refId": "Requests per Second" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "sum(rate(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) / sum(rate(microgateway_license_http_rq_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m]))", + "hide": false, + "instant": false, + "legendFormat": "% Blocks", + "range": true, + "refId": "Blocks" + } + ], + "title": "Requests vs. % Blocks", + "type": "timeseries" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Requests blocked by Airlock Microgateway categorized by their corresponding type.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "cellOptions": { + "barAlignment": 0, + "drawStyle": "line", + "gradientMode": "none", + "hideValue": false, + "lineInterpolation": "linear", + "lineStyle": { + "dash": [ + 10, + 10 + ], + "fill": "solid" + }, + "showPoints": "never", + "spanNulls": false, + "type": "sparkline" + }, + "inspect": false + }, + "displayName": "Block Type", + "fieldMinMax": false, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + } + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "block_type" + }, + "properties": [ + { + "id": "custom.width", + "value": 153 + }, + { + "id": "custom.cellOptions", + "value": { + "type": "auto" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "Trend #Block Types" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "orange", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 10, + "w": 12, + "x": 12, + "y": 6 + }, + "id": 7, + "options": { + "cellHeight": "lg", + "footer": { + "countRows": false, + "enablePagination": false, + "fields": [ + "Value" + ], + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": false, + "sortBy": [ + { + "desc": true, + "displayName": "block_type" + } + ] + }, + "pluginVersion": "11.0.0", + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "sum by (block_type) (increase(microgateway_http_downstream_rq_threats_blocked_total{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m] offset -1m))/(60000/$__interval_ms)", + "format": "time_series", + "instant": false, + "legendFormat": "__auto", + "range": true, + "refId": "Block Types" + } + ], + "title": "Blocked Requests by Type", + "transformations": [ + { + "id": "timeSeriesTable", + "options": { + "A": { + "timeField": "Time" + }, + "Block Types": { + "stat": "sum", + "timeField": "Time" + } + } + } + ], + "type": "table" + }, + { + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 16 + }, + "id": 1, + "title": "Latency", + "type": "row" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Percentiles of the application downstream latency over one minute.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "ms" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "25th Percentile" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "super-light-purple", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "50th Percentile" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "purple", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "95th Percentile" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "dark-purple", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 9, + "w": 12, + "x": 0, + "y": 17 + }, + "id": 8, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.25, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "instant": false, + "legendFormat": "25th Percentile", + "range": true, + "refId": "25th Percentile" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.5, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "hide": false, + "instant": false, + "legendFormat": "50th Percentile", + "range": true, + "refId": "50th Percentile" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.95, sum(rate(envoy_http_downstream_rq_time_bucket{envoy_http_conn_manager_prefix=\"http\", namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "hide": false, + "instant": false, + "legendFormat": "95th Percentile", + "range": true, + "refId": "95th Percentile" + } + ], + "title": "Application Downstream Latency", + "type": "timeseries" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "description": "Percentiles of the Airlock Microgateway processing time over one minute.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 0, + "gradientMode": "none", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": false, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "ms" + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "25th Percentile" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "super-light-purple", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "50th Percentile" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "purple", + "mode": "fixed" + } + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "95th Percentile" + }, + "properties": [ + { + "id": "color", + "value": { + "fixedColor": "dark-purple", + "mode": "fixed" + } + } + ] + } + ] + }, + "gridPos": { + "h": 9, + "w": 12, + "x": 12, + "y": 17 + }, + "id": 9, + "options": { + "legend": { + "calcs": [], + "displayMode": "list", + "placement": "bottom", + "showLegend": true + }, + "tooltip": { + "maxHeight": 600, + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.25, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "instant": false, + "legendFormat": "25th Percentile", + "range": true, + "refId": "0.25 Percentile" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.5, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "hide": false, + "instant": false, + "legendFormat": "50th Percentile", + "range": true, + "refId": "0.5 Percentile" + }, + { + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "editorMode": "code", + "expr": "histogram_quantile(0.95, sum(rate(microgateway_rq_processing_time_ms_bucket{namespace=~\"${namespace:regex}\", job=~\"${operator_namespace.regex}/.*-engine\"}[1m])) by (le))", + "hide": false, + "instant": false, + "legendFormat": "95th Percentile", + "range": true, + "refId": "0.95 Percentile" + } + ], + "title": "Airlock Microgateway Processing Time", + "type": "timeseries" + } + ], + "refresh": "", + "schemaVersion": 39, + "tags": [ + "airlock-microgateway" + ], + "templating": { + "list": [ + { + "current": { + "selected": false, + "text": "Prometheus", + "value": "PBFA97CFB590B2093" + }, + "hide": 2, + "includeAll": false, + "label": "DS_PROMETHEUS", + "multi": false, + "name": "DS_PROMETHEUS", + "options": [], + "query": "prometheus", + "refresh": 1, + "regex": "", + "skipUrlSync": false, + "type": "datasource" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_license_valid,namespace)", + "hide": 0, + "includeAll": true, + "label": "Operator Namespace", + "multi": true, + "name": "operator_namespace", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_license_valid,namespace)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": ".*", + "skipUrlSync": false, + "sort": 0, + "type": "query" + }, + { + "allValue": ".*", + "current": {}, + "datasource": { + "type": "prometheus", + "uid": "${DS_PROMETHEUS}" + }, + "definition": "label_values(microgateway_license_http_rq_total,namespace)", + "hide": 0, + "includeAll": true, + "label": "Application Namespace", + "multi": true, + "name": "namespace", + "options": [], + "query": { + "qryType": 1, + "query": "label_values(microgateway_license_http_rq_total,namespace)", + "refId": "PrometheusVariableQueryEditor-VariableQuery" + }, + "refresh": 2, + "regex": "", + "skipUrlSync": false, + "sort": 0, + "type": "query" + } + ] + }, + "time": { + "from": "now-24h", + "to": "now" + }, + "timeRangeUpdatedDuringEditOrView": false, + "timepicker": {}, + "timezone": "browser", + "title": "Airlock Microgateway Overview", + "uid": "fdp5jb8fnrmyoa", + "version": 1, + "weekStart": "" +} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/NOTES.txt b/charts/airlock/microgateway/4.3.0/templates/NOTES.txt new file mode 100644 index 000000000..e38e3caa0 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/NOTES.txt @@ -0,0 +1,34 @@ +Thank you for installing Airlock Microgateway. +If you have not already done so, make sure that Airlock Microgateway CNI is also installed on the cluster. + +For further information, please visit our documentation at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}. +Detailed CRD API reference documentation is also available at https://docs.airlock.com/microgateway/{{ include "airlock-microgateway.docsVersion" .}}/api/crds. +{{ if .Values.crds.skipVersionCheck }} +- CRD version check skipped +{{- else }} +{{- $outdatedCRDs := (include "airlock-microgateway.outdatedCRDs" .) -}} +{{- if $outdatedCRDs -}} + {{- fail (printf ` + +Helm does not automatically upgrade CRDs from the chart's 'crds/' directory during 'helm install/upgrade'. +Therefore, the CRDs must be manually upgraded with the following command before deploying this chart: + +kubectl apply -k https://github.com/airlock/microgateway/deploy/charts/airlock-microgateway/crds/?ref=%s --server-side --force-conflicts + +If you are not using the helm install/upgrade command and instead rely on some other mechanism which is able to upgrade CRDs for deploying this chart, you can suppress this error by setting the helm value 'crds.skipVersionCheck=true'.` + .Chart.AppVersion) + -}} +{{- end -}} +{{- end -}} +{{- if .Values.tests.enabled }} + {{- if .Values.operator.watchNamespaces -}} + {{- if not (has .Release.Namespace .Values.operator.watchNamespaces) }} + {{- fail (printf ` + +To execute 'helm test', it is necessary that the release namespace '%s' is part of the operator's watch scope. Either disable the tests or ensure that the release namespace is added to watch namspace list ('operator.watchNamespaces') in the helm values. +` + .Release.Namespace) + -}} + {{- end -}} + {{- end -}} +{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl b/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl new file mode 100644 index 000000000..733ba9648 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/_helpers.tpl @@ -0,0 +1,153 @@ +{{/* +Expand the name of the chart. +We truncate at 49 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest explicit suffix is 14 characters. +*/}} +{{- define "airlock-microgateway.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 49 | trimSuffix "-" }} +{{- end }} + +{{/* +Convert an image configuration object into an image ref string. +*/}} +{{- define "airlock-microgateway.image" -}} + {{- if .digest -}} + {{- printf "%s@%s" .repository .digest -}} + {{- else if .tag -}} + {{- printf "%s:%s" .repository .tag -}} + {{- else -}} + {{- printf "%s" .repository -}} + {{- end -}} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 36 chars because some Kubernetes name fields are limited to 63 chars (by the DNS naming spec) +and the longest implicit suffix is 27 characters. +If release name contains chart name it will be used as a full name. +*/}} +{{- define "airlock-microgateway.fullname" -}} +{{- if .Values.fullnameOverride }} +{{- .Values.fullnameOverride | trunc 36 | trimSuffix "-" }} +{{- else }} +{{- $name := default .Chart.Name .Values.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 36 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 36 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "airlock-microgateway.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "airlock-microgateway.sharedLabels" -}} +helm.sh/chart: {{ include "airlock-microgateway.chart" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +app.kubernetes.io/part-of: {{ .Chart.Name }} +{{- with .Values.commonLabels }} +{{ toYaml .}} +{{- end }} +{{- end }} + +{{/* +Common Selector labels +*/}} +{{- define "airlock-microgateway.sharedSelectorLabels" -}} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Restricted Container Security Context +*/}} +{{- define "airlock-microgateway.restrictedSecurityContext" -}} +allowPrivilegeEscalation: false +privileged: false +runAsNonRoot: true +capabilities: + drop: ["ALL"] +readOnlyRootFilesystem: true +seccompProfile: + type: RuntimeDefault +{{- end }} + +{{/* Precondition: May only be used if AppVersion is isSemver */}} +{{- define "airlock-microgateway.supportedCRDVersionPattern" -}} +{{- $version := (semver .Chart.AppVersion) -}} +{{- if $version.Prerelease -}} +>= {{ $version.Major }}.{{ $version.Minor }}.{{ $version.Patch }}-{{ $version.Prerelease }} +{{- else -}} +>= {{ $version.Major }}.{{ $version.Minor }}.0 || >= {{ $version.Major }}.{{ $version.Minor }}.{{ add1 $version.Patch }}-0 +{{- end -}} +{{- end -}} + +{{- define "airlock-microgateway.outdatedCRDs" -}} +{{- if (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) -}} + {{- $supportedVersion := (include "airlock-microgateway.supportedCRDVersionPattern" .) -}} + {{- range $path, $_ := .Files.Glob "crds/*.yaml" -}} + {{- $api := ($.Files.Get $path | fromYaml).metadata.name -}} + {{- $crd := (lookup "apiextensions.k8s.io/v1" "CustomResourceDefinition" "" $api) -}} + {{- $isOutdated := false -}} + {{- if $crd -}} + {{/* If CRD is already present in the cluster, it must have the minimum supported version */}} + {{- $isOutdated = true -}} + {{- if hasKey $crd.metadata "labels" -}} + {{- $crdVersion := get $crd.metadata.labels "app.kubernetes.io/version" -}} + {{- if (eq "true" (include "airlock-microgateway.isSemver" $crdVersion)) -}} + {{- if (semverCompare $supportedVersion $crdVersion) }} + {{- $isOutdated = false -}} + {{- end }} + {{- end -}} + {{- end -}} + {{- end -}} + {{- if $isOutdated }} +{{ base $path }} + {{- end }} + {{- end -}} +{{- end -}} +{{- end -}} + +{{- define "airlock-microgateway.isSemver" -}} +{{- regexMatch `^(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?$` . -}} +{{- end -}} + +{{- define "airlock-microgateway.docsVersion" -}} +{{- if and (eq "true" (include "airlock-microgateway.isSemver" .Chart.AppVersion)) (not (contains "-" .Chart.AppVersion)) -}} + {{- $version := (semver .Chart.AppVersion) -}} + {{- $version.Major }}.{{ $version.Minor -}} +{{- else -}} + {{- print "latest" -}} +{{- end -}} +{{- end -}} + +{{- define "airlock-microgateway.watchNamespaceSelector.labelQuery" -}} +{{- $list := list -}} +{{- with .matchLabels -}} + {{- range $key, $value := . -}} + {{- $list = append $list (printf "%s=%s" $key $value) -}} + {{- end -}} +{{- end -}} +{{- with .matchExpressions -}} + {{- range . -}} + {{- if has .operator (list "In" "NotIn") -}} + {{- $list = append $list (printf "%s %s (%s)" .key (lower .operator) (join "," .values)) -}} + {{- else if eq .operator "Exists" -}} + {{- $list = append $list .key -}} + {{- else if eq .operator "DoesNotExist" -}} + {{- $list = append $list (printf "!%s" .key) -}} + {{- end -}} + {{- end -}} +{{- end -}} +{{- join "," $list -}} +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl b/charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl new file mode 100644 index 000000000..a540ff9f4 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/_operator_helpers.tpl @@ -0,0 +1,42 @@ +{{/* +Create a default fully qualified name for operator components. +*/}} +{{- define "airlock-microgateway.operator.fullname" -}} +{{ include "airlock-microgateway.fullname" . }}-operator +{{- end }} + + +{{/* +Common operator labels +*/}} +{{- define "airlock-microgateway.operator.labels" -}} +{{ include "airlock-microgateway.sharedLabels" . }} +{{ include "airlock-microgateway.operator.selectorLabels" . }} +{{- end }} + +{{/* +Operator Selector labels +*/}} +{{- define "airlock-microgateway.operator.selectorLabels" -}} +{{ include "airlock-microgateway.sharedSelectorLabels" . }} +app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-operator +app.kubernetes.io/component: controller +{{- end }} + +{{/* +Create the name of the service account to use for the operator +*/}} +{{- define "airlock-microgateway.operator.serviceAccountName" -}} +{{- if .Values.operator.serviceAccount.create }} +{{- default (include "airlock-microgateway.operator.fullname" .) .Values.operator.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.operator.serviceAccount.name }} +{{- end }} +{{- end }} + +{{/* +ServiceMonitor metrics regex pattern for leader only metrics +*/}} +{{- define "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" -}} +^(microgateway_license|microgateway_sidecars).*$ +{{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl b/charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl new file mode 100644 index 000000000..83b314cbc --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/_rbac.gen.tpl @@ -0,0 +1,237 @@ +{{/* AUTOGENERATED FILE DO NOT EDIT */}} + +{{/* +Operator rbac permission rules +*/}} +{{- define "airlock-microgateway-operator.rbacRules" -}} +- apiGroups: + - "" + resources: + - configmaps + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - events + verbs: + - create + - patch +- apiGroups: + - "" + resources: + - namespaces + verbs: + - get + - list + - watch +- apiGroups: + - "" + resources: + - pods + verbs: + - delete + - get + - list + - patch + - update + - watch +- apiGroups: + - "" + resources: + - pods/finalizers + verbs: + - update +- apiGroups: + - "" + resources: + - pods/status + verbs: + - patch + - update +- apiGroups: + - "" + resources: + - secrets + verbs: + - create + - delete + - get + - list + - update + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - accesscontrols + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - contentsecurities + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - denyrules + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - envoyclusters + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - envoyconfigurations + verbs: + - create + - delete + - get + - list + - patch + - update + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - envoyconfigurations/status + verbs: + - get + - patch + - update +- apiGroups: + - microgateway.airlock.com + resources: + - envoyhttpfilters + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - graphqls + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - headerrewrites + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - identitypropagations + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - limits + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - oidcproviders + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - oidcrelyingparties + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - openapis + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - parsers + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - redisproviders + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - sessionhandlings + verbs: + - get + - list + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways + verbs: + - get + - list + - patch + - update + - watch +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways/finalizers + verbs: + - update +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways/status + verbs: + - get + - patch + - update +- apiGroups: + - microgateway.airlock.com + resources: + - telemetries + verbs: + - get + - list + - watch +{{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl b/charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl new file mode 100644 index 000000000..02e304890 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/_webhooks.gen.tpl @@ -0,0 +1,339 @@ +{{/* AUTOGENERATED FILE DO NOT EDIT */}} + +{{/* +Operator mutating webhooks +*/}} +{{- define "airlock-microgateway-operator.mutatingWebhooks" -}} +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /mutate-v1-pod + failurePolicy: Fail + name: mutate-pod.microgateway.airlock.com + reinvocationPolicy: IfNeeded + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + resources: + - pods + sideEffects: None + objectSelector: + matchLabels: + sidecar.microgateway.airlock.com/inject: "true" +{{- end }} + +{{/* +Operator validating webhooks +*/}} +{{- define "airlock-microgateway-operator.validatingWebhooks" -}} +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-v1-pod + failurePolicy: Fail + name: validate-pod.microgateway.airlock.com + rules: + - apiGroups: + - "" + apiVersions: + - v1 + operations: + - CREATE + - UPDATE + resources: + - pods + sideEffects: None + objectSelector: + matchLabels: + sidecar.microgateway.airlock.com/inject: "true" +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-accesscontrol + failurePolicy: Fail + name: validate-accesscontrol.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - accesscontrols + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-denyrules + failurePolicy: Fail + name: validate-denyrules.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - denyrules + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-envoycluster + failurePolicy: Fail + name: validate-envoycluster.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - envoyclusters + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-envoyhttpfilter + failurePolicy: Fail + name: validate-envoyhttpfilter.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - envoyhttpfilters + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-graphql + failurePolicy: Fail + name: validate-graphql.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - graphqls + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-headerrewrites + failurePolicy: Fail + name: validate-headerrewrites.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - headerrewrites + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-identitypropagation + failurePolicy: Fail + name: validate-identitypropagation.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - identitypropagations + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-limits + failurePolicy: Fail + name: validate-limits.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - limits + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-oidcprovider + failurePolicy: Fail + name: validate-oidcprovider.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - oidcproviders + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-oidcrelyingparty + failurePolicy: Fail + name: validate-oidcrelyingparty.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - oidcrelyingparties + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-openapi + failurePolicy: Fail + name: validate-openapi.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - openapis + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-parser + failurePolicy: Fail + name: validate-parser.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - parsers + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-redisprovider + failurePolicy: Fail + name: validate-redisprovider.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - redisproviders + sideEffects: None +- admissionReviewVersions: + - v1 + clientConfig: + service: + name: airlock-microgateway-operator-webhook + namespace: '{{ .Release.Namespace }}' + path: /validate-microgateway-airlock-com-v1alpha1-sidecargateway + failurePolicy: Fail + name: validate-sidecargateway.microgateway.airlock.com + rules: + - apiGroups: + - microgateway.airlock.com + apiVersions: + - v1alpha1 + operations: + - CREATE + - UPDATE + resources: + - sidecargateways + sideEffects: None +{{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml new file mode 100644 index 000000000..e86208023 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/configmap.yaml @@ -0,0 +1,394 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-config + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +data: + engine_bootstrap_config_template.yaml: | + # Base configuration, admin interface on port 19000 + admin: + address: + socket_address: + address: 127.0.0.1 + port_value: 19000 + dynamic_resources: + cds_config: + initial_fetch_timeout: 10s + resource_api_version: V3 + api_config_source: + api_type: GRPC + transport_api_version: V3 + grpc_services: + - envoy_grpc: + cluster_name: xds_cluster + set_node_on_first_message_only: true + # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC + rate_limit_settings: + max_tokens: 5 + fill_rate: 0.2 + lds_config: + resource_api_version: V3 + initial_fetch_timeout: 10s + api_config_source: + api_type: GRPC + transport_api_version: V3 + grpc_services: + - envoy_grpc: + cluster_name: xds_cluster + set_node_on_first_message_only: true + # Prevent Envoy Node from overloading the xDS server due to rejected configuration when using xDS SotW gRPC + rate_limit_settings: + max_tokens: 5 + fill_rate: 0.2 + static_resources: + listeners: + - name: probe + address: + socket_address: + address: 0.0.0.0 + port_value: 19001 + filter_chains: + - filters: + - name: http_connection_manager + typed_config: + '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager + stat_prefix: probe + codec_type: AUTO + http2_protocol_options: + initial_connection_window_size: 1048576 + initial_stream_window_size: 65536 + max_concurrent_streams: 100 + route_config: + name: probe + virtual_hosts: + - name: probe + domains: + - '*' + routes: + - name: ready + match: + path: /ready + headers: + - name: ':method' + string_match: + exact: 'GET' + route: + cluster: airlock_microgateway_engine_admin + http_filters: + - name: envoy.filters.http.router + typed_config: + '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router + - name: metrics + address: + socket_address: + address: 0.0.0.0 + port_value: 19002 + filter_chains: + - filters: + - name: http_connection_manager + typed_config: + '@type': type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager + stat_prefix: metrics + codec_type: AUTO + http2_protocol_options: + initial_connection_window_size: 1048576 + initial_stream_window_size: 65536 + max_concurrent_streams: 100 + route_config: + name: metrics + virtual_hosts: + - name: metrics + domains: + - '*' + routes: + - name: metrics + match: + path: /metrics + headers: + - name: ':method' + string_match: + exact: 'GET' + route: + prefix_rewrite: '/stats/prometheus' + cluster: airlock_microgateway_engine_admin + http_filters: + - name: envoy.filters.http.router + typed_config: + '@type': type.googleapis.com/envoy.extensions.filters.http.router.v3.Router + clusters: + - name: xds_cluster + connect_timeout: 1s + type: STRICT_DNS + load_assignment: + cluster_name: xds_cluster + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: airlock-microgateway-operator-xds.$(OPERATOR_NAMESPACE).svc.cluster.local + port_value: 13377 + typed_extension_protocol_options: + envoy.extensions.upstreams.http.v3.HttpProtocolOptions: + '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions + explicit_http_config: + http2_protocol_options: + connection_keepalive: + interval: 360s + timeout: 5s + transport_socket: + name: envoy.transport_sockets.tls + typed_config: + '@type': type.googleapis.com/envoy.extensions.transport_sockets.tls.v3.UpstreamTlsContext + common_tls_context: + tls_params: + tls_minimum_protocol_version: TLSv1_3 + tls_maximum_protocol_version: TLSv1_3 + validation_context_sds_secret_config: + name: validation_context_sds + sds_config: + resource_api_version: V3 + path_config_source: + path: /etc/envoy/validation_context_sds_secret.yaml + watched_directory: + path: /etc/envoy/ + tls_certificate_sds_secret_configs: + - name: tls_certificate_sds + sds_config: + resource_api_version: V3 + path_config_source: + path: /etc/envoy/tls_certificate_sds_secret.yaml + watched_directory: + path: /etc/envoy/ + - name: airlock_microgateway_engine_admin + connect_timeout: 1s + type: STATIC + load_assignment: + cluster_name: airlock_microgateway_engine_admin + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: 127.0.0.1 + port_value: 19000 + typed_extension_protocol_options: + envoy.extensions.upstreams.http.v3.HttpProtocolOptions: + '@type': type.googleapis.com/envoy.extensions.upstreams.http.v3.HttpProtocolOptions + explicit_http_config: + http2_protocol_options: + connection_keepalive: + interval: 360s + timeout: 5s + stats_config: + stats_tags: + - tag_name: "block_type" + regex: "\\.(block_type\\.([^.]+))" + - tag_name: "attack_type" + regex: "\\.(attack_type\\.([^.]+))" + - tag_name: "envoy_cluster_name" + regex: "\\.(cluster\\.([^.]+))" + - tag_name: "version" + regex: "\\.(version\\.([^.]+))" + use_all_default_tags: true + overload_manager: + resource_monitors: + - name: "envoy.resource_monitors.global_downstream_max_connections" + typed_config: + "@type": type.googleapis.com/envoy.extensions.resource_monitors.downstream_connections.v3.DownstreamConnectionsConfig + max_active_downstream_connections: 50000 + bootstrap_extensions: + - name: airlock.bootstrap.engine_build_info + typed_config: + '@type': type.googleapis.com/airlock.extensions.bootstrap.stats.v1alpha.Stats + application_log_config: + log_format: + text_format: '{"@timestamp":"%Y-%m-%dT%T.%e%z","log":{"logger":"%n","level":"%l","origin":{"file":{"name":"%g","line":%#},"function":"%!"}},"event":{"module":"envoy","dataset":"envoy.application"},"process":{"pid":%P,"thread":{"id":%t}},"ecs":{"version":"8.5"},"message":"%j"}' + engine_container_template.yaml: | + name: "$(ENGINE_NAME)" + image: "$(ENGINE_IMAGE)" + imagePullPolicy: {{ .Values.engine.image.pullPolicy }} + args: + - "--config-path" + - "/etc/envoy/bootstrap_config.yaml" + - "--base-id" + - "$(BASE_ID)" + - "--file-flush-interval-msec" + - '1000' + - "--drain-time-s" + - '60' + - "--service-node" + - "$(POD_NAME).$(POD_NAMESPACE)" + - "--service-cluster" + - "$(APP_NAME).$(POD_NAMESPACE)" + - "--log-path" + - "/dev/stdout" + - "--log-level" + - "$(LOG_LEVEL)" + volumeMounts: + - name: airlock-microgateway-bootstrap-secret-volume + mountPath: /etc/envoy + readOnly: true + env: + - name: POD_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + - name: POD_IP + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: status.podIP + ports: + - containerPort: 13378 + protocol: TCP + - containerPort: 19001 + protocol: TCP + - containerPort: 19002 + protocol: TCP + livenessProbe: + httpGet: + path: /ready + port: 19001 + scheme: HTTP + initialDelaySeconds: 5 + periodSeconds: 5 + failureThreshold: 5 + successThreshold: 1 + timeoutSeconds: 1 + readinessProbe: + httpGet: + path: /ready + port: 19001 + scheme: HTTP + initialDelaySeconds: 5 + periodSeconds: 5 + failureThreshold: 3 + successThreshold: 1 + timeoutSeconds: 1 + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + runAsUser: $(SECURITYCONTEXT_UID) + {{- with .Values.engine.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + session_agent_container_template.yaml: | + name: "$(SESSION_AGENT_NAME)" + image: "$(SESSION_AGENT_IMAGE)" + imagePullPolicy: {{ .Values.sessionAgent.image.pullPolicy }} + args: + - "--port" + - "19004" + - "--config-path" + - "/etc/microgateway-session-agent/config.json" + volumeMounts: + - name: airlock-microgateway-session-agent-volume + mountPath: /etc/microgateway-session-agent + readOnly: true + env: + - name: POD_NAME + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.name + - name: POD_NAMESPACE + valueFrom: + fieldRef: + apiVersion: v1 + fieldPath: metadata.namespace + ports: + - containerPort: 19004 + livenessProbe: + {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} + grpc: + port: 19004 + {{- else }} + tcpSocket: + port: 19004 + {{- end }} + initialDelaySeconds: 5 + periodSeconds: 5 + failureThreshold: 5 + successThreshold: 1 + timeoutSeconds: 5 + readinessProbe: + {{- if (semverCompare ">=1.27 || >=1.27.1-0" .Capabilities.KubeVersion.Version)}} + grpc: + port: 19004 + {{- else }} + tcpSocket: + port: 19004 + {{- end }} + initialDelaySeconds: 5 + periodSeconds: 5 + failureThreshold: 3 + successThreshold: 1 + timeoutSeconds: 5 + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + runAsUser: $(SECURITYCONTEXT_UID) + {{- with .Values.sessionAgent.resources }} + resources: + {{- toYaml . | nindent 6 }} + {{- end }} + network_validator_container_template.yaml: | + name: "$(NETWORK_VALIDATOR_NAME)" + image: "$(NETWORK_VALIDATOR_IMAGE)" + imagePullPolicy: {{ .Values.networkValidator.image.pullPolicy }} + command: ["/bin/sh", "-c"] + args: + - |- + echo 'pong' | nc -v -l 127.0.0.1 -p 13378 & + for i in 1 2 3; do + sleep 1s + if r=$(echo 'ping' | nc 127.0.0.1 19003) && [ $r == pong ]; then + echo -n 'Traffic redirection to Airlock Microgateway Engine is working.' > /dev/termination-log + exit 0 + fi + done + echo -en 'Traffic redirection to Airlock Microgateway Engine is not working.\nRestart the pod after ensuring that hostNetwork is disabled and a compatible Airlock Microgateway CNI version is installed on the node.\nCertain environments may also require additional configuration (see docs.airlock.com for more information).' > /dev/termination-log + exit 1 + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + runAsUser: $(SECURITYCONTEXT_UID) + operator_config.yaml: | + apiVersion: config.airlock.com/v1alpha1 + kind: OperatorConfig + health: + healthProbeBindAddress: :8081 + metrics: + bindAddress: 0.0.0.0:8080 + webhook: + port: 9443 + deployment: + sidecar: + engineContainerTemplate: "/sidecar/engine_container_template.yaml" + networkValidatorContainerTemplate: "/sidecar/network_validator_container_template.yaml" + sessionAgentContainerTemplate: "/sidecar/session_agent_container_template.yaml" + engine: + bootstrapConfigTemplate: "/engine_bootstrap_config_template.yaml" + log: + level: {{ .Values.operator.config.logLevel }} + {{- with $.Values.operator.watchNamespaceSelector }} + namespaces: + selector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with $.Values.operator.watchNamespaces }} + namespaces: + list: + {{- toYaml . | nindent 8 }} + {{- end }} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/dashboard-configmap.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/dashboard-configmap.yaml new file mode 100644 index 000000000..b71ac89b6 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/dashboard-configmap.yaml @@ -0,0 +1,28 @@ +{{- if .Values.dashboards.create -}} +{{- range $instance := (keys .Values.dashboards.instances | sortAlpha) -}} +{{- $dashboard := get $.Values.dashboards.instances $instance -}} +{{- if $dashboard.create }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "airlock-microgateway.fullname" $ }}-dashboard-{{ $instance | lower }} + namespace: {{ $.Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} + {{- with $.Values.dashboards.config.grafana.dashboardLabel -}} + {{- .name | nindent 4 -}}: {{ .value | quote }} + {{- end }} + annotations: + {{- with $.Values.dashboards.config.grafana.folderAnnotation -}} + {{- .name | nindent 4 -}}: {{ .value | quote }} + {{- end }} + {{- with $.Values.commonAnnotations }} + {{- toYaml . | nindent 4 }} + {{- end }} +data: + {{- printf "%s.json" $instance | nindent 2 }}: |- + {{- ($.Files.Get (printf "dashboards/%s.json" $instance)) | nindent 4 -}} +{{- end -}} +{{- end -}} +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/deployment.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/deployment.yaml new file mode 100644 index 000000000..db340cdec --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/deployment.yaml @@ -0,0 +1,143 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + replicas: {{ .Values.operator.replicaCount }} + {{- with .Values.operator.updateStrategy }} + strategy: + {{- toYaml . | trim | nindent 4 }} + {{- end }} + selector: + matchLabels: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + checksum/config: {{ include (print $.Template.BasePath "/operator/configmap.yaml") . | sha256sum }} + kubectl.kubernetes.io/default-container: manager + {{- with mustMerge .Values.operator.podAnnotations .Values.commonAnnotations}} + {{- toYaml . | nindent 8 }} + {{- end }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 8 }} + {{- with .Values.operator.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + spec: + containers: + - args: + - --config=operator_config.yaml + env: + - name: ENGINE_IMAGE + value: {{ include "airlock-microgateway.image" .Values.engine.image }} + - name: NETWORK_VALIDATOR_IMAGE + value: {{ include "airlock-microgateway.image" .Values.networkValidator.image }} + - name: SESSION_AGENT_IMAGE + value: {{ include "airlock-microgateway.image" .Values.sessionAgent.image }} + - name: OPERATOR_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: OPERATOR_NAMESPACE + valueFrom: + fieldRef: + fieldPath: metadata.namespace + image: {{ include "airlock-microgateway.image" .Values.operator.image }} + imagePullPolicy: {{ .Values.operator.image.pullPolicy }} + livenessProbe: + httpGet: + path: /healthz + port: 8081 + initialDelaySeconds: 15 + periodSeconds: 20 + timeoutSeconds: 5 + name: manager + ports: + - containerPort: 9443 + name: webhook-server + protocol: TCP + - containerPort: 13377 + name: xds-server + protocol: TCP + - containerPort: 8080 + protocol: TCP + - containerPort: 8081 + protocol: TCP + readinessProbe: + httpGet: + path: /readyz + port: 8081 + initialDelaySeconds: 5 + periodSeconds: 10 + timeoutSeconds: 5 + {{- with .Values.operator.resources }} + resources: + {{- toYaml . | nindent 10 }} + {{- end }} + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 10 }} + terminationMessagePath: /dev/termination-log + terminationMessagePolicy: FallbackToLogsOnError + volumeMounts: + - mountPath: /tmp/k8s-webhook-server/serving-certs + name: cert + readOnly: true + - mountPath: /opt/airlock/license/ + name: airlock-microgateway-license + readOnly: true + - mountPath: /operator_config.yaml + name: operator-config + subPath: operator_config.yaml + - mountPath: /sidecar/engine_container_template.yaml + name: operator-config + subPath: engine_container_template.yaml + - mountPath: /sidecar/network_validator_container_template.yaml + name: operator-config + subPath: network_validator_container_template.yaml + - mountPath: /sidecar/session_agent_container_template.yaml + name: operator-config + subPath: session_agent_container_template.yaml + - mountPath: /engine_bootstrap_config_template.yaml + name: operator-config + subPath: engine_bootstrap_config_template.yaml + securityContext: + runAsNonRoot: true + serviceAccountName: {{ include "airlock-microgateway.operator.serviceAccountName" . }} + terminationGracePeriodSeconds: 10 + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.operator.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.operator.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.operator.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - name: cert + secret: + defaultMode: 420 + secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert + - name: airlock-microgateway-license + secret: + defaultMode: 292 + optional: true + secretName: {{ .Values.license.secretName }} + - configMap: + name: {{ include "airlock-microgateway.operator.fullname" . }}-config + name: operator-config diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/manager-role.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/manager-role.yaml new file mode 100644 index 000000000..90335bcfe --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/manager-role.yaml @@ -0,0 +1,33 @@ +{{- if .Values.operator.rbac.create }} +{{- if empty .Values.operator.watchNamespaces }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +{{ include "airlock-microgateway-operator.rbacRules" . -}} +{{- else }} +{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager + namespace: {{ $namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} + {{- with $.Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: +{{ include "airlock-microgateway-operator.rbacRules" $ }} +--- +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/manager-rolebinding.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/manager-rolebinding.yaml new file mode 100644 index 000000000..ae99cfb7b --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/manager-rolebinding.yaml @@ -0,0 +1,45 @@ +{{- if .Values.operator.rbac.create }} +{{- if empty .Values.operator.watchNamespaces }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ include "airlock-microgateway.operator.fullname" . }}-manager-{{ .Release.Namespace }} +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} +{{- else }} +{{- range $namespace := (append .Values.operator.watchNamespaces .Release.Namespace | uniq) }} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager + namespace: {{ $namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" $ | nindent 4 }} + {{- with $.Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway.operator.fullname" $ }}-manager +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway.operator.serviceAccountName" $ }} + namespace: {{ $.Release.Namespace }} +--- +{{- end -}} +{{- end -}} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml new file mode 100644 index 000000000..34d23f6d6 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/metrics-service.yaml @@ -0,0 +1,47 @@ +apiVersion: v1 +kind: Service +metadata: + name: airlock-microgateway-operator-metrics + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.operator.serviceLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + ports: + - appProtocol: http + name: metrics + port: 8080 + protocol: TCP + selector: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} +--- +apiVersion: v1 +kind: Service +metadata: + name: airlock-microgateway-operator-leader-metrics + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.operator.serviceLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + operator.microgateway.airlock.com/isLeader: "true" + {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + ports: + - appProtocol: http + name: metrics + port: 8080 + protocol: TCP + selector: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} + operator.microgateway.airlock.com/isLeader: "true" \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/mutating-webhook.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/mutating-webhook.yaml new file mode 100644 index 000000000..311f9726a --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/mutating-webhook.yaml @@ -0,0 +1,28 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: MutatingWebhookConfiguration +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + annotations: + cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert + {{- with .Values.commonAnnotations }} + {{- toYaml . | nindent 4 }} + {{- end }} +webhooks: +{{- range $webhook := (include "airlock-microgateway-operator.mutatingWebhooks" .) | fromYamlArray }} +- {{ toYaml $webhook | indent 2 | trim }} + {{- with $.Values.operator.watchNamespaceSelector }} + namespaceSelector: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with $.Values.operator.watchNamespaces }} + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: In + values: + {{- toYaml . | nindent 10 }} + {{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/podmonitor.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/podmonitor.yaml new file mode 100644 index 000000000..1fe34fcb3 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/podmonitor.yaml @@ -0,0 +1,27 @@ +{{- if .Values.engine.sidecar.podMonitor.create }} +apiVersion: monitoring.coreos.com/v1 +kind: PodMonitor +metadata: + name: {{ include "airlock-microgateway.fullname" . }}-engine + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.engine.sidecar.podMonitor.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + namespaceSelector: + any: true + selector: + matchLabels: + sidecar.microgateway.airlock.com/inject: "true" + microgateway.airlock.com/managedBy: {{ .Release.Namespace }} + podMetricsEndpoints: + - targetPort: 19002 + path: /metrics + scheme: http +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/role.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/role.yaml new file mode 100644 index 000000000..5378be8ef --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/role.yaml @@ -0,0 +1,45 @@ +{{- if .Values.operator.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +rules: + - apiGroups: + - "" + resources: + - configmaps + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - coordination.k8s.io + resources: + - leases + verbs: + - get + - list + - watch + - create + - update + - patch + - delete + - apiGroups: + - "" + resources: + - events + verbs: + - create + - patch +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml new file mode 100644 index 000000000..bafec1015 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/rolebinding.yaml @@ -0,0 +1,20 @@ +{{- if .Values.operator.rbac.create -}} +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ include "airlock-microgateway.operator.fullname" . }}-leader-election +subjects: + - kind: ServiceAccount + name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml new file mode 100644 index 000000000..466c56338 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/selfsigned-issuer.yaml @@ -0,0 +1,13 @@ +apiVersion: cert-manager.io/v1 +kind: Issuer +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selfSigned: {} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml new file mode 100644 index 000000000..434d7e9d3 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/serviceaccount.yaml @@ -0,0 +1,13 @@ +{{- if .Values.operator.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "airlock-microgateway.operator.serviceAccountName" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with mustMerge .Values.operator.serviceAccount.annotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml new file mode 100644 index 000000000..ff85a9a31 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/servicemonitor.yaml @@ -0,0 +1,60 @@ +{{- if .Values.operator.serviceMonitor.create }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.operator.serviceMonitor.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} + matchExpressions: + - { key: "operator.microgateway.airlock.com/isLeader", operator: DoesNotExist } + endpoints: + - path: /metrics + port: metrics + scheme: http + metricRelabelings: + - sourceLabels: + - __name__ + regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} + action: drop +--- +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-leader + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.operator.serviceMonitor.labels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + selector: + matchLabels: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 6 }} + operator.microgateway.airlock.com/isLeader: "true" + endpoints: + - path: /metrics + port: metrics + scheme: http + metricRelabelings: + - sourceLabels: + - __name__ + regex: {{ include "airlock-microgateway.operator.metricsLeaderOnlyRegexPattern" . }} + action: keep +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml new file mode 100644 index 000000000..60b92e1e2 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/serving-certificate.yaml @@ -0,0 +1,19 @@ +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-serving-cert + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + dnsNames: + - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc + - airlock-microgateway-operator-webhook.{{ .Release.Namespace }}.svc.cluster.local + issuerRef: + kind: Issuer + name: {{ include "airlock-microgateway.operator.fullname" . }}-selfsigned-issuer + secretName: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-server-cert diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/validating-webhook.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/validating-webhook.yaml new file mode 100644 index 000000000..5d6b4396b --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/validating-webhook.yaml @@ -0,0 +1,28 @@ +apiVersion: admissionregistration.k8s.io/v1 +kind: ValidatingWebhookConfiguration +metadata: + name: {{ include "airlock-microgateway.operator.fullname" . }}-webhook-{{ .Release.Namespace }} + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + annotations: + cert-manager.io/inject-ca-from: {{ .Release.Namespace }}/{{ include "airlock-microgateway.operator.fullname" . }}-serving-cert + {{- with .Values.commonAnnotations }} + {{- toYaml . | nindent 4 }} + {{- end }} +webhooks: +{{- range $webhook := (include "airlock-microgateway-operator.validatingWebhooks" .) | fromYamlArray }} +- {{ toYaml $webhook | indent 2 | trim }} + {{- with $.Values.operator.watchNamespaceSelector }} + namespaceSelector: + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with $.Values.operator.watchNamespaces }} + namespaceSelector: + matchExpressions: + - key: kubernetes.io/metadata.name + operator: In + values: + {{- toYaml . | nindent 10 }} + {{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml new file mode 100644 index 000000000..477ea839f --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/webhook-service.yaml @@ -0,0 +1,23 @@ +apiVersion: v1 +kind: Service +metadata: + name: airlock-microgateway-operator-webhook + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.operator.serviceLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + ports: + - appProtocol: https + name: webhook + port: 443 + protocol: TCP + targetPort: 9443 + selector: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml b/charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml new file mode 100644 index 000000000..81b41acf5 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/operator/xds-service.yaml @@ -0,0 +1,24 @@ +apiVersion: v1 +kind: Service +metadata: + name: airlock-microgateway-operator-xds + namespace: {{ .Release.Namespace }} + labels: + {{- include "airlock-microgateway.operator.labels" . | nindent 4 }} + {{- with .Values.operator.serviceLabels }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- with mustMerge .Values.operator.serviceAnnotations .Values.commonAnnotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + ports: + - appProtocol: grpc + name: xds + port: 13377 + protocol: TCP + targetPort: 13377 + selector: + {{- include "airlock-microgateway.operator.selectorLabels" . | nindent 4 }} + operator.microgateway.airlock.com/isLeader: "true" diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml new file mode 100644 index 000000000..93bd4cd1b --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/tests/rbac.yaml @@ -0,0 +1,143 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: "{{ include "airlock-microgateway.fullname" . }}-tests" +subjects: +- kind: ServiceAccount + name: "{{ include "airlock-microgateway.fullname" . }}-tests" +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +rules: +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways + resourceNames: + - "{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway" + verbs: + - get + - list + - watch + - delete +- apiGroups: + - microgateway.airlock.com + resources: + - sidecargateways + verbs: + - create +- apiGroups: + - "" + resources: + - events + verbs: + - list +- apiGroups: + - "apps" + resources: + - deployments + resourceNames: + - "{{ include "airlock-microgateway.operator.fullname" . }}" + verbs: + - get + - list + - watch +- apiGroups: + - "apps" + resources: + - statefulsets + - statefulsets/scale + resourceNames: + - "{{ include "airlock-microgateway.fullname" . }}-test-backend" + verbs: + - get + - list + - watch + - patch +- apiGroups: + - "" + resources: + - pods + - pods/log + - pods/status + - pods/attach + resourceNames: + - "{{ include "airlock-microgateway.fullname" . }}-test-backend-0" + - "{{ include "airlock-microgateway.fullname" . }}-test-valid-request" + - "{{ include "airlock-microgateway.fullname" . }}-test-injection-request" + verbs: + - get + - list + - create + - watch + - delete +- apiGroups: + - "" + resources: + - pods + verbs: + - create +{{- if .Values.operator.watchNamespaceSelector }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" +subjects: + - kind: ServiceAccount + name: "{{ include "airlock-microgateway.fullname" . }}-tests" + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + labels: + app.kubernetes.io/component: tests + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + name: "{{ include "airlock-microgateway.fullname" . }}-tests-{{ .Release.Namespace }}" +rules: +- apiGroups: + - "" + resources: + - namespaces + verbs: + - get + - list +{{- end }} +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/service.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/service.yaml new file mode 100644 index 000000000..30ddc278d --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/tests/service.yaml @@ -0,0 +1,23 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: Service +metadata: + name: "{{ include "airlock-microgateway.fullname" . }}-test-service" + namespace: {{ .Release.Namespace }} + labels: + app: test-service + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} +spec: + selector: + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} + ports: + - name: http + port: 8080 + targetPort: 8080 +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml new file mode 100644 index 000000000..710a7b9f6 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/tests/statefulset.yaml @@ -0,0 +1,56 @@ +{{- if .Values.tests.enabled -}} +apiVersion: apps/v1 +kind: StatefulSet +metadata: + name: "{{ include "airlock-microgateway.fullname" . }}-test-backend" + namespace: {{ .Release.Namespace }} + labels: + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} +spec: + serviceName: nginx + replicas: 0 + selector: + matchLabels: + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 6 }} + template: + metadata: + annotations: + k8s.v1.cni.cncf.io/networks: default/airlock-microgateway-cni + labels: + sidecar.microgateway.airlock.com/inject: "true" + sidecar.istio.io/inject: "false" + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + app: "{{ include "airlock-microgateway.fullname" . }}-test-backend" + {{- include "airlock-microgateway.sharedLabels" . | nindent 8 }} + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 8 }} + spec: + containers: + - image: cgr.dev/chainguard/nginx + name: nginx + ports: + - containerPort: 8080 + volumeMounts: + - mountPath: /var/lib/nginx/tmp/ + name: nginx-tmp + - mountPath: /var/run + name: nginx-run + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 12 }} + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + volumes: + - emptyDir: {} + name: nginx-tmp + - emptyDir: {} + name: nginx-run +{{- end -}} \ No newline at end of file diff --git a/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml b/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml new file mode 100644 index 000000000..ab82abea7 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/templates/tests/test-install.yaml @@ -0,0 +1,227 @@ +{{- if .Values.tests.enabled -}} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ include "airlock-microgateway.fullname" . }}-test-install" + namespace: {{ .Release.Namespace }} + labels: + app.kubernetes.io/component: test-install + app.kubernetes.io/name: {{ include "airlock-microgateway.name" . }}-tests + sidecar.istio.io/inject: "false" + {{- include "airlock-microgateway.sharedLabels" . | nindent 4 }} + {{- include "airlock-microgateway.sharedSelectorLabels" . | nindent 4 }} + annotations: + helm.sh/hook: test + helm.sh/hook-delete-policy: before-hook-creation +spec: + restartPolicy: Never + containers: + - name: test + image: "bitnami/kubectl:{{ .Capabilities.KubeVersion.Major }}.{{ .Capabilities.KubeVersion.Minor }}" + securityContext: + {{- include "airlock-microgateway.restrictedSecurityContext" . | nindent 6 }} + command: + - sh + - -c + - | + set -eu + + clean_up() { + echo "" + echo "### Clean up test resources" + kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true + echo "" + echo "### Scale down '{{ include "airlock-microgateway.fullname" . }}-test-backend'" + kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=60s + sleep 3s + echo "" + } + + fail() { + echo "" + echo "### Error: ${1}" + echo "" + + if kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway >/dev/null 2>&1; then + echo "" + echo 'Microgateway Sidecargateway status:' + kubectl get -n {{ .Release.Namespace }} sidecargateway.microgateway.airlock.com/{{ include "airlock-microgateway.fullname" . }}-test-sidecargateway -o jsonpath-as-json='{.status}' || true + echo "" + echo "" + fi + + if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 >/dev/null 2>&1; then + echo "Pod '{{ include "airlock-microgateway.fullname" . }}-test-backend-0':" + kubectl describe -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 || true + echo "" + echo "" + echo 'Logs of Nginx container:' + kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c nginx --tail 5 || true + echo "" + echo "" + # Wait for engine logs + sleep 10s + echo 'Logs of Microgateway Engine container:' + kubectl logs -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -c airlock-microgateway-engine --tail 5 || true + fi + + exit 1 + } + + create_sidecargateway() { + # create SidecarGateway resource for testing purposes + kubectl delete --ignore-not-found=true -n {{ .Release.Namespace }} sidecargateways.microgateway.airlock.com {{ include "airlock-microgateway.fullname" . }}-test-sidecargateway || true + kubectl apply -f - </dev/null 2>&1; do sleep 1s; i=$((i+1)); done + kubectl logs -f -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request + kubectl delete pod --ignore-not-found=true -n {{ .Release.Namespace }} {{ include "airlock-microgateway.fullname" . }}-test-valid-request + } + + {{- if .Values.operator.watchNamespaceSelector }} + echo "### Verify that Namespace Selector matches Namespace '{{ .Release.Namespace }}'" + if ! kubectl get namespace -l '{{ include "airlock-microgateway.watchNamespaceSelector.labelQuery" .Values.operator.watchNamespaceSelector }}' | grep -q {{ .Release.Namespace }}; then + labels=$(kubectl get namespace {{ .Release.Namespace }} -o jsonpath={.metadata.labels} | jq | awk '{print " " $0}') + fail {{printf `"Operator namespace '%s' is not part of the operator's watch scope. To execute 'helm test', the selector configured in the helm value 'operator.watchNamespaceSelector' must match the namespace's labels:\n* Current selector:\n%s\n\n* Current labels:\n$labels\n###"` + .Release.Namespace + (replace "\"" "\\\"" (replace "\n" "\\n" (.Values.operator.watchNamespaceSelector | toPrettyJson | indent 2))) + }} + fi + echo "" + {{- end }} + + trap clean_up EXIT + echo "" + + echo "### Waiting for Microgateway Operator Deployments to be ready" + if ! kubectl rollout status -n {{ .Release.Namespace }} --timeout=90s \ + deployments/{{ include "airlock-microgateway.operator.fullname" . }}; then + fail 'Timout occurred' + fi + echo "" + + echo "### Scale '{{ include "airlock-microgateway.fullname" . }}-test-backend' to '1' replica" + # scale to zero replicas to ensure no pods are present from previous runs + kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=0 --timeout=10s + kubectl scale -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --replicas=1 --timeout=10s + echo "" + + echo "### Waiting for backend pod" + i=0 + while true; do + if kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0; then + break + elif [ $i -gt 3 ]; then + fail 'Pod not ready' + fi + sleep 2s + i=$((i+1)) + done + + echo "### Checking Microgateway Engine sidecar container was injected" + if ! kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.spec.containers[?(@.name=="airlock-microgateway-engine")]}' | grep -q "airlock-microgateway-engine"; then + fail 'Microgateway Engine sidecar container not injected' + fi + echo "True" + echo "" + + echo "### Checking for valid license" + i=0 + while true; do + if [ "$(kubectl get -n {{ .Release.Namespace }} pods/{{ include "airlock-microgateway.fullname" . }}-test-backend-0 -o jsonpath='{.metadata.labels.sidecar\.microgateway\.airlock\.com/licensed}')" = 'true' ]; then + break + elif [ $i -gt 30 ]; then + fail 'Microgateway license is missing or invalid' + fi + sleep 2s + i=$((i+1)) + done + echo "True" + echo "" + + echo "### Create SidecarGateway resource for testing" + if ! create_sidecargateway ; then + fail 'Creation of SidecarGateway resource failed' + fi + echo "" + + echo "### Waiting for '{{ include "airlock-microgateway.fullname" . }}-test-backend' to be ready" + if ! kubectl rollout status -n {{ .Release.Namespace }} statefulset/{{ include "airlock-microgateway.fullname" . }}-test-backend --timeout=90s; then + fail 'Timout occurred' + fi + echo "" + + echo "### Waiting for 'engine-config-valid' condition" + if ! kubectl wait -n {{ .Release.Namespace }} pods --field-selector=metadata.name={{ include "airlock-microgateway.fullname" . }}-test-backend-0 --timeout=90s --for=condition=microgateway.airlock.com/engine-config-valid=True; then + fail 'Configuration was never accepted by the Microgateway Engine' + fi + sleep 5s + echo "" + echo "" + + echo "### Checking whether a valid request is successful and returns HTTP status code '200'" + out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/" || true) + echo "Response:" + echo "${out}" + if ! echo "${out}" | grep -q "200 OK"; then + fail 'A valid request was not successful' + fi + echo "" + echo "" + + echo "### Checking whether a request with an injection attack is blocked and returns HTTP status code '400'" + out=$(curl -vsS --retry 3 --retry-connrefused --connect-timeout 10 "http://{{ include "airlock-microgateway.fullname" . }}-test-service:8080/?token='%20UnION%20all%20select%20A" || true) + echo "Response:" + echo "${out}" + if ! echo "${out}" | grep -q "400 Bad Request"; then + fail 'A malicious request was not blocked' + fi + echo "" + echo "" + + echo "### Installation of '{{ include "airlock-microgateway.fullname" . }}' succeeded" + exit 0 + serviceAccountName: "{{ include "airlock-microgateway.fullname" . }}-tests" +{{- end -}} diff --git a/charts/airlock/microgateway/4.3.0/values.schema.json b/charts/airlock/microgateway/4.3.0/values.schema.json new file mode 100644 index 000000000..173d6b084 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/values.schema.json @@ -0,0 +1,540 @@ +{ + "$schema": "https://json-schema.org/draft/2020-12/schema", + "type": "object", + "properties": { + "nameOverride": { + "type": "string" + }, + "fullnameOverride": { + "type": "string" + }, + "commonLabels": { + "$ref": "#/definitions/StringMap" + }, + "commonAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "crds": { + "type": "object", + "properties": { + "skipVersionCheck": { + "type": "boolean" + } + }, + "additionalProperties": false + }, + "imagePullSecrets": { + "type": "array", + "items": { + "type": "object", + "properties": { + "name": { + "type": "string", + "minLength": 1 + } + }, + "required": [ + "name" + ], + "additionalProperties": true + } + }, + "operator": { + "type": "object", + "properties": { + "replicaCount": { + "type": "integer", + "minimum": 0 + }, + "updateStrategy": { + "$ref": "#/definitions/UpdateStrategy" + }, + "image": { + "$ref": "#/definitions/Image" + }, + "podAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "podLabels": { + "$ref": "#/definitions/StringMap" + }, + "serviceAnnotations": { + "$ref": "#/definitions/StringMap" + }, + "serviceLabels": { + "$ref": "#/definitions/StringMap" + }, + "resources": { + "type": "object" + }, + "nodeSelector": { + "$ref": "#/definitions/StringMap" + }, + "tolerations": { + "type": "array", + "items": { + "type": "object" + } + }, + "affinity": { + "type": "object" + }, + "config": { + "type": "object", + "properties": { + "logLevel": { + "type": "string", + "enum": [ + "debug", + "info", + "warn", + "error" + ] + } + }, + "required": [ + "logLevel" + ], + "additionalProperties": false + }, + "serviceAccount": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "annotations": { + "$ref": "#/definitions/StringMap" + }, + "name": { + "type": "string" + } + }, + "required": [ + "annotations", + "create", + "name" + ], + "additionalProperties": false + }, + "watchNamespaces": { + "type": "array", + "items": { + "type": "string" + } + }, + "watchNamespaceSelector": { + "$ref": "#/definitions/LabelSelector" + }, + "rbac": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + } + }, + "required": [ + "create" + ], + "additionalProperties": false + }, + "serviceMonitor": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "labels": { + "$ref": "#/definitions/StringMap" + } + }, + "required": [ + "create" + ], + "additionalProperties": false + } + }, + "oneOf": [ + { + "properties": { + "watchNamespaces": { + "minItems": 1 + }, + "watchNamespaceSelector": { + "additionalProperties": false + } + } + }, + { + "properties": { + "watchNamespaces": { + "maxItems": 0 + }, + "watchNamespaceSelector": { + "$ref": "#/definitions/LabelSelector" + } + } + } + ], + "required": [ + "affinity", + "config", + "image", + "updateStrategy", + "nodeSelector", + "podAnnotations", + "podLabels", + "rbac", + "replicaCount", + "resources", + "serviceAccount", + "serviceAnnotations", + "serviceLabels", + "serviceMonitor", + "tolerations" + ], + "additionalProperties": false + }, + "engine": { + "type": "object", + "properties": { + "image": { + "$ref": "#/definitions/Image" + }, + "resources": { + "type": "object" + }, + "sidecar": { + "type": "object", + "properties":{ + "podMonitor": { + "type": "object", + "properties": { + "create": { + "type": "boolean" + }, + "labels": { + "$ref": "#/definitions/StringMap" + } + }, + "required": [ + "create" + ], + "additionalProperties": false + } + }, + "required": [ + "podMonitor" + ], + "additionalProperties": false + } + }, + "required": [ + "image", + "resources", + "sidecar" + ], + "additionalProperties": false + }, + "networkValidator": { + "type": "object", + "properties": { + "image": { + "$ref": "#/definitions/Image" + } + }, + "required": [ + "image" + ], + "additionalProperties": false + }, + "sessionAgent": { + "type": "object", + "properties": { + "image": { + "$ref": "#/definitions/Image" + }, + "resources": { + "type": "object" + } + }, + "required": [ + "image", + "resources" + ], + "additionalProperties": false + }, + "license": { + "type": "object", + "properties": { + "secretName": { + "type": "string", + "minLength": 1 + } + }, + "required": [ + "secretName" + ], + "additionalProperties": false + }, + "dashboards": { + "type": "object", + "properties" : { + "create": { + "type": "boolean" + }, + "config": { + "type": "object", + "properties": { + "grafana": { + "type": "object", + "properties": { + "folderAnnotation": { + "$ref": "#/definitions/NameValuePair" + }, + "dashboardLabel": { + "$ref": "#/definitions/NameValuePair" + } + }, + "required": [ + "folderAnnotation", + "dashboardLabel" + ], + "additionalProperties": false + } + }, + "required": [ + "grafana" + ], + "additionalProperties": false + }, + "instances": { + "type": "object", + "properties": { + "overview": { + "$ref": "#/definitions/DashboardInstance" + }, + "license" : { + "$ref": "#/definitions/DashboardInstance" + }, + "blockMetrics" : { + "$ref": "#/definitions/DashboardInstance" + }, + "blockLogs" : { + "$ref": "#/definitions/DashboardInstance" + } + }, + "required": [ + "overview", + "license", + "blockMetrics", + "blockLogs" + ], + "additionalProperties": false + } + }, + "required": [ + "create", + "config", + "instances" + ], + "additionalProperties": false + }, + "tests": { + "type": "object", + "properties": { + "enabled": { + "type": "boolean" + } + }, + "required": [ + "enabled" + ], + "additionalProperties": false + }, + "global": { + "type": "object" + } + }, + "required": [ + "commonAnnotations", + "commonLabels", + "crds", + "engine", + "fullnameOverride", + "imagePullSecrets", + "license", + "nameOverride", + "operator", + "networkValidator", + "sessionAgent", + "dashboards", + "tests" + ], + "additionalProperties": false, + "definitions": { + "StringMap": { + "type": "object", + "additionalProperties": { + "type": "string" + } + }, + "Image": { + "type": "object", + "properties": { + "repository": { + "type": "string", + "minLength": 1 + }, + "tag": { + "type": "string" + }, + "digest": { + "type": "string", + "pattern": "^$|^sha256:[a-f0-9]{64}$" + }, + "pullPolicy": { + "type": "string", + "enum": [ + "Always", + "IfNotPresent", + "Never" + ] + } + }, + "required": [ + "digest", + "pullPolicy", + "repository", + "tag" + ], + "additionalProperties": false + }, + "LabelSelector": { + "type": "object", + "properties": { + "matchExpressions": { + "type": "array", + "items": { + "type": "object", + "required": [ + "key", + "operator" + ], + "properties": { + "key": { + "type": "string" + }, + "operator": { + "type": "string" + }, + "values": { + "type": "array", + "items": { + "type": "string" + } + } + }, + "additionalProperties": false + } + }, + "matchLabels": { + "$ref": "#/definitions/StringMap" + } + }, + "additionalProperties": false + }, + "UpdateStrategy": { + "type": "object", + "oneOf" : [ + { + "properties": { + "type": { + "$ref": "#/definitions/RecreateType" + } + }, + "required": [ + "type" + ], + "additionalProperties": false + }, + { + "properties": { + "type": { + "$ref": "#/definitions/RollingUpdateType" + }, + "rollingUpdate": { + "$ref": "#/definitions/RollingUpdate" + } + }, + "required": [ + "type" + ], + "additionalProperties": false + } + ] + }, + "RecreateType": { + "type": "string", + "enum": [ + "Recreate" + ] + }, + "RollingUpdateType": { + "type": "string", + "enum": [ + "RollingUpdate" + ] + }, + "RollingUpdate": { + "type": "object", + "properties": { + "maxSurge": { + "type": ["integer", "string"], + "minimum": 0, + "pattern": "^\\d+%?$" + }, + "maxUnavailable": { + "type": ["integer", "string"], + "minimum": 0, + "pattern": "^\\d+%?$" + } + }, + "anyOf": [ + {"required": ["maxSurge"]}, + {"required": ["maxUnavailable"]} + ], + "additionalProperties": false + }, + "DashboardInstance" : { + "type" : "object", + "properties" : { + "create" : { + "type" : "boolean" + } + }, + "required" : [ + "create" + ], + "additionalProperties": false + }, + "NameValuePair" : { + "type" : "object", + "properties" : { + "name" : { + "type": "string", + "minLength": 1 + }, + "value" : { + "type" : "string", + "minLength": 1 + } + }, + "required" : [ + "name", + "value" + ], + "additionalProperties": false + } + } +} diff --git a/charts/airlock/microgateway/4.3.0/values.yaml b/charts/airlock/microgateway/4.3.0/values.yaml new file mode 100644 index 000000000..b13232aa9 --- /dev/null +++ b/charts/airlock/microgateway/4.3.0/values.yaml @@ -0,0 +1,213 @@ +# -- Allows overriding the name to use instead of "microgateway". +nameOverride: "" +# -- Allows overriding the name to use as full name of resources. +fullnameOverride: "" +# -- Labels to add to all resources. +commonLabels: {} +# -- Annotations to add to all resources. +commonAnnotations: {} +# -- ImagePullSecrets to use when pulling images. +imagePullSecrets: [] +# - name: myRegistryKeySecretName + +crds: + # -- Whether to skip the sanity check which prevents installing/upgrading the helm chart in a cluster with outdated Airlock Microgateway CRDs. + # The check aims to prevent unexpected behavior and issues due to Helm v3 not automatically upgrading CRDs which are already present in the cluster + # when performing a "helm install/upgrade". + skipVersionCheck: false +operator: + # -- Number of replicas for the operator Deployment. + replicaCount: 2 + # -- Specifies the operator update strategy. + updateStrategy: + type: RollingUpdate + # Specifies the Airlock Microgateway Operator image. + image: + # -- Image repository from which to pull the Airlock Microgateway Operator image. + repository: "quay.io/airlock/microgateway-operator" + # -- Image tag to pull. + tag: "4.3.0" + # -- SHA256 image digest to pull (in the format "sha256:c79ee3f85862fb386e9dd62b901b607161d27807f512d7fbdece05e9ee3d7c63"). + # Overrides tag when specified. + digest: "sha256:dc6f0f9a11d0336c10f6b8a5c7f64d98ac91bd90c49aa1dc4fe7b68cfdea8217" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Annotations to add to all Pods. + podAnnotations: {} + # -- Labels to add to all Pods. + podLabels: {} + # -- Annotations to add to the Service. + serviceAnnotations: {} + # prometheus.io/scrape: "true" + # prometheus.io/port: "8080" + + # -- Labels to add to the Service. + serviceLabels: {} + # -- Resource restrictions to apply to the operator container. + resources: {} + # We recommend at least the following resource specification. + # limits: + # cpu: 1000m + # memory: 512Mi + # requests: + # cpu: 100m + # memory: 512Mi + + # -- Custom nodeSelector to apply to the operator Deployment in order to constrain its Pods to certain nodes. + nodeSelector: {} + # -- Custom tolerations to apply to the operator Deployment in order to allow its Pods to run on tainted nodes. + tolerations: [] + # -- Custom affinity to apply to the operator Deployment. Used to influence the scheduling. + affinity: {} + # Parameters for the operator configuration. + config: + # -- Operator application log level. + logLevel: "info" + # Configures the generation of the ServiceAccount. + serviceAccount: + # -- Whether a ServiceAccount should be created. + create: true + # -- Annotations to add to the ServiceAccount. + annotations: {} + # -- Name of the ServiceAccount to use. + # If not set and create is true, a name is generated using the fullname template. + name: "" + # -- Allows to restrict the operator to specific namespaces, depending on your needs. + # For a `OwnNamespace` or `SingleNamespace` installation the list may only contain one namespace (e.g., `watchNamespaces: ["airlock-microgateway-system"]`). + # In case of the `OwnNamespace` installation mode the specified namespace should be equal to the installation namespace. + # For a static `MultiNamespace` installation, the complete list of namespaces must be provided in the `watchNamespaces`. + # An `AllNamespaces` installation or the usage of the `watchNamespaceSelector` requires the `watchNamespaces` to be empty. + # Regardless of the installation modes supported by `watchNamespaces`, RBAC is created only namespace-scoped (using Roles and RoleBindings) in the respective namespaces. + # Please note that this feature requires a Premium license. + watchNamespaces: [] + # -- Allows to dynamically select watch namespaces of the operator and the scope of the webhooks based on a Namespace label selector. + # It is able to detect and reconcile resources in all namespaces that match the label selector automatically, even for new namespaces, without restarting the operator. + # This facilitates a dynamic `MultiNamespace` installation mode, but still requires cluster-scoped permissions (i.e., ClusterRoles and ClusterRoleBindings). + # An `AllNamespaces` installation or the usage of the `watchNamespaces` requires the `watchNamespaceSelector` to be empty. + # Please note that this feature requires a Premium license. + watchNamespaceSelector: {} + # For further examples, see: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#resources-that-support-set-based-requirements. + # matchLabels: + # microgateway.airlock.com/enable: "true" + # matchExpressions: + # - { key: environment, operator: NotIn, values: [dev] } + + # Configures the generation of Role and RoleBinding as well as ClusterRoles and ClusterRoleBinding pairs for the ServiceAccount specified above. + rbac: + # -- Whether to create RBAC resources which are required for the Airlock Microgateway Operator to function. + create: true + # Configures the generation of a Prometheus Operator ServiceMonitor. + serviceMonitor: + # -- Whether to create a ServiceMonitor resource for monitoring. + create: false + # -- Labels to add to the ServiceMonitor. + labels: {} + # release: "" +engine: + # Specifies the Airlock Microgateway Engine image. + image: + # -- Image repository from which to pull the Airlock Microgateway Engine image. + repository: "quay.io/airlock/microgateway-engine" + # -- Image tag to pull. + tag: "4.3.0" + # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). + # Overrides tag when specified. + digest: "sha256:f442143294f3138965c9fa2734cafd39ebebe8e289600332b12f8a59c23dd9ef" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Resource restrictions to apply to the Airlock Microgateway Engine container. + resources: {} + # We recommend at least the following resource specification. + # limits: + # cpu: 500m + # memory: 128Mi + # requests: + # cpu: 10m + # memory: 40Mi + + # Additional configuration when deployed as a sidecar. + sidecar: + # Configures the generation of a Prometheus Operator PodMonitor. + podMonitor: + # -- Whether to create a PodMonitor resource for monitoring. + create: false + # -- Labels to add to the PodMonitor. + labels: {} + # release: "" +networkValidator: + # Specifies the Airlock Microgateway Network Validator image to be injected as an init-container. + image: + # -- Image repository from which to pull the busybox image for the Airlock Microgateway Network Validator init-container. + repository: "cgr.dev/chainguard/busybox" + # -- Image tag to pull. + tag: "" + # -- SHA256 image digest to pull (in the format "sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6"). + # Overrides tag when specified. + digest: "sha256:7d87405b123c89058a0b64ca9393c45a1366a6a580aced1def900a812beb29f6" + # -- Pull policy for this image. + pullPolicy: IfNotPresent +sessionAgent: + # Specifies the Airlock Microgateway Session Agent image. + image: + # -- Image repository from which to pull the Airlock Microgateway Session Agent image. + repository: "quay.io/airlock/microgateway-session-agent" + # -- Image tag to pull. + tag: "4.3.0" + # -- SHA256 image digest to pull (in the format "sha256:a3051f42d3013813b05f7513bb86ed6a3209cb3003f1bb2f7b72df249aa544d3"). + # Overrides tag when specified. + digest: "sha256:579dfded99145f9c2c1491ff1aeccb08721d63239a8b7f61bb9f455e17e968b2" + # -- Pull policy for this image. + pullPolicy: IfNotPresent + # -- Resource restrictions to apply to the Airlock Microgateway Session Agent container. + resources: {} + # We recommend at least the following resource specification. + # limits: + # cpu: 150m + # memory: 32Mi + # requests: + # cpu: 10m + # memory: 8Mi +license: + # -- Name of the secret containing the "microgateway-license.txt" key. + secretName: "airlock-microgateway-license" +# Creates dashboards in the form of ConfigMaps that can be imported +# by Grafana using its sidecar setup. +dashboards: + # -- Whether to create any ConfigMaps containing Grafana dashboards to import. + create: false + config: + # Configures the necessary label and annotations along with their values + # to enable Grafana to correctly identify the ConfigMaps containing + # dashboards and file them within a dedicated folder in the dashboard overview. + # These settings need to match the Grafana sidecar configuration. + grafana: + folderAnnotation: + # -- Name of the annotation containing the folder name to file dashboards into. + name: "grafana_folder" + # -- Name of the folder dashboards are filed into within the Grafana UI. + value: "Airlock Microgateway" + dashboardLabel: + # -- Name of the label that lets Grafana identify ConfigMaps that represent dashboards. + name: "grafana_dashboard" + # -- Value of the label that lets Grafana identify ConfigMaps that represent dashboards. + value: "1" + instances: + # Available dashboard instances that can be individually created/deployed. + overview: + # -- Whether to create the overview dashboard. + create: true + license: + # -- Whether to create the license dashboard. + create: true + blockMetrics: + # -- Whether to create the block metrics dashboard. + create: true + blockLogs: + # -- Whether to create the block logs dashboard. + create: true +# Check whether the installation of the Airlock Microgateway Helm Chart was successful. +# Requires a secret with a valid Airlock Microgateway license key already to be present. +tests: + # -- Whether additional resources required for running `helm test` should be created (e.g. Roles and ServiceAccounts). + # If set to false, `helm test` will not run any tests. + enabled: false diff --git a/charts/cerbos/cerbos/0.37.0/.helmignore b/charts/cerbos/cerbos/0.37.0/.helmignore new file mode 100644 index 000000000..0e8a0eb36 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/.helmignore @@ -0,0 +1,23 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ diff --git a/charts/cerbos/cerbos/0.37.0/Chart.yaml b/charts/cerbos/cerbos/0.37.0/Chart.yaml new file mode 100644 index 000000000..144e01a2d --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/Chart.yaml @@ -0,0 +1,27 @@ +annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Cerbos + catalog.cattle.io/kube-version: '>=1.23.0-0' + catalog.cattle.io/release-name: cerbos +apiVersion: v2 +appVersion: 0.37.0 +description: A Helm chart to deploy Cerbos. Cerbos is an open core, language agnostic, + scalable solution that makes user permissions and authorization simple to implement + and manage by writing context-aware access control policies for your application + resources. +home: https://cerbos.dev +icon: file://assets/icons/cerbos.png +keywords: +- abac +- authorization +- developer tools +- policies +- rbac +- security +kubeVersion: '>=1.23.0-0' +maintainers: +- email: help+helm@cerbos.dev + name: Cerbos authors +name: cerbos +type: application +version: 0.37.0 diff --git a/charts/cerbos/cerbos/0.37.0/README.md b/charts/cerbos/cerbos/0.37.0/README.md new file mode 100644 index 000000000..4276df6a0 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/README.md @@ -0,0 +1,9 @@ +Cerbos Helm Chart +================= + +Cerbos is the open core, language-agnostic, scalable authorization solution that makes user permissions and authorization simple to implement and manage by writing context-aware access control policies for your application resources. + +* [Cerbos website](https://cerbos.dev) +* [Cerbos documentation](https://docs.cerbos.dev) +* [Cerbos GitHub repository](https://github.com/cerbos/cerbos) +* [Cerbos Slack community](http://go.cerbos.io/slack) diff --git a/charts/cerbos/cerbos/0.37.0/app-readme.md b/charts/cerbos/cerbos/0.37.0/app-readme.md new file mode 100644 index 000000000..5c1012779 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/app-readme.md @@ -0,0 +1,5 @@ +## Cerbos + +Implement roles and permissions in your application in minutes with Cerbos. Cerbos is a plug and play collaborative authorization service for developer, product and security teams. + +A guide for how to deploy Cerbos can be found on the [Cerbos Documentation](https://docs.cerbos.dev/cerbos/latest/installation/helm) site. diff --git a/charts/cerbos/cerbos/0.37.0/templates/NOTES.txt b/charts/cerbos/cerbos/0.37.0/templates/NOTES.txt new file mode 100644 index 000000000..1e0b7dc36 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/NOTES.txt @@ -0,0 +1,17 @@ +You have successfully deployed Cerbos. + +You can get started with the API by accessing the Cerbos OpenAPI definitions as follows: +{{- $scheme := (include "cerbos.httpScheme" .) -}} +{{- if contains "NodePort" .Values.service.type }} + export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ include "cerbos.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}") + echo {{ $scheme }}://$NODE_IP:$NODE_PORT/schema/swagger.json +{{- 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 {{ .Release.Namespace }} svc -w {{ include "cerbos.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ include "cerbos.fullname" . }} --template "{{"{{ range (index .status.loadBalancer.ingress 0) }}{{.}}{{ end }}"}}") + echo {{ $scheme }}://$SERVICE_IP:{{ .Values.service.port }}/schema/swagger.json +{{- else if contains "ClusterIP" .Values.service.type }} + echo "Download OpenAPI definition by accesing {{ $scheme }}://127.0.0.1:{{ .Values.service.httpPort }}/schema/swagger.json" + kubectl --namespace {{ .Release.Namespace }} port-forward svc/{{ include "cerbos.fullname" . }} {{ .Values.service.httpPort }} +{{- end }} diff --git a/charts/cerbos/cerbos/0.37.0/templates/_helpers.tpl b/charts/cerbos/cerbos/0.37.0/templates/_helpers.tpl new file mode 100644 index 000000000..daab30736 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/_helpers.tpl @@ -0,0 +1,191 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "cerbos.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 "cerbos.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 "cerbos.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "cerbos.labels" -}} +helm.sh/chart: {{ include "cerbos.chart" . }} +{{ include "cerbos.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- with .Values.commonLabels }} +{{ toYaml . }} +{{- end }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "cerbos.selectorLabels" -}} +app.kubernetes.io/name: {{ include "cerbos.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Name of the secret used to read the TLS certificates from +*/}} +{{- define "cerbos.tlsSecretName" -}} +{{ coalesce .Values.cerbos.tlsSecretName .Values.certManager.certSpec.secretName "None" }} +{{- end }} + +{{/* +Determine the scheme based on whether the TLS secret is defined or not +*/}} +{{- define "cerbos.httpScheme" -}} +{{- $tlsDisabled := (eq (include "cerbos.tlsSecretName" .) "None") -}} +{{- if $tlsDisabled -}} +http +{{- else -}} +https +{{- end -}} +{{- end }} + +{{/* +Prometheus annotations +*/}} +{{- define "cerbos.promAnnotations" -}} +prometheus.io/scrape: "true" +prometheus.io/port: "{{ .Values.cerbos.httpPort }}" +prometheus.io/path: "/_cerbos/metrics" +prometheus.io/scheme: {{ include "cerbos.httpScheme" . }} +{{- end }} + +{{/* +Generate pod annotations based on config +*/}} +{{- define "cerbos.podAnnotations" -}} +{{- $annotations := mustMergeOverwrite .Values.podAnnotations (dict "checksum/config" (include "cerbos.config" . | sha256sum)) -}} +{{- if .Values.cerbos.prometheusPodAnnotationsEnabled -}} +{{- $promAnnotations := (include "cerbos.promAnnotations" .)| fromYaml -}} +{{- $annotations = mustMergeOverwrite $annotations $promAnnotations -}} +{{- end -}} +annotations: + {{- $annotations | toYaml | nindent 2 }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "cerbos.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "cerbos.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} + + +{{/* +Default configuration if none is provided +*/}} +{{- define "cerbos.defaultConfig" -}} +storage: + driver: "disk" + disk: + directory: /work + watchForChanges: false +{{- end }} + + +{{/* +Configuration derived from values provided by the user +*/}} +{{- define "cerbos.derivedConfig" -}} +{{- $tlsDisabled := (eq (include "cerbos.tlsSecretName" .) "None") -}} +server: + httpListenAddr: ":{{ .Values.cerbos.httpPort }}" + grpcListenAddr: ":{{ .Values.cerbos.grpcPort }}" + {{- if not $tlsDisabled }} + tls: + cert: /certs/tls.crt + key: /certs/tls.key + caCert: /certs/ca.crt + {{- end }} +{{- end }} + + +{{/* +Merge the configurations to obtain the final configuration file +*/}} +{{- define "cerbos.config" -}} +{{- $defaultConf := (include "cerbos.defaultConfig" .) | fromYaml -}} +{{- $derivedConf := (include "cerbos.derivedConfig" .) | fromYaml -}} +{{ mustMergeOverwrite $defaultConf .Values.cerbos.config $derivedConf | toYaml }} +{{- end }} + +{{/* +Detect if hub driver is used with default config +*/}} +{{- define "cerbos.defaultHubDriverEnabled" -}} +{{- $isBundleDriver := (eq (dig "config" "storage" "driver" "" .Values.cerbos) "bundle") -}} +{{- $isHubDriver := (eq (dig "config" "storage" "driver" "" .Values.cerbos) "hub") -}} +{{- $isBundleStorage := (or $isBundleDriver $isHubDriver) -}} +{{- $isDefaultTmp := (eq (dig "config" "storage" "bundle" "remote" "tempDir" "" .Values.cerbos) "") -}} +{{- $isDefaultCache := (eq (dig "config" "storage" "bundle" "remote" "cacheDir" "" .Values.cerbos) "") -}} +{{- if (and $isBundleStorage $isDefaultTmp $isDefaultCache) -}}yes{{- else -}}no{{- end -}} +{{- end }} + +{{/* +The image reference to use in pods +*/}} +{{- define "cerbos.image" -}} +"{{ .Values.image.repository }} +{{- with .Values.image.digest -}} +@{{ . }} +{{- else -}} +:{{ .Values.image.tag | default .Chart.AppVersion }} +{{- end -}} +" +{{- end }} + +{{/* +Topology spread constraints with label selector injected +*/}} +{{- define "cerbos.topologySpreadConstraints" -}} +{{- if .Values.topologySpreadConstraints }} +{{- $defaultLabels := (fromYaml (include "cerbos.selectorLabels" $)) }} +{{- $defaultLabelSelector := (dict "labelSelector" (dict "matchLabels" $defaultLabels)) }} +{{- $constraints := list }} +{{- range $c := .Values.topologySpreadConstraints }} +{{- if (hasKey $c "labelSelector") }} +{{- $constraints = (append $constraints $c) }} +{{- else }} +{{- $constraints = (append $constraints (mergeOverwrite $c $defaultLabelSelector)) }} +{{- end }} +{{- end }} +topologySpreadConstraints: +{{ toYaml $constraints | indent 2 }} +{{- end }} +{{- end }} diff --git a/charts/cerbos/cerbos/0.37.0/templates/certificate.yaml b/charts/cerbos/cerbos/0.37.0/templates/certificate.yaml new file mode 100644 index 000000000..250907056 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/certificate.yaml @@ -0,0 +1,10 @@ +{{- if .Values.certManager.certSpec -}} +apiVersion: cert-manager.io/v1 +kind: Certificate +metadata: + name: {{ include "cerbos.fullname" . }} + labels: + {{- include "cerbos.labels" . | nindent 4 }} +spec: + {{- toYaml .Values.certManager.certSpec | nindent 2 }} +{{- end -}} diff --git a/charts/cerbos/cerbos/0.37.0/templates/configmap.yaml b/charts/cerbos/cerbos/0.37.0/templates/configmap.yaml new file mode 100644 index 000000000..e5c3c36ad --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/configmap.yaml @@ -0,0 +1,9 @@ +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ include "cerbos.fullname" . }} + labels: + {{- include "cerbos.labels" . | nindent 4 }} +data: + ".cerbos.yaml": |- + {{- include "cerbos.config" . | nindent 8 }} diff --git a/charts/cerbos/cerbos/0.37.0/templates/deployment.yaml b/charts/cerbos/cerbos/0.37.0/templates/deployment.yaml new file mode 100644 index 000000000..bbbd89cd8 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/deployment.yaml @@ -0,0 +1,149 @@ +{{- $tlsDisabled := (eq (include "cerbos.tlsSecretName" .) "None") -}} +{{- $defaultHubDriverEnabled := (eq (include "cerbos.defaultHubDriverEnabled" .) "yes") -}} +apiVersion: apps/v1 +{{- if eq .Values.type "deployment" }} +kind: Deployment +{{- else if eq .Values.type "daemonset" }} +kind: DaemonSet +{{- else }} +{{- fail "valid values for .Values.type are deployment or daemonset" }} +{{- end}} +metadata: + name: {{ include "cerbos.fullname" . }} + labels: + {{- include "cerbos.labels" . | nindent 4 }} + {{- with .Values.deployment.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + {{- if not .Values.autoscaling.enabled }} + {{- if eq .Values.type "deployment" }} + replicas: {{ .Values.replicaCount }} + {{- end }} + {{- end }} + selector: + matchLabels: + {{- include "cerbos.selectorLabels" . | nindent 6 }} + template: + metadata: + labels: + {{- include "cerbos.selectorLabels" . | nindent 8 }} + {{- with .Values.commonLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.podLabels }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- include "cerbos.podAnnotations" . | nindent 6 }} + spec: + {{- with .Values.imagePullSecrets }} + imagePullSecrets: + {{- toYaml . | nindent 8 }} + {{- end }} + serviceAccountName: {{ include "cerbos.serviceAccountName" . }} + {{- with .Values.podSecurityContext }} + securityContext: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.initContainers }} + initContainers: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- if .Values.priorityClassName }} + priorityClassName: {{- .Values.priorityClassName }} + {{- end }} + containers: + - name: {{ .Chart.Name }} + {{- with .Values.securityContext }} + securityContext: + {{- toYaml . | nindent 12 }} + {{- end }} + image: {{ include "cerbos.image" . }} + imagePullPolicy: {{ .Values.image.pullPolicy }} + args: + - "server" + - "--config=/config/.cerbos.yaml" + - "--log-level={{ .Values.cerbos.logLevel }}" + ports: + - name: http + containerPort: {{ .Values.cerbos.httpPort }} + - name: grpc + containerPort: {{ .Values.cerbos.grpcPort }} + livenessProbe: + httpGet: + path: /_cerbos/health + port: http + scheme: {{ include "cerbos.httpScheme" . | upper }} + readinessProbe: + httpGet: + path: /_cerbos/health + port: http + scheme: {{ include "cerbos.httpScheme" . | upper }} + {{- with .Values.resources }} + resources: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.env }} + env: + {{- toYaml . | nindent 12 }} + {{- end }} + {{- with .Values.envFrom }} + envFrom: + {{- toYaml . | nindent 12 }} + {{- end }} + volumeMounts: + - name: config + mountPath: /config + readOnly: true + {{- if $defaultHubDriverEnabled }} + - name: bundletmp + mountPath: /tmp + - name: bundlecache + mountPath: /.cache + {{- else }} + - name: work + mountPath: /work + {{- end }} + {{- if not $tlsDisabled }} + - name: certs + mountPath: /certs + readOnly: true + {{- end }} + {{- with .Values.volumeMounts }} + {{- toYaml . | nindent 12 }} + {{- end }} + volumes: + - name: config + configMap: + name: {{ include "cerbos.fullname" . }} + {{- if $defaultHubDriverEnabled }} + - name: bundletmp + emptyDir: {} + - name: bundlecache + emptyDir: {} + {{- else }} + - name: work + emptyDir: {} + {{- end }} + {{- if not $tlsDisabled }} + - name: certs + secret: + secretName: {{ include "cerbos.tlsSecretName" . }} + {{- end }} + {{- with .Values.volumes }} + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.nodeSelector }} + nodeSelector: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.affinity }} + affinity: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- with .Values.tolerations }} + tolerations: + {{- toYaml . | nindent 8 }} + {{- end }} + {{- include "cerbos.topologySpreadConstraints" . | nindent 6}} diff --git a/charts/cerbos/cerbos/0.37.0/templates/hpa.yaml b/charts/cerbos/cerbos/0.37.0/templates/hpa.yaml new file mode 100644 index 000000000..adab0ade9 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/hpa.yaml @@ -0,0 +1,32 @@ +{{- if and .Values.autoscaling.enabled (eq .Values.type "deployment") }} +apiVersion: autoscaling/v2 +kind: HorizontalPodAutoscaler +metadata: + name: {{ include "cerbos.fullname" . }} + labels: + {{- include "cerbos.labels" . | nindent 4 }} +spec: + scaleTargetRef: + apiVersion: apps/v1 + kind: Deployment + name: {{ include "cerbos.fullname" . }} + minReplicas: {{ .Values.autoscaling.minReplicas }} + maxReplicas: {{ .Values.autoscaling.maxReplicas }} + metrics: + {{- if .Values.autoscaling.targetCPUUtilizationPercentage }} + - type: Resource + resource: + name: cpu + target: + type: Utilization + averageUtilization: {{ .Values.autoscaling.targetCPUUtilizationPercentage }} + {{- end }} + {{- if .Values.autoscaling.targetMemoryUtilizationPercentage }} + - type: Resource + resource: + name: memory + target: + type: Utilization + averageUtilization: {{ .Values.autoscaling.targetMemoryUtilizationPercentage }} + {{- end }} +{{- end }} diff --git a/charts/cerbos/cerbos/0.37.0/templates/service.yaml b/charts/cerbos/cerbos/0.37.0/templates/service.yaml new file mode 100644 index 000000000..51c37574d --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/service.yaml @@ -0,0 +1,40 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ include "cerbos.fullname" . }} + labels: + {{- include "cerbos.labels" . | nindent 4 }} + {{- with .Values.service.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +spec: + type: {{ .Values.service.type }} + {{- with .Values.service.clusterIP }} + clusterIP: {{ . }} + {{- end }} + {{- with .Values.service.loadBalancerIP }} + loadBalancerIP: {{ . }} + {{- end }} + {{- if .Values.service.internalTrafficPolicy }} + internalTrafficPolicy: {{ .Values.service.internalTrafficPolicy }} + {{- else if eq .Values.type "daemonset" }} + internalTrafficPolicy: Local + {{- end }} + ports: + - port: {{ .Values.service.httpPort }} + targetPort: http + protocol: TCP + name: http + {{ if eq .Values.service.type "NodePort" -}} + nodePort: {{ .Values.service.httpNodePort }} + {{- end }} + - port: {{ .Values.service.grpcPort }} + targetPort: grpc + protocol: TCP + name: grpc + {{ if eq .Values.service.type "NodePort" -}} + nodePort: {{ .Values.service.grpcNodePort }} + {{- end }} + selector: + {{- include "cerbos.selectorLabels" . | nindent 4 }} diff --git a/charts/cerbos/cerbos/0.37.0/templates/serviceaccount.yaml b/charts/cerbos/cerbos/0.37.0/templates/serviceaccount.yaml new file mode 100644 index 000000000..7dc14605f --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/templates/serviceaccount.yaml @@ -0,0 +1,12 @@ +{{- if .Values.serviceAccount.create -}} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "cerbos.serviceAccountName" . }} + labels: + {{- include "cerbos.labels" . | nindent 4 }} + {{- with .Values.serviceAccount.annotations }} + annotations: + {{- toYaml . | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/cerbos/cerbos/0.37.0/values-audit-log.yaml b/charts/cerbos/cerbos/0.37.0/values-audit-log.yaml new file mode 100644 index 000000000..1c909f42a --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values-audit-log.yaml @@ -0,0 +1,40 @@ +# Illustrates how to deploy Cerbos with an SQLite3 backend and audit logs. + +cerbos: + config: + # Configure the SQLite3 storage driver + storage: + driver: "sqlite3" + sqlite3: + dsn: "file:/data/cerbos.sqlite?mode=rwc&_fk=true" + # Configure audit logging + audit: + enabled: true + accessLogsEnabled: true + decisionLogsEnabled: true + backend: local + local: + storagePath: /audit/cerbos + +# Create volumes to hold the SQLite3 database and the audit log. +# Note that this example uses emptyDir volumes that lose data when the pod or node is killed. +# Use persistent volumes in production to preserve the data between pod restarts. + +volumes: + - name: cerbos-policies + emptyDir: {} + - name: cerbos-auditlog + emptyDir: {} + +volumeMounts: + - name: cerbos-policies + mountPath: /data + - name: cerbos-auditlog + mountPath: /audit + +# Optional: Autoscale the Cerbos deployment using CPU and memory utilization. +autoscaling: + enabled: true + targetCPUUtilizationPercentage: 80 + targetMemoryUtilizationPercentage: 80 + diff --git a/charts/cerbos/cerbos/0.37.0/values-cert-manager.yaml b/charts/cerbos/cerbos/0.37.0/values-cert-manager.yaml new file mode 100644 index 000000000..9bbed0200 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values-cert-manager.yaml @@ -0,0 +1,40 @@ +# Illustrates how to deploy Cerbos with a TLS certificate issued by cert-manager. + +cerbos: + config: + # Configure the SQLite3 storage driver + storage: + driver: "sqlite3" + sqlite3: + dsn: "file:/data/cerbos.sqlite?mode=rwc&_fk=true" + +certManager: + certSpec: + secretName: cerbos-certs + secretTemplate: + labels: + app.kubernetes.io/component: certificate + app.kubernetes.io/part-of: cerbos + duration: 48h + renewBefore: 36h + subject: + organizations: + - cerbos + isCA: false + dnsNames: ["cerbos.security.svc.cluster.local"] + issuerRef: + name: ca-cluster-issuer + kind: ClusterIssuer + + +# Create volumes to hold the SQLite3 database. +# Note that this example uses emptyDir volumes that lose data when the pod or node is killed. +# Use persistent volumes in production to preserve the data between pod restarts. +volumes: + - name: cerbos-policies + emptyDir: {} + +volumeMounts: + - name: cerbos-policies + mountPath: /data + diff --git a/charts/cerbos/cerbos/0.37.0/values-git-storage.yaml b/charts/cerbos/cerbos/0.37.0/values-git-storage.yaml new file mode 100644 index 000000000..1885d9fcb --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values-git-storage.yaml @@ -0,0 +1,33 @@ +# Illustrates how to connect Cerbos to a policy repository hosted on GitHub. +# Prerequisites: +# - Create a personal access token (PAT) in GitHub with `repo` permissions (https://docs.github.com/en/github/authenticating-to-github/creating-a-personal-access-token). +# - Create a Kubernetes secret named `cerbos-github-token`: +# PAT=YOUR_GITHUB_PAT kubectl create secret generic cerbos-github-token --from-literal=GITHUB_TOKEN=$PAT + +cerbos: + config: + # Configure the git storage driver + storage: + driver: "git" + git: + protocol: https + # Replace with the URL of your GitHub repo. + url: https://github.com/cerbos/sample-policies.git + # Replace with the branch name of your repo. + branch: main + # Remove or leave empty if the policies are not stored in a subdirectory. + subDir: hr + # Path to checkout. By default, /work is a Kubernetes emptyDir volume that is only available for the lifetime of the pod. + # If you want the work directory to persist between pod restarts, specify the mount path of a persistent volume here. + checkoutDir: /work + # How often the remote repo should be checked for updates. + updatePollInterval: 60s + # Credentials used to login to the remote GitHub repo. We are using an environment variable mounted from the secret we created earlier. + https: + username: ${GITHUB_TOKEN} + password: "" + +# Create an environment variable from the GitHub PAT stored in a secret. +envFrom: + - secretRef: + name: cerbos-github-token diff --git a/charts/cerbos/cerbos/0.37.0/values-hub-storage.yaml b/charts/cerbos/cerbos/0.37.0/values-hub-storage.yaml new file mode 100644 index 000000000..8491d66f0 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values-hub-storage.yaml @@ -0,0 +1,22 @@ +# Illustrates how to connect to Cerbos Hub +# Prerequisites: +# - Sign-up to Cerbos Hub and follow the instructions to create client credentials. +# - Create a Kubernetes secret named `cerbos-hub-credentials`: +# kubectl create secret generic cerbos-hub-credentials \ +# --from-literal=CERBOS_HUB_CLIENT_ID= \ +# --from-literal=CERBOS_HUB_CLIENT_SECRET= \ +# --from-literal=CERBOS_HUB_WORKSPACE_SECRET= + +cerbos: + config: + # Configure the bundle storage driver + storage: + driver: "hub" + hub: + remote: + bundleLabel: "YOUR_LABEL" # Alternatively, add `CERBOS_HUB_BUNDLE=` to the secret you created above. + +# Create environment variables from the secret. +envFrom: + - secretRef: + name: cerbos-hub-credentials diff --git a/charts/cerbos/cerbos/0.37.0/values-otlp.yaml b/charts/cerbos/cerbos/0.37.0/values-otlp.yaml new file mode 100644 index 000000000..b52db6a44 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values-otlp.yaml @@ -0,0 +1,43 @@ +# Illustrates how to deploy Cerbos with traces exported via OTLP. + +cerbos: + config: + # Configure the SQLite3 storage driver + storage: + driver: "sqlite3" + sqlite3: + dsn: "file:/data/cerbos.sqlite?mode=rwc&_fk=true" + +# Environment variables to configure OTLP exporter. +env: + - name: OTEL_SERVICE_NAME + value: cerbos.myns.svc + - name: OTEL_TRACE_SAMPLER + value: parentbased_always_on + - name: OTEL_EXPORTER_OTLP_TRACES_ENDPOINT + value: https://otlp.monitoring.svc.cluster.local + +# Optional common labels for resources. +commonLabels: + app.kubernetes.io/part-of: my-awesome-app + +# Optional annotations for the service. +service: + annotations: + a8r.io/owner: my-awesome-team + a8r.io/uptime: dashboard.example.com/cerbos + +# Optional annotations for the Cerbos pod. +podAnnotations: + a8r.io/owner: my-awesome-team + +# Create volumes to hold the SQLite3 database. +# Note that this example uses emptyDir volumes that lose data when the pod or node is killed. +# Use persistent volumes in production to preserve the data between pod restarts. +volumes: + - name: cerbos-policies + emptyDir: {} + +volumeMounts: + - name: cerbos-policies + mountPath: /data diff --git a/charts/cerbos/cerbos/0.37.0/values-volume-storage.yaml b/charts/cerbos/cerbos/0.37.0/values-volume-storage.yaml new file mode 100644 index 000000000..9b1f87da5 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values-volume-storage.yaml @@ -0,0 +1,19 @@ +# Illustrates how to mount a volume containing policies to the Cerbos pod. + +volumes: + - name: cerbos-policies + hostPath: + path: /data/cerbos-policies + +volumeMounts: + - name: cerbos-policies + mountPath: /policies + readOnly: true + +cerbos: + config: + storage: + driver: "disk" + disk: + directory: /policies + watchForChanges: true diff --git a/charts/cerbos/cerbos/0.37.0/values.yaml b/charts/cerbos/cerbos/0.37.0/values.yaml new file mode 100644 index 000000000..2d6ab1711 --- /dev/null +++ b/charts/cerbos/cerbos/0.37.0/values.yaml @@ -0,0 +1,149 @@ +# Default values for cerbos. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. + +nameOverride: "" +fullnameOverride: "" + +# Number of Cerbos pods to run +replicaCount: 1 + +# Container image details +image: + repository: ghcr.io/cerbos/cerbos + pullPolicy: IfNotPresent + # Image digest to use. Takes precedence over tag if specified. + digest: "" + # Image tag to use. Defaults to the chart appVersion. + tag: "" + +imagePullSecrets: [] + +initContainers: [] + +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: "" + +# Annotations to add to the pod. +podAnnotations: {} + +# Labels to add to the pod. +podLabels: {} + +# Common labels to add to the resources. +commonLabels: {} + +# Annotations to add to the deployment. +deployment: + annotations: {} + +# Security context for the whole pod. +podSecurityContext: {} + # fsGroup: 2000 + +# Security context for the Cerbos container. +securityContext: {} + # capabilities: + # drop: + # - ALL + # readOnlyRootFilesystem: true + # runAsNonRoot: true + # runAsUser: 1000 + +# Resource limits for the pod. +resources: {} + # limits: + # cpu: 100m + # memory: 128Mi + # requests: + # cpu: 100m + # memory: 128Mi + +# Autoscaling configuration. +autoscaling: + enabled: false + minReplicas: 1 + maxReplicas: 100 + targetCPUUtilizationPercentage: 80 + # targetMemoryUtilizationPercentage: 80 + +# Node selector for the pod. +nodeSelector: {} + +# Pod tolerations. +tolerations: [] + +# Pod affinity rules. +affinity: {} + +# Topology Spread Constraints rules. +topologySpreadConstraints: [] + # - topologyKey: topology.kubernetes.io/zone + # maxSkew: 1 + # whenUnsatisfiable: ScheduleAnyway + +# Volumes to add to the pod. +volumes: [] + +# Volume mounts to add to the Cerbos container. +volumeMounts: [] + +# Environment variables to add to the pod. +env: [] + +# Source environment variables from config maps or secrets. +envFrom: [] + +# Spec of the cert-manager certificate to create for the Cerbos deployment. +# If certSpec is not empty, a cert-manager.io/v1/Certificate resource will be created with its spec populated with values from certSpec. +# The certSpec value must be a valid Certificate spec. This Helm chart does not provide any defaults or inject any values into it. +# If cerbos.tlsSecretName is defined, it takes precedence over the generated certificate. +certManager: + certSpec: {} + +# Kubernetes workload type to use. Valid values are `deployment` or `daemonset`. +type: deployment + +# PriorityClassName to set on deployed pods +priorityClassName: "" + +# Cerbos service settings. +service: + type: ClusterIP + httpPort: 3592 + grpcPort: 3593 + httpNodePort: 13592 + grpcNodePort: 13593 + annotations: {} + clusterIP: null + loadBalancerIP: null + # Set the internalTrafficPolicy. If this is unset, and .Values.type + # is set to daemonset, this will default to "Local" + internalTrafficPolicy: "" + +# Cerbos deployment settings. +cerbos: + # Port to expose the http service on. + httpPort: 3592 + # Port to expose the gRPC service on. + grpcPort: 3593 + # Secret containing the TLS certificate. + # Leave empty to disable TLS. + # The secret must contain the following keys: + # - tls.crt: Required. Certificate file contents. + # - tls.key: Required. Private key for the certificate. + # - ca.crt: Optional. CA certificate to add to the trust pool. + tlsSecretName: "" + # Cerbos log level. Valid values are DEBUG, INFO, WARN and ERROR + logLevel: INFO + # Add Prometheus service discovery annotations to the pod. + prometheusPodAnnotationsEnabled: true + # Cerbos config file contents. + # Some server settings like server.httpListenAddr, server.grpcListenAddr, server.tls will be overwritten by the chart based on values provided above. + config: {} diff --git a/charts/jenkins/jenkins/5.4.2/CHANGELOG.md b/charts/jenkins/jenkins/5.4.2/CHANGELOG.md new file mode 100644 index 000000000..82a06742f --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/CHANGELOG.md @@ -0,0 +1,3029 @@ +# Changelog + +This file documents all notable changes to the Jenkins Helm Chart. +The release numbering uses [semantic versioning](http://semver.org). + +Use the following links to reference issues, PRs, and commits prior to v2.6.0. + +* Issue: `https://github.com/helm/charts/issues/[issue#]` +* PR: `https://github.com/helm/charts/pull/[pr#]` +* Commit: `https://github.com/helm/charts/commit/[commit]/stable/jenkins` + +The changelog until v1.5.7 was auto-generated based on git commits. +Those entries include a reference to the git commit to be able to get more details. + +## 5.4.2 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.5` + +## 5.4.1 + +Update `jenkins/jenkins` to version `2.452.3` + +## 5.4.0 + +Introduce capability of additional mountPaths and logging file paths for config reload container + +## 5.3.6 + +Update `workflow-aggregator` to version `600.vb_57cdd26fdd7` + +## 5.3.5 + +Update `kubernetes` to version `4253.v7700d91739e5` + +## 5.3.4 + +Update `jenkins/jenkins` to version `2.452.3-jdk17` + +## 5.3.3 + +Update `jenkins/inbound-agent` to version `3256.v88a_f6e922152-1` + +## 5.3.2 + +Update `kubernetes` to version `4248.vfa_9517757b_b_a_` + +## 5.3.1 + +Fix Tiltfile deprecated value reference + +## 5.3.0 + +Add `controller.topologySpreadConstraints` + +## 5.2.2 + +Update `kubernetes` to version `4246.v5a_12b_1fe120e` + +## 5.2.1 + +Update `jenkins/jenkins` to version `2.452.2-jdk17` + +## 5.2.0 + +Add `agent.inheritYamlMergeStrategy` to allow configuring this setting on the default agent pod template. + +## 5.1.31 + +Update `kubernetes` to version `4245.vf5b_83f1fee6e` + +## 5.1.30 + +Add `controller.JCasC.configMapAnnotations` to allow setting annotations on the JCasC ConfigMaps. + +## 5.1.29 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.4` + +## 5.1.28 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.3` + +## 5.1.27 + +Update `kubernetes` to version `4244.v4fb_b_00994a_90` + +## 5.1.26 + +Update `kubernetes` to version `4238.v41b_3ef14a_5d8` + +## 5.1.25 + +Update `kubernetes` to version `4236.vc06f753c3234` + +## 5.1.24 + +Update `kubernetes` to version `4234.vdf3e78112369` + +## 5.1.23 + +Update `kubernetes` to version `4233.vb_67a_0e11a_039` + +## 5.1.22 + +Update `configuration-as-code` to version `1810.v9b_c30a_249a_4c` + +## 5.1.21 + +Update `kubernetes` to version `4231.vb_a_6b_8936497d` + +## 5.1.20 + +Update `kubernetes` to version `4230.vceef11cb_ca_37` + +## 5.1.19 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.2` + +## 5.1.18 + +Update `configuration-as-code` to version `1807.v0175eda_00a_20` + +## 5.1.17 + +Update `jenkins/inbound-agent` to version `3248.v65ecb_254c298-1` + +## 5.1.16 + +Update `configuration-as-code` to version `1805.v1455f39c04cf` + +## 5.1.15 + +Update `jenkins/jenkins` to version `2.452.1-jdk17` + +## 5.1.14 + +Update `kubernetes` to version `4219.v40ff98cfb_d6f` + +## 5.1.13 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.1` + +## 5.1.12 + +Update `git` to version `5.2.2` + +## 5.1.11 + +Update `kubernetes` to version `4214.vf10083a_42e70` + +## 5.1.10 + +Update `kubernetes` to version `4211.v08850dd0dfa_3` + +## 5.1.9 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.26.2` + +## 5.1.8 + +Update `kubernetes` to version `4209.vc646b_71e5269` + +## 5.1.7 + +Update `kubernetes` to version `4208.v4017b_a_27a_d67` + +## 5.1.6 + +Update `jenkins/jenkins` to version `2.440.3-jdk17` + +## 5.1.5 + +Fix Prometheus controller name. + +## 5.1.4 + +Update `docker.io/bats/bats` to version `1.11.0` + +## 5.1.3 + +Update `jenkins/jenkins` to version `2.440.2-jdk17` + +## 5.1.2 + +Update `kubernetes` to version `4203.v1dd44f5b_1cf9` + +## 5.1.1 + +Update `kubernetes` to version `4199.va_1647c280eb_2` + +## 5.1.0 + +Add `agent.restrictedPssSecurityContext` to automatically inject in the jnlp container a securityContext that is suitable for the use of the restricted Pod Security Standard + +## 5.0.20 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.26.1` + +## 5.0.19 + +Introduced helm-docs to automatically generate `values.yaml` documentation. + +## 5.0.18 + +Update `kubernetes` to version `4193.vded98e56cc25` + +## 5.0.17 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.26.0` + +## 5.0.16 + +Enable support for deleting plugin configuration files at startup. + +## 5.0.15 + +Fixed changelog entries for previous version bumps + + +## 5.0.14 + +Update `jenkins/jenkins` to version `2.440.1-jdk17` + +## 5.0.13 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `1.25.4` + +## 5.0.12 + +Fix controller.sidecars.additionalSidecarContainers renaming and add tests + +## 5.0.11 + +* Add controller.sidecars.configAutoReload.scheme to specify protocol scheme when connecting Jenkins configuration-as-code reload endpoint +* Add controller.sidecars.configAutoReload.skipTlsVerify to force the k8s-sidecar container to skip TLS verification when connecting to an HTTPS Jenkins configuration-as-code reload endpoint + +## 5.0.10 + +Update `jenkins/inbound-agent` to version `3206.vb_15dcf73f6a_9-3` + +## 5.0.9 + +Update `kubernetes` to version `4186.v1d804571d5d4` + +## 5.0.8 + +Update `configuration-as-code` to version `1775.v810dc950b_514` + +## 5.0.7 + +Update `docker.io/kiwigrid/k8s-sidecar` to version `docker.io/kiwigrid/k8s-sidecar` + +## 5.0.6 + +Removed `docker.io` prefix from inbound-agent image + +## 5.0.5 + +Prefixed artifacthub.io/images with `docker.io` + +## 5.0.4 + +Updated super-linter to v6. Updated README.md and CHANGELOG.md to fix linting issues. + +## 5.0.2 + +Update `git` to version `5.2.1` + +## 5.0.1 + +Update `docker.io/bats/bats` to version `v1.10.0` + +## 5.0.0 + + > [!CAUTION] + > Several fields have been renamed or removed. See [UPGRADING.md](./UPGRADING.md#to-500) + +The Helm Chart is now updated automatically via [Renovate](https://docs.renovatebot.com/) + +## 4.12.1 + +Update Jenkins image and appVersion to jenkins lts release version 2.426.3 + +## 4.12.0 + +Add support for [generic ephemeral storage](https://github.com/jenkinsci/kubernetes-plugin/pull/1489) in `agent.volumes` and `agents.workspaceVolume`. + +| plugin | old version | new version | +|------------|---------------------|--------------------| +| kubernetes | 4029.v5712230ccb_f8 | 4174.v4230d0ccd951 | + +## 4.11.2 + +Fixed documentation for controller.initScripts. + +## 4.11.1 + +Updated helm-unittest and made unittests compatible. + +## 4.11.0 + +Add multi-cloud support. + +## 4.10.0 + +Bumped Jenkins inbound agent from 3107.v665000b_51092-15 to 3192.v713e3b_039fb_e-5. + +## 4.9.2 + +Update Jenkins image and appVersion to jenkins lts release version 2.426.2 + + +Notes about [Artifact Hub](https://artifacthub.io/packages/helm/jenkinsci/jenkins?modal=changelog) changelog processing: +- Remove empty lines +- Keep only ASCII characters (no emojis) +- One change per line +- Remove table(s) (lines starting by "|") +- Backticks aren't rendered on artifacthub.io changelog + +## 4.9.1 + +Restore artifact hub notes location in CHANGELOG.md + +## 4.9.0 + +Update base images from JDK 11 to JDK 17. + +## 4.8.6 + +Proper `artifacthub.io/changes` changelog annotation preprocessing. + +## 4.8.5 + +Fix `artifacthub.io/changes` changelog annotation added to the released chart. + +## 4.8.4 + +Add `artifacthub.io/changes` changelog annotation to the released chart. + +## 4.8.3 + +Update Jenkins image and appVersion to jenkins lts release version 2.426.1 + +## 4.8.2 + +Add the ability to modify `retentionTimeout` and `waitForPodSec` default value in JCasC + +## 4.8.1 + +Reintroduces changes from 4.7.0 (reverted in 4.7.1), with additional fixes: + +- METHOD is now allowed in `env` and is not duplicated anymore +- No calls to JCasC reload endpoint from the init container + +## 4.8.0 + +Adds support for ephemeralStorage request and limit in Kubernetes plugin JCasC template + +## 4.7.4 + +Add the config-init-script checksum into the controller statefullset pod annotations to trigger restart of the pod in case of updated init scripts. + +## 4.7.3 + +Update Jenkins image and appVersion to jenkins lts release version 2.414.3 + +## 4.7.1 + +Changes in 4.7.0 were reverted. + +## 4.7.0 + +Runs `config-reload` as an init container, in addition to the sidecar container, to ensure that JCasC YAMLS are present before the main Jenkins container starts. This should fix some race conditions and crashes on startup. + +## 4.6.7 + +Change jenkins-test image label to match the other jenkins images + +## 4.6.5 + +Update Jenkins image and appVersion to jenkins lts release version 2.414.2 + +## 4.6.4 + +Introducing TPL function on variables related to hostname in `./charts/jenkins/templates/jenkins-controller-ingress.yaml` + +## 4.6.3 + +Add values to documentation + +## 4.6.2 + +Update word from hundreds to over 1800 to align with blurb at . + +## 4.6.1 + +Update `configuration-as-code` plugin to fix dependency issues with `azure-ad` plugin + +## 4.6.0 + +Added `.Values.controller.httpsKeyStore.jenkinsHttpsJksSecretKey` to allow overriding the default secret key containing the JKS file. +Added `.Values.controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretName` to allow getting the JKS password from a different secret. +Added `.Values.controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretKey` to allow overriding the default secret key containing the JKS password. + +## 4.5.1 + +Update Jenkins image and appVersion to jenkins lts release version 2.414.1 + + +## 4.5.0 + +Added `.Values.persistence.dataSource` to allow cloning home PVC from existing dataSource. + +## 4.4.2 + +Update Jenkins image and appVersion to jenkins lts release version 2.401.3 + + +## 4.4.1 + +Added `.Values.agent.jnlpregistry` to allow agents to be configured with private registry. + +## 4.4.0 + +Add config keys for liveness probes on agent containers. + + +## 4.3.30 + +Update Jenkins version in controller test matching LTS version + +## 4.3.29 + +Update Jenkins image and appVersion to jenkins lts release version 2.401.2 + + +## 4.3.28 + +Allow the kubernetes API server URL to be configurable. + +## 4.3.27 + +Bump kiwigrid/k8s-sidecar from 1.23.1 to 1.24.4 and jenkins/inbound-agent from 3107.v665000b_51092-5 to 3107.v665000b_51092-15. + +## 4.3.26 + +Fix various typos in the chart documentation. + +## 4.3.25 + +| plugin | old version | new version | +|-----------------------|----------------------|-----------------------| +| kubernetes | 3900.va_dce992317b_4 | 3937.vd7b_82db_e347b_ | +| configuration-as-code | 1625.v27444588cc3d | 1647.ve39ca_b_829b_42 | +| git | 5.0.0 | 5.1.0 | +| ldap | 671.v2a_9192a_7419d | 682.v7b_544c9d1512 | + +## 4.3.24 + +Update Jenkins image and appVersion to jenkins lts release version 2.401.1 + + +## 4.3.23 + +Update Jenkins image and appVersion to jenkins lts release version 2.387.3 + + +## 4.3.22 + + +Bump chart version. + +## 4.3.21 + + +Document building charts for weekly releases. + +## 4.3.20 + + +Enhance repository appearance and miscellaneous cleanup. + +## 4.3.19 + + +Comply with superlinter rules and address ShellCheck issues. + +## 4.3.18 + + +Bump kiwigrid/k8s-sidecar from 1.15.0 to 1.23.1. + +## 4.3.17 + + +Bump jenkins/inbound-agent from 4.11.2-4 to 3107.v665000b_51092-5. + +## 4.3.16 + + +Update bundled plugins: +- [ldap](https://plugins.jenkins.io/ldap/): From 2.5 to 671.v2a_9192a_7419d +- [kubernetes](https://plugins.jenkins.io/kubernetes/): From 3734.v562b_b_a_627ea_c to 3900.va_dce992317b_4 +- [workflow-aggregator](https://plugins.jenkins.io/workflow-aggregator/): From 590.v6a_d052e5a_a_b_5 to 590.v6a_d052e5a_a_b_5 +- [configuration-as-code](https://plugins.jenkins.io/configuration-as-code/): From 1569.vb_72405b_80249 to 1625.v27444588cc3d + +## 4.3.15 + + +Update bats from 1.2.1 to 1.9.0. + +## 4.3.14 + + +Update various GH actions, typo fixes, and miscellaneous chores. + +## 4.3.13 + + +Bump helm-unittest from 0.2.8 to 0.2.11. + +## 4.3.12 + + +Update wording in values.yml. + +## 4.3.11 + +Update Jenkins image and appVersion to jenkins lts release version 2.387.2 + + +## 4.3.10 + +Correct incorrect env var definition +Disable volume mount if disableSecretMount enabled + +## 4.3.9 + +Document `.Values.agent.directConnection` in README. +Add default value for `.Values.agent.directConnection` to `values.yaml` + +## 4.3.8 + +Added `.Values.agent.directConnection` to allow agents to be configured to connect direct to the JNLP port on the +controller, preventing the need for an external HTTP endpoint for this purpose. + +## 4.3.7 + +Added `.Values.controller.shareProcessNamespace` and `.Values.controller.httpsKeyStore.disableSecretMount` to enable sourcing TLS certs from external issuers + +## 4.3.6 + +Update Jenkins image and appVersion to jenkins lts release version 2.387.1 + +## 4.3.5 + +Added `.Values.helmtest.bats.image` and `.Values.helmtest.bats.image` to allow unit tests to be configurable. Fixes [https://github.com/jenkinsci/helm-charts/issues/683] + +## 4.3.4 + +Update Jenkins image and appVersion to jenkins lts release version 2.375.3 + + +## 4.3.3 + +Removed hardcoding of chart version in tests to make maintenance easier + +## 4.3.2 + +Added `.Values.serviceAccount.extraLabels` on Service Account +Added `.Values.serviceAccountAgent.extraLabels` on Agent's Service Account + + +## 4.3.0 + +Moved use of `.Values.containerEnv` within `jenkins` Container to top of `env` block to allow for subsequent Environment Variables to reference these additional ones. + +## 4.2.21 + +Update Jenkins image and appVersion to jenkins lts release version 2.375.2 + + +## 4.2.20 + +Fixed the `controller.prometheus.metricRelabelings` being unable to convert the value to the ServiceMonitor. +Added `controller.prometheus.relabelings` to allow relabling before scrape. +Added default values for `controller.prometheus.relabelings` and `controller.prometheus.metricRelabelings`. + +## 4.2.19 + +CronJob API version upgraded to batch/v1 + +## 4.2.18 + +Added option to set secretEnvVars. + +## 4.2.17 + +Update Jenkins image and appVersion to jenkins lts release version 2.375.1 + + +## 4.2.16 + +Fixed chart notes not rendering Jenkins URL with prefix when `controller.jenkinsUriPrefix` is set. +Fixed chart notes not rendering Jenkins URL with `https` when `controller.ingress.tls` or `controller.controller.httpsKeyStore.enable` is set. +Fixed chart notes rendering wrong JCasC URL when not using `controller.ingress`. + +## 4.2.15 + +Update Jenkins image and appVersion to jenkins lts release version 2.361.4 + +## 4.2.14 + +Added option to mount all keys from an existing k8s secret + +## 4.2.13 + +Adding `tpl` to `controller.additionalExistingSecrets` + +## 4.2.12 + +Update Jenkins image and appVersion to jenkins lts release version 2.361.3 + + +## 4.2.11 + +Update default plugin versions + +| plugin | old version | new version | +|-----------------------|-----------------------|------------------------| +| kubernetes | 3706.vdfb_d599579f3 | 3734.v562b_b_a_627ea_c | +| git | 4.11.5 | 4.13.0 | +| configuration-as-code | 1512.vb_79d418d5fc8 | 1569.vb_72405b_80249 | + +## 4.2.10 +Fix grammar and typos + +## 4.2.9 +Update Jenkins image and appVersion to jenkins lts release version 2.361.2 + +## 4.2.8 +Modify the condition to trigger copying jenkins_config files when configAutoReload option is disabled during Jenkins initialization + +## 4.2.7 +Support for remote URL for configuration + +## 4.2.6 +Add option to set hostnetwork for agents + +## 4.2.5 +Add an extra optional argument to extraPorts in order to specify targetPort + +## 4.2.4 +Remove k8s capibility requirements when setting priority class for controller + +## 4.2.3 Update plugin versions + +| plugin | old version | new version | +| --------------------- | --------------------- | --------------------- | +| kubernetes | 3600.v144b_cd192ca_a_ | 3706.vdfb_d599579f3 | +| workflow-aggregator | 581.v0c46fa_697ffd | 590.v6a_d052e5a_a_b_5 | +| configuration-as-code | 1429.v09b_044a_c93de | 1512.vb_79d418d5fc8 | +| git | 4.11.3 | 4.11.5 | + +Resolve version conflict between default install of plugins. + +## 4.2.2 + +Support Google Managed Prometheus + +## 4.2.1 + +Remove option to provide command and args of agent as YAML. This feature was never supported by the Jenkins Kubernetes +plugin. + +## 4.2.0 + +Add option to provide additional containers to agents + +## 4.1.18 + +Update Jenkins image and appVersion to jenkins lts release version 2.361.1 + + +## 4.1.17 + +Update Jenkins casc default settings to allow `security` configs to be provided + + +## 4.1.16 + +Update Jenkins image and appVersion to jenkins lts release version 2.346.3 + + +## 4.1.15 + +`projectNamingStrategy` is configurable in default config. + +## 4.1.14 + +If `installPlugins` is disabled, don't create unused plugins volume. + +## 4.1.13 + +Update Jenkins image and appVersion to jenkins lts release version 2.346.2 + + +## 4.1.12 + +If keystore is defined, it is now also made available in the initContainer. + +## 4.1.11 + +JCasC ConfigMaps now generate their name from the `jenkins.casc.configName` helper + +## 4.1.10 + +Update Jenkins image and appVersion to jenkins lts release version 2.346.1 + + +## 4.1.9 + +Allow setting `imagePullSecret` for backup job via `backup.imagePullSecretName` + +## 4.1.8 + +Fix path of projected secrets from `additionalExistingSecrets`. + +## 4.1.7 + +Update README with explanation on the required environmental variable `AWS_REGION` in case of using an S3 bucket. + +## 4.1.6 + +project adminSecret, additionalSecrets and additionalExistingSecrets instead of mount with subPath + +## 4.1.5 + +Update README to fix `JAVA_OPTS` name. + +## 4.1.4 +Update plugins + +## 4.1.3 +Update jenkins-controller-statefulset projected volumes definition + +## 4.1.1 +Added 'controller.prometheus.metricRelabelings' to allow relabling and dropping unused prometheus metrics + +## 4.1.0 + +Added `controller.sidecars.configAutoReload.envFrom`, `controller.initContainerEnvFrom`, `controller.containerEnvFrom` + +## 4.0.1 + +No code changes - CI updated to run unit tests using Helm 3.8.2. + +## 4.0.0 + +Removes automatic `remotingSecurity` setting when using a container tag older than `2.326` (introduced in [`3.11.7`](#3117)). If you're using a version older than `2.326`, you should explicitly set `.controller.legacyRemotingSecurityEnabled` to `true`. + +## 3.12.2 + +Update Jenkins image and appVersion to jenkins lts release version 2.332.3 + +## 3.12.1 + +Make namespace configurable for agents and additional agents. + +## 3.12.0 + +Added a flag for disabling the default Jenkins Agent configuration. + +## 3.11.10 + +Update Jenkins image and appVersion to jenkins lts release version 2.332.2 + +## 3.11.9 Bump configuration-as-code plugin version + +| plugin | old version | new version | +| --------------------- | ----------- | ----------- | +| configuration-as-code | 1.51 | 1414.v878271fc496f | + +## 3.11.8 + +Make [externalTrafficPolicy](https://kubernetes.io/docs/concepts/services-networking/service/#traffic-policies) and `loadBalancerSourceRanges` fields customizable for Agent listener service via `controller.agentListenerExternalTrafficPolicy` and `controller.loadBalancerSourceRanges`. + +## 3.11.7 + +Removed Configuration as Code `remotingSecurity` section for Jenkins 2.326 or newer. See [Documentation](https://www.jenkins.io/redirect/AdminWhitelistRule) to learn more. + +## 3.11.6 + +Update Jenkins image and appVersion to jenkins lts release version 2.332.1 + + +## 3.11.5 + +Change Backup Role name function call to match the RoleDef function call in the Backup RoleBinding + +## 3.11.4 + +Update Jenkins image and appVersion to jenkins lts release version 2.319.3 + + +## 3.11.3 + +Update kiwigrid/k8s-sidecar:1.15.0 +Update jenkins/inbound-agent:4.11.2-4 + +## 3.11.2 + +Improve example for workspaceVolume. Clarify that this is not a list. + +## 3.11.1 + +Update configuration-as-code plugin to 1.55.1 + + +## 3.11.0 + +Update default plugin versions + +| plugin | old version | new version | +| --------------------- | ----------- | ----------- | +| kubernetes | 1.31.1 | 1.31.3 | +| git | 4.10.1 | 4.10.2 | + +## 3.10.3 + +Update Jenkins image and appVersion to jenkins lts release version 2.319.2 + + +## 3.10.2 + +Fix definition of startupProbe when deploying on a Kubernetes cluster < 1.16 + +## 3.10.1 + +correct VALUES_SUMMARY.md for installLatestPlugins + +## 3.10.0 + +Update default plugin versions + +| plugin | old version | new version | +| --------------------- | ----------- | ----------- | +| kubernetes | 1.30.11 | 1.31.1 | +| git | 4.10.0 | 4.10.1 | +| configuration-as-code | 1.54 | 1.55 | + +## 3.9.4 + +Add JAVA_OPTIONS to the README so proxy settings get picked by jenkins-plugin-cli + +## 3.9.3 + +Fix config reload request URL when httpsKeystore in use + +## 3.9.2 + +Update Jenkins image and appVersion to jenkins lts release version 2.319.1 +Update following plugins: + +* kubernetes:1.30.11 +* git:4.10.0 +* configuration-as-code:1.54 + +## 3.9.1 + +Adding `tpl` to `controller.overrideArgs` + +## 3.9.0 + +Added containerSecurityContext + +## 3.8.9 + +Fix mounting of HTTPS keystore secret when httpsKeyStore is enabled + +## 3.8.8 + +Update Jenkins image and appVersion to jenkins lts release version 2.303.3 + +## 3.8.7 + +Adding `tpl` to `initScripts` + +## 3.8.6 + +Add `controller.tagLabel` to specify the label for the image tag, for example `jdk11` or `alpine` + +## 3.8.5 + +Move jenkins web root outside of home dir + +## 3.8.4 + +Add `controller.initConfigMap` to pass pre-existing `init.groovy.d` ConfigMaps to the controller + +## 3.8.3 + +Update missed reference to jenkins/inbound-agent:4.11-1 + +## 3.8.2 + +Update jenkins/inbound-agent:4.11-1 + +## 3.8.1 + +Update jenkins/inbound-agent:4.10-3 + +## 3.8.0 + +Update kiwigrid/k8s-sidecar:1.14.2 + +## 3.7.1 + +Update git and casc plugins versions + +## 3.7.0 + +Added the option to create AWS SecurityGroupPolicy resources + +## 3.6.2 + +Fix httpsKeyStore mount when `controller.httpsKeyStore.enable` is `true` + +## 3.6.1 + +Update Jenkins image and appVersion to jenkins lts release version 2.303.2 + + +## 3.6.0 +Support custom agent pod labels + +## 3.5.20 +Disallow ingress on port 50000 when agent listener is disabled + +## 3.5.19 +Add support for specifying termination-log behaviour for Jenkins controller + +## 3.5.18 +Add support for creating a Pod Disruption Budget for Jenkins controller + +## 3.5.17 +Update workdingDir to `/home/jenkins/agent` + +## 3.5.16 +Update location of icon (wiki.jenkins.io is down) + +## 3.5.15 +Add support for adding labels to the Jenkins home Persistent Volume Claim (pvc) + +## 3.5.14 + +* Updated versions of default plugins +* Use verbose logging during plugin installation +* download the latest version of all plugin dependencies (Fixes #442) + +## 3.5.13 + +Update Jenkins image and appVersion to jenkins lts release version 2.303.1 + +## 3.5.12 + +Added extended documentation for Backup and Restore. + +## 3.5.11 + +Sanitized the Jenkins Label + +## 3.5.10 + +Fixed `controller.customJenkinsLabels` not getting templated into the controller `labelString:` field in JCasC + +## 3.5.9 + +Update Jenkins image and appVersion to jenkins lts release version 2.289.3 + + +## 3.5.8 + +Add parameter `backup.serviceAccount.create` to disable service account creation for backup service and `backup.serviceAccount.name` to allow change of the SA name. +`backup.annotations` was moved to `backup.serviceAccount.annotations` + +## 3.5.7 + +Enable setting `controller.serviceExternalTrafficPolicy` to set [the standard Service option](https://kubernetes.io/docs/tasks/access-application-cluster/create-external-load-balancer/#preserving-the-client-source-ip). `externalTrafficPolicy` denotes if this Service desires to route external traffic to node-local or cluster-wide endpoints. + +## 3.5.6 + +Add optional `controller.initContainerResources`, if set, it will change resources allocation for init controller, overwise the `controller.resources` will be used + +## 3.5.5 + +Allow to configure nodeUsageMode via `agent.nodeUsageMode` + +## 3.5.4 + +Update tests to work with unittest 0.2.6 + +## 3.5.3 + +Update Jenkins image and appVersion to jenkins lts release version 2.289.2 + +## 3.5.2 + +Enable setting `controller.installLatestSpecifiedPlugins` to set whether to download the latest dependencies of any plugin that is requested to have the latest version. + +## 3.5.1 +Fix activeDeadlineSeconds wrong type bug in jenkins-backup-cronjob template + +## 3.5.0 + +Allow `controller.podAnnotations` to be render as a template + +## 3.4.1 + +Allow showRawYaml for the default agent's pod template to be customized. + +## 3.4.0 + +configAutoReload container updated from `kiwigrid/k8s-sidecar:0.1.275` to `kiwigrid/k8s-sidecar:1.12.2` + +## 3.3.23 + +Make `controller.ingress.resourceRootUrl` compatible with API version networking.k8s.io/v1 on k8s >= 1.19.x + +## 3.3.22 + +Update Jenkins image and appVersion to jenkins lts release version 2.289.1 + +## 3.3.21 +`persistence.mounts` additionally mount to init container to allow custom CA certificate keystore + +## 3.3.18 +Added `controller.overrideArgs` so any cli argument can be passed to the WAR. + +## 3.3.17 +Correct docs on disabling plugin installation + +## 3.3.16 +Support generating `SecretClaim` resources in order to read secrets from HashiCorp Vault into Kubernetes using `kube-vault-controller`. + +## 3.3.15 +Prevent `controller.httpsKeyStore` from improperly being quoted, leading to an invalid location on disk + +## 3.3.14 +Correct docs on disabling plugin installation + +## 3.3.13 +Update plugins + +## 3.3.12 +Add `controller.additionalExistingSecrets` property + +## 3.3.11 +Add support for disabling the Agent listener service via `controller.agentListenerEnabled`. + +## 3.3.10 +Update Jenkins image and appVersion to jenkins lts release version 2.277.4 + +## 3.3.9 +* Change helper template so user defined `agent.jenkinsUrl` value will always be used, if set +* Simplify logic for `jenkinsUrl` and `jenkinsTunnel` generation: always use fully qualified address + +## 3.3.8 +Update Jenkins image and appVersion to jenkins lts release version 2.277.3 + +## 3.3.7 +fix controller-ingress line feed bug + +## 3.3.6 + +Update Git plugin version to v4.7.1 +Update ldap plugin version to v2.5 + +## 3.3.5 + +Use tpl function for environment vars. Fixes [https://github.com/jenkinsci/helm-charts/issues/324] + +## 3.3.4 + +Update Jenkins image and appVersion to jenkins lts release version 2.277.2 + + +## 3.3.3 + +Enable setting `controller.installLatestPlugins` to set whether to download the minimum required version of all dependencies. + +## 3.3.2 + +Add `controller.additionalSecrets` documentation + +## 3.3.1 + +Add `controller.additionalSecrets` property + +## 3.3.0 + +Change default Jenkins image to `jdk11` variant + +## 3.2.6 + +Add missing `controller.jenkinsUrlProtocol` property + +## 3.2.5 + +Add additional metadata `artifacthub.io/images` for artifacthub + +## 3.2.4 +Update Jenkins image and appVersion to jenkins lts release version 2.277.1 +Update Git plugin version to v4.6.0 +Update kubernetes plugin version to v1.29.2 + +## 3.2.3 + +Fix rendering `controller.ingress.path` + +## 3.2.2 + +Added description for `controller.jenkinsUrl` value + +## 3.2.1 + +Enable setting ImagePullSecrets to controller and agent service accounts. + +## 3.2.0 + +Calculate consistent unique agent IDs to be used in pod templates. Fixes [https://github.com/jenkinsci/helm-charts/issues/270] + +## 3.1.15 + +Fix documentation for the kubernetes probes + +## 3.1.14 + +Typo in documentation + +## 3.1.13 + +Update Jenkins image and appVersion to jenkins lts release version 2.263.4 + +## 3.1.12 + +Added GitHub action to automate the updating of LTS releases. + +## 3.1.11 + +Enable setting controller.updateStrategy to change the update strategy for StatefulSet + +## 3.1.10 + +Fixed issue for the AgentListener where it was not possible to attribute a NodePort + +## 3.1.9 + +Upgrade kubernetes plugin to 1.29.0 and CasC plugin to 1.47 + +## 3.1.8 + +Fix init scripts config map name + +## 3.1.7 + +Fix missing newline when `httpsKeyStore` is enabled + +## 3.1.6 + +Mount controller init scripts from ConfigMap + +## 3.1.5 + +Fix `namespaceOverride` not applied when loading JCasC + +## 3.1.4 + +Update Git plugin version to v4.5.2 + +## 3.1.3 + +Update Jenkins image and appVersion to jenkins lts release version 2.263.3 + +## 3.1.2 + +Enable setting maxRequestsPerHostStr to change the max concurrent connections to Kubernetes API + +## 3.1.1 + +Update Jenkins image and appVersion to jenkins lts release version 2.263.2 + +## 3.1.0 + +* Added `.Values.controller.podSecurityContextOverride` and `.Values.backup.podSecurityContextOverride`. +* Added simple default values tests for `jenkins-backup-cronjob.yaml`. + +## 3.0.14 + +Enable to only backup job folder instead of whole jenkins + +## 3.0.13 + +Improve Documentation around JCasc and Custom Image + +## 3.0.12 + +Added GitHub Action testing on Kind 1.16, 1.17, 1.18, 1.19 & 1.20 + +## 3.0.11 + +Fixes & unit tests for Ingress resources on Kubernetes 1.19 and above + +## 3.0.10 + +Ingress resources on Kubernetes 1.19 (or above) are created with the version `networking.k8s.io/v1` + +## 3.0.9 + +Added support for backing up to Azure Blob Storage. + +## 3.0.8 + +* Typo in documentation + +## 3.0.7 + +* Add support for setting default agent workspaceVolume + +## 3.0.6 + +Use 2.263.1 image + +## 3.0.5 + +* Update appVersion to reflect new jenkins lts release version 2.263.1 + +## 3.0.4 + +* Fix documentation for additional secret mounts + +## 3.0.3 + +* Update `README.md` with explanation on how to mount additional secrets + +## 3.0.2 + +* Fix `.Values.controller.tolerations` and `.Values.controller.nodeSelector` variable names in templates\jenkins-backup-cronjob.yaml + +## 3.0.1 + +* added 'runAsNonroot' to security context + +## 3.0.0 + +* Chart uses StatefulSet instead of Deployment +* XML configuration was removed in favor of JCasC +* chart migrated to helm 3.0.0 (apiVersion v2) +* offending terms have been removed +* values have been renamed and re-ordered to make it easier to use +* already deprecated items have been removed +* componentName for the controller is now `jenkins-controller` +* componentName for the agent is now `jenkins-agent` +* container names are now + * `init` for the init container which downloads Jenkins plugins + * `jenkins` for the Jenkins controller + * `config-reload` for the sidecar container which automatically reloads JCasC +* Updated UI tests to use official `bats/bats` image instead of `dduportal/bats` + +For migration instructions from previous versions and additional information check README.md. + +## 2.19.0 + +* Use lts version 2.249.3 +* Update kubernetes, workflow-aggregator, git and configuration-as-code plugins. +* Fail apply_config.sh script if an error occurs. + +## 2.18.2 + +Fix: `master.javaOpts` issue with quoted values + +## 2.18.1 + +Recommend installing plugins in custom image + +## 2.18.0 + +Removed /tmp volume. Making /tmp a volume causes permission issues with jmap/jstack on certain Kubernetes clusters + +## 2.17.1 + +Fix location of jenkins.war file. +It is located in `/usr/share/jenkins/jenkins.war` and can be fonfigured via `master.jenkinsWar`. + +## 2.17.0 + +Add support for plugin-installation-manager-tool + +## 2.16.0 + +Added Startup probe for Jenkins pod when Kubernetes cluster is 1.16 or newer + +## 2.15.5 + +scriptApproval is taken into account when enableXmlConfig is false. + +## 2.15.4 + +Add Tilt support for easier helm chart development. + +## 2.15.3 + +Fix error on missing `ingress.paths` value + +## 2.15.2 + +Added documentation for ingress and jenkins URL + +## 2.15.1 + +Fix priorityClassName entry in values.yaml file + +## 2.15.0 + +Added support for disabling the helm.sh/chart annotation + +## 2.14.0 + +Added support for annotations in podTemplates + +## 2.13.2 + +Add nodeSelector in the backup pod +Fix tolerations in the backup pod + +## 2.13.1 + +Update list of maintainers + +## 2.13.0 + +Added Support for websockets in the default Jcasc config +Added trailing slash to JENKINS_URL env var + +## 2.12.2 + +Added unit tests for most resources in the Helm chart. + +## 2.12.1 + +Helm chart README update + +## 2.12.0 + +Add option to configure securityContext capabilities + +## 2.11.0 + +Added configurable security context for jenkins backup CronJob and annotations to its serviceaccount. + +## 2.10.0 + +Make activeDeadlineSeconds for backup job configurable + +## 2.9.0 + +Make namespace of PrometheusRule configurable + +## 2.8.2 + +Bumped configuration-as-code plugin version from 1.41 to 1.43. +See [configuration-as-code plugin issue #1478](https://github.com/jenkinsci/configuration-as-code-plugin/issues/1478) + +## 2.8.1 + +Fix indentation of JAVA_OPTS + +## 2.8.0 + +Add support for helm unittest and include first tests + +## 2.7.2 + +Target port of container `jenkins-sc-config` taken the value from values.yaml. + +## 2.7.0 + +Add a secondary ingress template for those who want a second ingress with different labels or annotations or whatever else. + +Example: You want /github-webhook to be on a public ingress, while the main Jenkins intance to be on a private locked down ingress. + +## 2.6.5 + +Update configScripts example + +## 2.6.4 + +Add timja as a maintainer + +## 2.6.3 + +Update k8s-sidecar image to 0.1.193 + +## 2.6.2 + +Only mount empty dir secrets-dir if either `master.enableXmlConfig` or `master.secretsFilesSecret` is set +Fixes #19 + +## 2.6.1 Do not render empty JCasC templates + +## 2.6.0 First release in jenkinsci GitHub org + +Updated README for new location + +## 2.5.2 + +Fix as per JENKINS-47112 + +## 2.5.1 + +Support Jenkins Resource Root URL + +## 2.5.0 + +Add an option to specify that Jenkins master should be initialized only once, during first install. + +## 2.4.1 + +Reorder README parameters into sections to facilitate chart usage and maintenance + +## 2.4.0 Update default agent image + +`jenkins/jnlp-slave` is deprected and `jenkins/inbound-agent` should be used instead. +Also updated it to newest version (4.3-4). + +## 2.3.3 correct templating of master.slaveJenkinsUrl + +Fixes #22708 + +## 2.3.2 Fix wrong value for overwritePluginsFromImage + +Fixes #23003 +Fixes #22633 + +Also fixes indentation for #23114 + +## 2.3.1 + +Always mount {{ .Values.master.jenkinsRef }}/secrets/ directory. Previous it +was mounted only when `master.enableXmlConfig` was enabled. + +## 2.3.0 + +Add an option to specify pod based on labels that can connect to master if NetworkPolicy is enabled + +## 2.2.0 increase retry for config auto reload + +Configure `REQ_RETRY_CONNECT` to `10` to give Jenkins more time to start up. + + +Value can be configured via `master.sidecars.configAutoReload.reqRetryConnect` + +## 2.1.2 updated README + +## 2.1.1 update credentials-binding plugin to 1.23 + +## 2.1.0 + +Add support to set `runAsUser` and `runAsGroup` for `agent`. + +## 2.0.1 + +Only render authorizationStrategy and securityRealm when values are set. + +## 2.0.0 Configuration as Code now default + container does not run as root anymore + +The README contains more details for this update. +Please note that the updated values contain breaking changes. + +## 1.27.0 Update plugin versions & sidecar container + +| plugin | old version | new version | +| --------------------- | ----------- | ----------- | +| kubernetes | 1.25.3 | 1.25.7 | +| workflow-job | 2.38 | 2.39 | +| credentials-binding | 1.21 | 1.22 | +| configuration-as-code | 1.39 | 1.41 | + +configAutoReload container updated from `kiwigrid/k8s-sidecar:0.1.132` to `kiwigrid/k8s-sidecar:0.1.144` + +## 1.26.0 + +Add support to override `workingDir` for default pod template + +## 1.25.0 + +Add support for installing plugins in addition to the chart's default plugins via `master.additionalPlugins` + +## 1.24.0 + +Allow configuration of yamlMergeStrategy via `agent.yamlMergeStrategy` + +## 1.23.2 + +In the `jenkins.xml.podTemplate` helper function, allow templating of all string values under `agent.volumes` except `type` by rendering them with the `tpl` function + +## 1.23.1 + +Added auto detection for Ingress API version + +## 1.23.0 + +Allow to use an existing secret for the jenkins admin credentials + +## 1.22.0 + +Add support for UI security in the default JCasC via `master.JCasC.securityRealm` and `master.JCasC.authorizationStrategy` which deny anonymous access by default + +## 1.21.3 + +Render `agent.envVars` in kubernetes pod template JCasC + +## 1.21.2 + +Cleanup `agent.yamlTemplate` rendering in kubernetes pod template XML configuration + +## 1.21.1 + +Render `agent.nodeSelector` in the kubernetes pod template JCasC + +## 1.21.0 + +Add support for overriding Ingress paths via `master.ingress.paths` + +## 1.20.0 + +Add the following options for configuring the Kubernetes plugin. + +- master.slaveDefaultsProviderTemplate +- master.slaveJenkinsUrl +- master.slaveJenkinsTunnel +- master.slaveConnectTimeout +- master.slaveReadTimeout + +## 1.19.0 + +Add support for disabling remember me via `master.disableRememberMe` +Add support for using a different markup formatter via `master.markupFormatter` + +## 1.18.1 + +Add support for executor mode configuraton with `master.executorMode`. + +## 1.18.0 Make installation of configuration-as-code plugin explicit + +Instead of configuring the configuration-as-code plugin version via +`master.JCasC.pluginVersion` it is now installed via `master.installPlugins` + +## 1.17.2 + +Allow templating of `serviceAccount.annotations` and `serviceAccountAgent.annotations` by rendering them with the `tpl` function + +## 1.17.1 + +Add support for Persistent Volume Claim (PVC) in `agent.volumes` + +## 1.17.0 + +Render `agent.volumes` in kubernetes pod template JCasC + +## 1.16.2 + +Reverts 1.16.1 as it introduced an error #22047 + +## 1.16.1 + +Fixed a bug with master.runAsUser variable due to use wrong type for comparison. + +## 1.16.0 + +Add `master.overwritePluginsFromImage` to allow support for jenkins plugins installed in the master image to persist. + +## 1.15.0 Update plugin versions & sidecar container + +| plugin | old version | new version | +| --------------------- | ----------- | ----------- | +| kubernetes | 1.25.1 | 1.25.3 | +| workflow-job | 2.36 | 2.38 | +| git | 4.2.0 | 4.2.2 | +| configuration-as-code | 1.36 | 1.39 | + +configAutoReload container updated from `kiwigrid/k8s-sidecar:0.1.20` to `kiwigrid/k8s-sidecar:0.1.132` + +## 1.14.0 + +support auto-reload container environment variables configuration + +## 1.13.3 + +Fix wrong indent in tolerations + +## 1.13.2 + +Add support for custom ClusterIP + +## 1.13.1 + +Fix `agent.yamlTemplate` rendering in kubernetes pod template JCasC + +## 1.13.0 + +Add `master.networkPolicy.internalAgents` and `master.networkPolicy.externalAgents` stanzas to fine grained controls over where internal/external agents can connect from. Internal ones are allowed based on pod labels and (optionally) namespaces, and external ones are allowed based on IP ranges. + +## 1.12.0 Support additional agents + +Add support for easy configuration of additional agents which inherit values from `agent`. + +## 1.11.3 + +Update the kubernetes plugin from 1.24.1 to 1.25.1 and grant 'watch' permission to 'events' which is required since this plugin version. + +## 1.11.2 Configure agent.args in values.yaml + +## 1.11.1 Support for master.additionalConfig + +Fixed a bug with jenkinsHome variable in range block when master.additionalConfig is set - Helm cannot evaluate field Values in type interface {}. + +## 1.11.0 Add support for configuring custom pod templates + +Add `agent.podTemplates` option for declaring custom pod templates in the default configured kubernetes cloud. + +## 1.10.1 Only copy JCasC files if there are any + +The chart always tried to copy Configuration as Code configs even if there are none. That resulted in an error which is resolved with this. + +## 1.10.0 Remove configuration-as-code-support plugins + +In recent version of configuration-as-code-plugin this is no longer necessary. + +## 1.9.24 + +Update JCasC auto-reload docs and remove stale ssh key references from version "1.8.0 JCasC auto reload works without ssh keys" + +## 1.9.23 Support jenkinsUriPrefix when JCasC is enabled + +Fixed a bug in the configuration as code reload URL, where it wouldn't work with a jenkinsUriPrefix set. + +## 1.9.22 + +Add `master.jenkinsHome` and `master.jenkinsRef` options to use docker images derivates from Jenkins + +## 1.9.21 + +Add `master.terminationGracePeriodSeconds` option + +## 1.9.20 + +Update default plugins + +- kubernetes:1.24.1 +- workflow-job:2.36 +- workflow-aggregator:2.6 +- credentials-binding:1.21 +- git:4.2.0 +- configuration-as-code:1.36 + +## 1.9.19 + +Update docs for Helm 3 + +## 1.9.18 + +Make `jenkins-home` attachable to Azure Disks without pvc + +```yaml + volumes: + - name: jenkins-home + azureDisk: + kind: Managed + diskName: myAKSDisk + diskURI: /subscriptions//resourceGroups/MC_myAKSCluster_myAKSCluster_eastus/providers/Microsoft.Compute/disks/myAKSDisk +``` + +## 1.9.16 + +Fix PodLabel for NetworkPolicy to work if enabled + +## 1.9.14 + +Properly fix case sense in `Values.master.overwriteConfig` in `config.yaml` + +## 1.9.13 + +Fix case sense in `Values.master.overwriteConfig` in `config.yaml` + +## 1.9.12 + +Scriptapprovals are overwritten when overwriteConfig is enabled + +## 1.9.10 + +Added documentation for `persistence.storageClass`. + +## 1.9.9 +Make `master.deploymentAnnotation` configurable. + +## 1.9.8 + +Make `agent.slaveConnectTimeout` configurable: by increasing this value Jenkins will not cancel&ask k8s for a pod again, while it's on `ContainerCreating`. Useful when you have big images or autoscaling takes some time. + +## 1.9.7 Update plugin versions + +| plugin | old version | new version | +|-----------------------|-------------|-------------| +| kubernetes | 1.18.2 | 1.21.2 | +| workflow-job | 2.33 | 2.36 | +| credentials-binding | 1.19 | 1.20 | +| git | 3.11.0 | 4.0.0 | +| configuration-as-code | 1.27 | 1.32 | + +## 1.9.6 + +Enables jenkins to use keystore inorder to have native ssl support #17790 + +## 1.9.5 Enable remoting security + +`Manage Jenkins` -> `Configure Global Security` -> `Enable Agent → Master Access Control` is now enabled via configuration as code plugin + +## 1.9.4 Option to set existing secret with Google Application Default Credentials + +Google application credentials are kept in a file, which has to be mounted to a pod. You can set `gcpcredentials` in `existingSecret` as follows: + +```yaml + existingSecret: + jenkins-service-account: + gcpcredentials: application_default_credentials.json +``` + +Helm template then creates the necessary volume mounts and `GOOGLE_APPLICATION_CREDENTIALS` environmental variable. + +## 1.9.3 Fix `JAVA_OPTS` when config auto-reload is enabled + +## 1.9.2 Add support for kubernetes-credentials-provider-plugin + +[kubernetes-credentials-provider-plugin](https://jenkinsci.github.io/kubernetes-credentials-provider-plugin/) needs permissions to get/watch/list kubernetes secrets in the namespaces where Jenkins is running. + +The necessary role binding can be created using `rbac.readSecrets` when `rbac.create` is `true`. + +To quote from the plugin documentation: + +> Because granting these permissions for secrets is not something that should be done lightly it is highly advised for security reasons that you both create a unique service account to run Jenkins as, and run Jenkins in a unique namespace. + +Therefor this is disabled by default. + +## 1.9.1 Update kubernetes plugin URL + +## 1.9.0 Change default serviceType to ClusterIP + +## 1.8.2 + +Revert fix in `1.7.10` since direct connection is now disabled by default. + +## 1.8.1 + +Add `master.schedulerName` to allow setting a Kubernetes custom scheduler + +## 1.8.0 JCasC auto reload works without ssh keys + +We make use of the fact that the Jenkins Configuration as Code Plugin can be triggered via http `POST` to `JENKINS_URL/configuration-as-code/reload`and a pre-shared key. +The sidecar container responsible for reloading config changes is now `kiwigrid/k8s-sidecar:0.1.20` instead of it's fork `shadwell/k8s-sidecar`. + +References: + +- [Triggering Configuration Reload](https://github.com/jenkinsci/configuration-as-code-plugin/blob/master/docs/features/configurationReload.md) +- [kiwigrid/k8s-sidecar](https://hub.docker.com/r/kiwigrid/k8s-sidecar) + +`master.sidecars.configAutoReload.enabled` now works using `casc.reload.token` + +## 1.7.10 + +Disable direct connection in default configuration (when kubernetes plugin version >= 1.20.2). +Note: In case direct connection is going to be used `jenkins/jnlp-slave` needs to be version `3.35-5` or newer. + +## 1.7.9 + +Prevented Jenkins Setup Wizard on new installations + +## 1.7.8 + +Extend extraPorts to be opened on the Service object, not just the container. + +## 1.7.7 + +Add persistentvolumeclaim permission to the role to support new dynamic pvc workspaces. + +## 1.7.6 + +Updated `master.slaveKubernetesNamespace` to parse helm templates. +Defined an sensible empty value to the following variables, to silence invalid warnings: + +- master.extraPorts +- master.scriptApproval +- master.initScripts +- master.JCasC.configScripts +- master.sidecars.other +- agent.envVars +- agent.volumes + +## 1.7.5 + +Fixed an issue where the JCasC won't run if JCasC auto-reload is enabled [issue #17135](https://github.com/helm/charts/issues/17135) + +## 1.7.4 + +Comments out JCasC example of jenkins.systemMessage so that it can be used by end users. Previously, an attempt to set systemMessage causes Jenkins to startup, citing duplicate JCasC settings for systemMessage [issue #13333](https://github.com/helm/charts/issues/13333) + +## 1.7.2 + +Update kubernetes-plugin to version 1.18.2 which fixes frequently encountered [JENKINS-59000](https://issues.jenkins-ci.org/plugins/servlet/mobile#issue/JENKINS-59000) + +## 1.7.1 + +Update the default requirements for jenkins-agent to 512Mi which fixes frequently encountered [issue #3723](https://github.com/helm/charts/issues/3723) + +## 1.7.0 + +[Jenkins Configuration as Code Plugin](https://github.com/jenkinsci/configuration-as-code-plugin) default configuration can now be enabled via `master.JCasC.defaultConfig`. + +JCasC default configuration includes: + +- Jenkins URL +- Admin email `master.jenkinsAdminEmail` +- crumbIssuer +- disableRememberMe: false +- mode: NORMAL +- numExecutors: {{ .Values.master.numExecutors }} +- projectNamingStrategy: "standard" +- kubernetes plugin + - containerCapStr via `agent.containerCap` + - jenkinsTunnel + - jenkinsUrl + - maxRequestsPerHostStr: "32" + - name: "kubernetes" + - namespace + - serverUrl: `"https://kubernetes.default"` + - template + - containers + - alwaysPullImage: `agent.alwaysPullImage` + - args + - command + - envVars + - image: `agent.image:agent.imageTag` + - name: `.agent.sideContainerName` + - privileged: `.agent.privileged` + - resourceLimitCpu: `agent.resources.limits.cpu` + - resourceLimitMemory: `agent.resources.limits.memory` + - resourceRequestCpu: `agent.resources.requests.cpu` + - resourceRequestMemory: `agent.resources.requests.memory` + - ttyEnabled: `agent.TTYEnabled` + - workingDir: "/home/jenkins" + - idleMinutes: `agent.idleMinutes` + - instanceCap: 2147483647 + - imagePullSecrets: + - name: `.agent.imagePullSecretName` + - label + - name + - nodeUsageMode: "NORMAL" + - podRetention: `agent.podRetention` + - serviceAccount + - showRawYaml: true + - slaveConnectTimeoutStr: "100" + - yaml: `agent.yamlTemplate` + - yamlMergeStrategy: "override" +- security: + - apiToken: + - creationOfLegacyTokenEnabled: false + - tokenGenerationOnCreationEnabled: false + - usageStatisticsEnabled: true + +Example `values.yaml` which enables JCasC, it's default config and configAutoReload: + +```yaml +master: + JCasC: + enabled: true + defaultConfig: true + sidecars: + configAutoReload: + enabled: true +``` + +add master.JCasC.defaultConfig and configure location + +- JCasC configuration is stored in template `jenkins.casc.defaults` + so that it can be used in `config.yaml` and `jcasc-config.yaml` + depending on if configAutoReload is enabled or not + +- Jenkins Location (URL) is configured to provide a startin point + for the config + +## 1.6.1 + +Print error message when `master.sidecars.configAutoReload.enabled` is `true`, but the admin user can't be found to configure the SSH key. + +## 1.6.0 + +Add support for Google Cloud Storage for backup CronJob (migrating from nuvo/kube-tasks to maorfr/kube-tasks) + +## 1.5.9 + +Fixed a warning when sidecar resources are provided through a parent chart or override values + +## 1.5.8 + +Fixed an issue when master.enableXmlConfig is set to false: Always mount jenkins-secrets volume if secretsFilesSecret is set (#16512) + +## 1.5.7 + +added initial changelog (#16324) +commit: cee2ebf98 + +## 1.5.6 + +enable xml config misspelling (#16477) +commit: a125b99f9 + +## 1.5.5 + +Jenkins master label (#16469) +commit: 4802d14c9 + +## 1.5.4 + +add option enableXmlConfig (#16346) +commit: 387d97a4c + +## 1.5.3 + +extracted "jenkins.URL" into template (#16347) +commit: f2fdf5332 + +## 1.5.2 + +Fix backups when deployment has custom name (#16279) +commit: 16b89bfff + +## 1.5.1 + +Ability to set custom namespace for ServiceMonitor (#16145) +commit: 18ee6cf01 + +## 1.5.0 + +update Jenkins plugins to fix security issue (#16069) +commit: 603cf2d2b + +## 1.4.3 + +Use fixed container name (#16068) +commit: b3e4b4a49 + +## 1.4.2 + +Provide default job value (#15963) +commit: c462e2017 + +## 1.4.1 + +Add Jenkins backendconfig values (#15471) +commit: 7cc9b54c7 + +## 1.4.0 + +Change the value name for docker image tags - standartise to helm preferred value name - tag; this also allows auto-deployments using weaveworks flux (#15565) +commit: 5c3d920e7 + +## 1.3.6 + +jenkins deployment port should be target port (#15503) +commit: 83909ebe3 + +## 1.3.5 + +Add support for namespace specification (#15202) +commit: e773201a6 + +## 1.3.4 + +Adding sub-path option for scraping (#14833) +commit: e04021154 + +## 1.3.3 + +Add existingSecret to Jenkins backup AWS credentials (#13392) +commit: d9374f57d + +## 1.3.2 + +Fix JCasC version (#14992) +commit: 26a6d2b99 + +## 1.3.1 + +Update affinity for a backup cronjob (#14886) +commit: c21ed8331 + +## 1.3.0 + +only install casc support plugin when needed (#14862) +commit: a56fc0540 + +## 1.2.2 + +DNS Zone customization (#14775) +commit: da2910073 + +## 1.2.1 + +only render comment if configAutoReload is enabled (#14754) +commit: e07ead283 + +## 1.2.0 + +update plugins to latest version (#14744) +commit: 84336558e + +## 1.1.24 + +add example for EmptyDir volume (#14499) +commit: cafb60209 + +## 1.1.23 + +check if installPlugins is set before using it (#14168) +commit: 1218f0359 + +## 1.1.22 + +Support servicemonitor and alerting rules (#14124) +commit: e15a27f48 + +## 1.1.21 + +Fix: healthProbe timeouts mapping to initial delay (#13875) +commit: 825b32ece + +## 1.1.20 + +Properly handle overwrite config for additional configs (#13915) +commit: 18ce9b558 + +## 1.1.18 + +update maintainer (#13897) +commit: 223002b27 + +## 1.1.17 + +add apiVersion (#13795) +commit: cd1e5c35a + +## 1.1.16 + +allow changing of the target port to support TLS termination sidecar (#13576) +commit: a34d3bbcc + +## 1.1.15 + +fix wrong pod selector in jenkins-backup (#13542) +commit: b5df4fd7e + +## 1.1.14 + +allow templating of customInitContainers (#13536) +commit: d1e1421f4 + +## 1.1.13 + +fix #13467 (wrong deprecation message) (#13511) +commit: fbe28fa1c + +## 1.1.12 + +Correct customInitContainers Name example. (#13405) +commit: 6c6e40405 + +## 1.1.11 + +fix master.runAsUser, master.fsGroup examples (#13389) +commit: 2d7e5bf72 + +## 1.1.10 + +Ability to specify raw yaml template (#13319) +commit: 77aaa9a5f + +## 1.1.9 + +correct NOTES.txt - use master.ingress.hostname (#13318) +commit: b08ef6280 + +## 1.1.8 + +explain how to upgrade major versions (#13273) +commit: e7617a97e + +## 1.1.7 + +Add support for idleMinutes and serviceAccount (#13263) +commit: 4595ee033 + +## 1.1.6 + +Use same JENKINS_URL no matter if slaves use different namespace (#12564) +commit: 94c90339f + +## 1.1.5 + +fix deprecation checks (#13224) +commit: c7d2f8105 + +## 1.1.4 + +Fix issue introduced in #13136 (#13232) +commit: 0dbcded2e + +## 1.1.3 + +fix chart errors (#13197) +commit: 692a1e3da + +## 1.1.2 + +correct selector for jenkins pod (#13200) +commit: 4537e7fda + +## 1.1.1 + +Fix rendering of customInitContainers and lifecycle for Jenkins helm chart (#13189) +commit: e8f6b0ada + +## 1.1.0 + +Add support for openshift route in jenkins (#12973) +commit: 48c58a430 + +## 1.0.0 + +helm chart best practices (#13136) +commit: b02ae3f48 + +### Breaking changes + +- values have been renamed to follow helm chart best practices for naming conventions so + that all variables start with a lowercase letter and words are separated with camelcase + +- all resources are now using recommended standard labels + + +As a result of the label changes also the selectors of the deployment have been updated. +Those are immutable so trying an updated will cause an error like: + +```text +Error: Deployment.apps "jenkins" is invalid: spec.selector: Invalid value: v1.LabelSelector{MatchLabels:map[string]string{"app.kubernetes.io/component":"jenkins-master", "app.kubernetes.io/instance":"jenkins"}, MatchExpressions:[]v1.LabelSelectorRequirement(nil)}: field is immutable +``` + +In order to upgrade, delete the Jenkins Deployment before upgrading: + +```console +kubectl delete deploy jenkins +``` + +## 0.40.0 + +Allow to override jenkins location protocol (#12257) +commit: 18a830626 + +## 0.39.0 + +Add possibility to add custom init-container and lifecycle for master-container (#13062) +commit: 14d043593 + +## 0.38.0 + +Support `priorityClassName` on Master Deployment (#13069) +commit: e896c62bc + +## 0.37.3 + +Add support for service account annotations in jenkins (#12969) +commit: b22774e2f + +## 0.37.2 + +fix: add hostName to ingress in values.yaml (#12946) +commit: 041045e9b + +## 0.37.1 + +Update to match actual defaults in value.yaml (#12904) +commit: 73b6d37eb + +## 0.37.0 + +Support multiple Jenkins instances in same namespace (#12748) +commit: 32ff2f343 + +## 0.36.5 + +Fix wrong comment in values.yaml (#12761) +commit: 9db8ced23 + +## 0.36.4 + +Re-add value for Ingress API Version (#12753) +commit: ecb7791b5 + +## 0.36.3 + +allow templating of volumes (#12734) +commit: adbda2ca6 + +## 0.36.2 + +Fix self-introduced whitespace bug (#12528) +commit: eec1678eb + +## 0.36.1 + +Add flag to overwrite jobs definition from values.yaml (#12427) +commit: fd349b2fc + +## 0.36.0 + +Replace OwnSshKey with AdminSshKey (#12140) (#12466) +commit: 80a8c9eb6 + +## 0.35.2 + +add note for breaking changes (#12203) +commit: e779c5a54 + +## 0.35.1 + +Allow Jenkins to run with READONLYROOTFS psp (#12338) +commit: 7c419e191 + +## 0.35.0 + +Jenkins OverwriteConfig setting also overwrites init scripts (#9468) +commit: 501335b76 + +## 0.34.1 + +Fix typo on hostname variable (#12156) +commit: 3d337d8dd + +## 0.34.0 + +Allow ingress without host rule (#11960) +commit: ddc966d1e + +## 0.33.2 + +Improve documentation - clarify that rbac is needed for autoreload (#11739) +commit: 9d75a5c34 + +## 0.33.1 + +use object for rollingUpdate (#11909) +commit: cb9cf21e8 + +## 0.33.0 + +Add hostAliases (#11701) +commit: 0b89e1094 + +## 0.32.10 + +Fix slave jnlp port always being reset when container is restarted (#11685) +commit: d7d51797b + +## 0.32.9 + +add ingress Hostname an ApiVersion to docs (#11576) +commit: 4d3e77137 + +## 0.32.8 + +Support custom master pod labels in deployment (#9714) (#11511) +commit: 9de96faa0 + +## 0.32.7 + +Fix Markdown syntax in README (#11496) +commit: a32221a95 + +## 0.32.6 + +Added custom labels on jenkins ingress (#11466) +commit: c875d2b9b + +## 0.32.5 + +fix typo in default jenkins agent image fixes #11356 (#11463) +commit: 30adb9a91 + +## 0.32.4 + +fix incorrect Deployment when using sidecars (#11413) +commit: 362b4cef8 + +## 0.32.3 + +[]: #10131 (#11411) +commit: 49cb72055 + +## 0.32.2 + +Option to expose the slave listener port as host port (#11187) +commit: 2f85a9663 + +## 0.32.1 + +Updating Jenkins deployment fails appears rollingUpdate needs to be (#11166) +commit: 07fc9dbde + +## 0.32.0 + +Merge Sidecard configs (#11339) +commit: 3696090b9 + +## 0.31.0 + +Add option to overwrite plugins (#11231) +commit: 0e9aa00a5 + +## 0.30.0 + +Added slave Pod env vars (#8743) +commit: 1499f6608 + +## 0.29.3 + +revert indentation to previous working version (#11293) +commit: 61662f17a + +## 0.29.2 + +allow running sidecar containers for Jenkins master (#10950) +commit: 9084ce54a + +## 0.29.1 + +Indent lines related to EnableRawHtmlMarkupFormatter (#11252) +commit: 20b310c08 + +## 0.29.0 + +Jenkins Configuration as Code (#9057) +commit: c3e8c0b17 + +## 0.28.11 + +Allow to enable OWASP Markup Formatter Plugin (#10851) +commit: 9486e5ddf + +## 0.28.10 + +Fixes #1341 -- update Jenkins chart documentation (#10290) +commit: 411c81cd0 + +## 0.28.9 + +Quoted JavaOpts values (#10671) +commit: 926a843a8 + +## 0.28.8 + +Support custom labels in deployment (#9714) (#10533) +commit: 3e00b47fa + +## 0.28.7 + +separate test resources (#10597) +commit: 7b7ae2d11 + +## 0.28.6 + +allow customizing livenessProbe periodSeconds (#10534) +commit: 3c94d250d + +## 0.28.5 + +Add role kind option (#8498) +commit: e791ad124 + +## 0.28.4 + +workaround for busybox's cp (Closes: #10471) (#10497) +commit: 0d51a4187 + +## 0.28.3 + +fix parsing java options (#10140) +commit: 9448d0293 + +## 0.28.2 + +Fix job definitions in standard values.yaml (#10184) +commit: 6b6355ae7 + +## 0.28.1 + +add numExecutors as a variable in values file (#10236) +commit: d5ea2050f + +## 0.28.0 + +various (#10223) +commit: e17d2a65d + +## 0.27.0 + +add backup cronjob (#10095) +commit: 863ead8db + +## 0.26.2 + +add namespace flag for port-forwarding in jenkins notes (#10399) +commit: 846b589a9 + +## 0.26.1 + +- fixes #10267 when executed with helm template - otherwise produces an invalid template. (#10403) + commit: 266f9d839 + +## 0.26.0 + +Add subPath for jenkins-home mount (#9671) +commit: a9c76ac9b + +## 0.25.1 + +update readme to indicate the correct image that is used by default (#9915) +commit: 6aba9631c + +## 0.25.0 + +Add ability to manually set Jenkins URL (#7405) +commit: a0178fcb4 + +## 0.24.0 + +Make AuthorizationStrategy configurable (#9567) +commit: 06545b226 + +## 0.23.0 + +Update Jenkins public chart (#9296) +commit: 4e5f5918b + +## 0.22.0 + +allow to override jobs (#9004) +commit: dca9f9ab9 + +## 0.21.0 + +Simple implementation of the option to define the ingress path to the jenkins service (#8101) +commit: 013159609 + +## 0.20.2 + +Cosmetic change to remove necessity of changing "appVersion" for every new LTS release (#8866) +commit: f52af042a + +## 0.20.1 + +Added ExtraPorts to open in the master pod (#7759) +commit: 78858a2fb + +## 0.19.1 + +Fix component label in NOTES.txt ... (#8300) +commit: c5494dbfe + +## 0.19.0 + +Kubernetes 1.9 support as well as automatic apiVersion detection (#7988) +commit: 6853ad364 + +## 0.18.1 + +Respect SlaveListenerPort value in config.xml (#7220) +commit: 0a5ddac35 + +## 0.18.0 + +Allow replacement of Jenkins config with configMap. (#7450) +commit: c766da3de + +## 0.17.0 + +Add option to allow host networking (#7530) +commit: dc2eeff32 + +## 0.16.25 + +add custom jenkins labels to the build agent (#7167) +commit: 3ecde5dbf + +## 0.16.24 + +Move kubernetes and job plugins to latest versions (#7438) +commit: 019e39456 + +## 0.16.23 + +Add different Deployment Strategies based on persistence (#6132) +commit: e0a20b0b9 + +## 0.16.22 + +avoid lint errors when adding Values.Ingress.Annotations (#7425) +commit: 99eacc854 + +## 0.16.21 + +bump appVersion to reflect new jenkins lts release version 2.121.3 (#7217) +commit: 296df165d + +## 0.16.20 + +Configure kubernetes plugin for including namespace value (#7164) +commit: c0dc6cc48 + +## 0.16.19 + +make pod retention policy setting configurable (#6962) +commit: e614c1033 + +## 0.16.18 + +Update plugins version (#6988) +commit: bf8180018 + +## 0.16.17 + +Add Master.AdminPassword in README (#6987) +commit: 13e754ad7 + +## 0.16.16 + +Added jenkins location configuration (#6573) +commit: 79de7026c + +## 0.16.15 + +use generic env var, not oracle specific env var (#6116) +commit: 6084ab4a4 + +## 0.16.14 + +Allow to specify resource requests and limits on initContainers (#6723) +commit: 942a33b1a + +## 0.16.13 + +Added support for NodePort service type for jenkens agent svc (#6571) +commit: 89a213c2b + +## 0.16.12 + +Added ability to configure multiple LoadBalancerSourceRanges (#6243) +commit: 01604ddbc + +## 0.16.11 + +Removing ContainerPort configuration as at the moment it does not work when you change this setting (#6411) +commit: e1c0468bd + +## 0.16.9 + +Fix jobs parsing for configmap by adding toYaml to jobs.yaml template (#3747) +commit: b2542a123 + +## 0.16.8 + +add jenkinsuriprefix in healthprobes (#5737) +commit: 435d7a7b9 + +## 0.16.7 + +Added the ability to switch from ClusterRoleBinding to RoleBinding. (#6190) +commit: dde03ede0 + +## 0.16.6 + +Make jenkins master pod security context optional (#6122) +commit: 63653fd59 + +## 0.16.5 + +Rework resources requests and limits (#6077) (#6077) +commit: e738f99d0 + +## 0.16.4 + +Add jenkins master pod annotations (#6313) +commit: 5e7325721 + +## 0.16.3 + +Split Jenkins readiness and liveness probe periods (#5704) +commit: fc6100c38 + +## 0.16.1 + +fix typo in jenkins README (#5228) +commit: 3cd3f4b8b + +## 0.16.0 + +Inherit existing plugins from Jenkins image (#5409) +commit: fd93bff82 + +## 0.15.1 + +Allow NetworkPolicy.ApiVersion and Master.Ingress.ApiVersion to Differ (#5103) +commit: 78ee4ba15 + +## 0.15.0 + +Secure Defaults (#5026) +commit: 0fe90b520 + +## 0.14.6 + +Wait for up to 2 minutes before failing liveness check (#5161) +commit: 2cd3fc481 + +## 0.14.5 + +correct ImageTag setting (#4371) +commit: 8ea04174d + +## 0.14.4 + +Update jenkins/README.md (#4559) +commit: d4e6352dd + +## 0.14.3 + +Bump appVersion (#4177) +commit: 605d3d441 + +## 0.14.2 + +Master.InitContainerEnv: Init Container Env Vars (#3495) +commit: c64abe27d + +## 0.14.1 + +Allow more configuration of Jenkins agent service (#4028) +commit: fc82f39b2 + +## 0.14.0 + +Add affinity settings (#3839) +commit: 64e82fa6a + +## 0.13.5 + +bump test timeouts (#3886) +commit: cd05dd99c + +## 0.13.4 + +Add OWNERS to jenkins chart (#3881) +commit: 1c106b9c8 + +## 0.13.3 + +Add fullnameOverride support (#3705) +commit: ec8080839 + +## 0.13.2 + +Update README.md (#3638) +commit: f6d274c37 + +## 0.13.1 + +Lower initial healthcheck delay (#3463) +commit: 9b99db67c + +## 0.13.0 + +Provision credentials.xml, secrets files and jobs (#3316) +commit: d305c5961 + +## 0.12.1 + +fix the default value for nodeUsageMode. (#3299) +commit: b68d19516 + +## 0.12.0 + +Recreate pods when CustomConfigMap is true and there are changes to the ConfigMap (which is how the vanilla chart works) (#3181) +commit: 86d29f804 + +## 0.11.1 + +Optionally adds liveness and readiness probes to jenkins (#3245) +commit: 8b9aa73ee + +## 0.11.0 + +Feature/run jenkins as non root user (#2899) +commit: 8918f4175 + +## 0.10.3 + +template the version to keep them synced (#3084) +commit: 35e7fa49a + +## 0.10.2 + +Update Chart.yaml +commit: e3e617a0b + +## 0.10.1 + +Merge branch 'master' into jenkins-test-timeout +commit: 9a230a6b1 + +Double retry count for Jenkins test +commit: 129c8e824 + +Jenkins: Update README | Master.ServiceAnnotations (#2757) +commit: 6571810bc + +## 0.10.0 + +Update Jenkins images and plugins (#2496) +commit: 2e2622682 + +## 0.9.4 + +Updating to remove the `.lock` directory as well (#2747) +commit: 6e676808f + +## 0.9.3 + +Use variable for service port when testing (#2666) +commit: d044f99be + +## 0.9.2 + +Review jenkins networkpolicy docs (#2618) +commit: 49911e458 + +Add image pull secrets to jenkins templates (#1389) +commit: 4dfae21fd + +## 0.9.1 + +Added persistent volume claim annotations (#2619) +commit: ac9e5306e + +Fix failing CI lint (#2758) +commit: 26f709f0e + +## 0.9.0 + +namespace defined templates with chart name (#2140) +commit: 408ae0b3f + +## 0.8.9 + +added useSecurity and adminUser to params (#1903) +commit: 39d2a03cd + +Use storageClassName for jenkins. (#1997) +commit: 802f6449b + +## 0.8.8 + +Remove old plugin locks before installing plugins (#1746) +commit: 6cd7b8ff4 + +promote initContainrs to podspec (#1740) +commit: fecc804fc + +## 0.8.7 + +add optional LoadBalancerIP option. (#1568) +commit: d39f11408 + +## 0.8.6 + +Fix bad key in values.yaml (#1633) +commit: dc27e5af3 + +## 0.8.5 + +Update Jenkins to support node selectors for agents. (#1532) +commit: 4af5810ff + +## 0.8.4 + +Add support for supplying JENKINS_OPTS and/or uri prefix (#1405) +commit: 6a331901a + +## 0.8.3 + +Add serviceAccountName to deployment (#1477) +commit: 0dc349b44 + +## 0.8.2 + +Remove path from ingress specification to allow other paths (#1599) +commit: e727f6b32 + +Update git plugin to 3.4.0 for CVE-2017-1000084 (#1505) +commit: 03482f995 + +## 0.8.1 + +Use consistent whitespace in template placeholders (#1437) +commit: 912f50c71 + +add configurable service annotations #1234 (#1244) +commit: 286861ca8 + +## 0.8.0 + +Jenkins v0.8.0 (#1385) +commit: 0009a2393 + +## 0.7.4 + +Use imageTag as version in config map (#1333) +commit: e8bb6ebb4 + +## 0.7.3 + +Add NetworkPolicy to Jenkins (#1228) +commit: 572b36c6d + +## 0.7.2 + +- Workflow plugin pin (#1178) + commit: ac3a0c7bc + +## 0.7.1 + +copy over plugins.txt in case of update (#1222) +commit: 75b5b1174 + +## 0.7.0 + +add jmx option (#964) +commit: 6ae8d1945 + +## 0.6.4 + +update jenkins to latest LTS 2.46.3 (#1182) +commit: ad90b4c27 + +## 0.6.3 + +Update chart maints to gh u/n (#1107) +commit: f357b77ed + +## 0.6.2 + +Add Agent.Privileged option (#957) +commit: 2cf4aced2 + +## 0.6.1 + +Upgrade jenkins to 2.46.2 (#971) +commit: 41bd742b4 + +## 0.6.0 + +Smoke test for Jenkins Chart (#944) +commit: 110441054 + +## 0.5.1 + +removed extra space from hardcoded password (#925) +commit: 85a9b9123 + +## 0.5.0 + +move config to init-container allowing use of upstream containers (#921) +commit: 1803c3d33 + +## 0.4.1 + +add ability to toggle jnlp-agent podTemplate generation (#918) +commit: accd53203 + +## 0.4.0 + +Jenkins add script approval (#916) +commit: c1746656e + +## 0.3.1 + +Update Jenkins to Latest LTS fixes #731 (#733) +commit: e9a3aed8b + +## 0.3.0 + +Added option to add Jenkins init scripts (#617) +commit: b889623d0 + +## 0.2.0 + +Add existing PVC (#716) +commit: 05271f145 + +## 0.1.15 + +use Master.ServicePort in config.xml (#769) +commit: f351f4b16 + +## 0.1.14 + +Added option to disable security on master node (#403) +commit: 3a6113d18 + +## 0.1.13 + +Added: extra mount points support for jenkins master (#474) +commit: fab0f7eb1 + +## 0.1.12 + +fix storageclass config typo (#548) +commit: 6fc0ff242 + +## 0.1.10 + +Changed default value of Kubernetes Cloud name to match one in kubernetes plugin (#404) +commit: 68351304a + +Add support for overriding the Jenkins ConfigMap (#524) +commit: f97ca53b1 + +## 0.1.9 + +Added jenkins-master ingress support (#402) +commit: d76a09588 + +## 0.1.8 + +Change description (#553) +commit: 91f5c24e1 + +Removed default Persistence.StorageClass: generic (#530) +commit: c87494c10 + +Update to the recommended pvc patterns. (#448) +commit: a7fc595aa + +Remove helm.sh/created annotations (#505) +commit: f380da2fb + +## 0.1.7 + +add support for explicit NodePort on jenkins chart (#342) +commit: f63c188da + +Add configurable loadBalancerSourceRanges for jenkins chart (#360) +commit: 44007c50e + +Update Jenkins version to current LTS (2.19.4) and Kubernetes Plugin to 0.10 (#341) +commit: 6c8678167 + +## 0.1.6 + +Add imagePullPolicy to init container (#295) +commit: 103ee1952 + +## 0.1.5 + +bump chart version with PVC metadata label additions +commit: 4aa9cf5b1 + +## 0.1.4 + +removed `*` from `jenkins/templates/NOTES.txt` +commit: 76212230b + +apply standard metadata labels to PVC's +commit: 58b730836 + +specify namespace in `kubectl get svc` commands in NOTES.txt +commit: 7d3287e81 + +Update Jenkins version to current LTS (#194) +commit: 2c0404049 + +## 0.1.1 + +escape fixed +commit: 2026e1d15 + +.status.loadBalancer.ingress[0].ip is empty in AWS +commit: 1810e37f4 + +.status.loadBalancer.ingress[0].ip is empty in AWS +commit: 3cbd3ced6 + +Remove 'Getting Started:' from various NOTES.txt. (#181) +commit: 2f63fd524 + +docs(\*): update READMEs to reference chart repos (#119) +commit: c7d1bff05 + +## 0.1.0 + +Move first batch of PVC charts to stable +commit: d745f4879 diff --git a/charts/jenkins/jenkins/5.4.2/Chart.yaml b/charts/jenkins/jenkins/5.4.2/Chart.yaml new file mode 100644 index 000000000..aed21e1e1 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/Chart.yaml @@ -0,0 +1,54 @@ +annotations: + artifacthub.io/category: integration-delivery + artifacthub.io/changes: | + - Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.5` + artifacthub.io/images: | + - name: jenkins + image: docker.io/jenkins/jenkins:2.452.3-jdk17 + - name: k8s-sidecar + image: docker.io/kiwigrid/k8s-sidecar:1.27.5 + - name: inbound-agent + image: jenkins/inbound-agent:3256.v88a_f6e922152-1 + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/jenkinsci/helm-charts/tree/main/charts/jenkins + - name: Jenkins + url: https://www.jenkins.io/ + - name: support + url: https://github.com/jenkinsci/helm-charts/issues + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Jenkins + catalog.cattle.io/kube-version: '>=1.14-0' + catalog.cattle.io/release-name: jenkins +apiVersion: v2 +appVersion: 2.452.3 +description: 'Jenkins - Build great things at any scale! As the leading open source + automation server, Jenkins provides over 1800 plugins to support building, deploying + and automating any project. ' +home: https://www.jenkins.io/ +icon: file://assets/icons/jenkins.svg +keywords: +- jenkins +- ci +- devops +kubeVersion: '>=1.14-0' +maintainers: +- email: maor.friedman@redhat.com + name: maorfr +- email: mail@torstenwalter.de + name: torstenwalter +- email: garridomota@gmail.com + name: mogaal +- email: wmcdona89@gmail.com + name: wmcdona89 +- email: timjacomb1@gmail.com + name: timja +name: jenkins +sources: +- https://github.com/jenkinsci/jenkins +- https://github.com/jenkinsci/docker-inbound-agent +- https://github.com/maorfr/kube-tasks +- https://github.com/jenkinsci/configuration-as-code-plugin +type: application +version: 5.4.2 diff --git a/charts/jenkins/jenkins/5.4.2/README.md b/charts/jenkins/jenkins/5.4.2/README.md new file mode 100644 index 000000000..4ddd1faa4 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/README.md @@ -0,0 +1,706 @@ +# Jenkins + +[![Artifact Hub](https://img.shields.io/endpoint?url=https://artifacthub.io/badge/repository/jenkins)](https://artifacthub.io/packages/helm/jenkinsci/jenkins) +[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0) +[![Releases downloads](https://img.shields.io/github/downloads/jenkinsci/helm-charts/total.svg)](https://github.com/jenkinsci/helm-charts/releases) +[![Join the chat at https://app.gitter.im/#/room/#jenkins-ci:matrix.org](https://badges.gitter.im/badge.svg)](https://app.gitter.im/#/room/#jenkins-ci:matrix.org) + +[Jenkins](https://www.jenkins.io/) is the leading open source automation server, Jenkins provides over 1800 plugins to support building, deploying and automating any project. + +This chart installs a Jenkins server which spawns agents on [Kubernetes](http://kubernetes.io) utilizing the [Jenkins Kubernetes plugin](https://plugins.jenkins.io/kubernetes/). + +Inspired by the awesome work of [Carlos Sanchez](https://github.com/carlossg). + +## Get Repository Info + +```console +helm repo add jenkins https://charts.jenkins.io +helm repo update +``` + +_See [`helm repo`](https://helm.sh/docs/helm/helm_repo/) for command documentation._ + +## Install Chart + +```console +# Helm 3 +$ helm install [RELEASE_NAME] jenkins/jenkins [flags] +``` + +_See [configuration](#configuration) below._ + +_See [helm install](https://helm.sh/docs/helm/helm_install/) for command documentation._ + +## Uninstall Chart + +```console +# Helm 3 +$ 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._ + +## Upgrade Chart + +```console +# Helm 3 +$ helm upgrade [RELEASE_NAME] jenkins/jenkins [flags] +``` + +_See [helm upgrade](https://helm.sh/docs/helm/helm_upgrade/) for command documentation._ + +Visit the chart's [CHANGELOG](https://github.com/jenkinsci/helm-charts/blob/main/charts/jenkins/CHANGELOG.md) to view the chart's release history. +For migration between major version check [migration guide](#migration-guide). + +## Building weekly releases + +The default charts target Long-Term-Support (LTS) releases of Jenkins. +To use other versions the easiest way is to update the image tag to the version you want. +You can also rebuild the chart if you want the `appVersion` field to match. + +## 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](https://github.com/jenkinsci/helm-charts/blob/main/charts/jenkins/values.yaml), or run these configuration commands: + +```console +# Helm 3 +$ helm show values jenkins/jenkins +``` + +For a summary of all configurable options, see [VALUES_SUMMARY.md](https://github.com/jenkinsci/helm-charts/blob/main/charts/jenkins/VALUES_SUMMARY.md). + +### Configure Security Realm and Authorization Strategy + +This chart configured a `securityRealm` and `authorizationStrategy` as shown below: + +```yaml +controller: + JCasC: + securityRealm: |- + local: + allowsSignup: false + enableCaptcha: false + users: + - id: "${chart-admin-username}" + name: "Jenkins Admin" + password: "${chart-admin-password}" + authorizationStrategy: |- + loggedInUsersCanDoAnything: + allowAnonymousRead: false +``` + +With the configuration above there is only a single user. +This is fine for getting started quickly, but it needs to be adjusted for any serious environment. + +So you should adjust this to suite your needs. +That could be using LDAP / OIDC / .. as authorization strategy and use globalMatrix as authorization strategy to configure more fine-grained permissions. + +### Consider using a custom image + +This chart allows the user to specify plugins which should be installed. However, for production use cases one should consider to build a custom Jenkins image which has all required plugins pre-installed. +This way you can be sure which plugins Jenkins is using when starting up and you avoid trouble in case of connectivity issues to the Jenkins update site. + +The [docker repository](https://github.com/jenkinsci/docker) for the Jenkins image contains [documentation](https://github.com/jenkinsci/docker#preinstalling-plugins) how to do it. + +Here is an example how that can be done: + +```Dockerfile +FROM jenkins/jenkins:lts +RUN jenkins-plugin-cli --plugins kubernetes workflow-aggregator git configuration-as-code +``` + +NOTE: If you want a reproducible build then you should specify a non-floating tag for the image `jenkins/jenkins:2.249.3` and specify plugin versions. + +Once you built the image and pushed it to your registry you can specify it in your values file like this: + +```yaml +controller: + image: "registry/my-jenkins" + tag: "v1.2.3" + installPlugins: false +``` + +Notice: `installPlugins` is set to false to disable plugin download. In this case, the image `registry/my-jenkins:v1.2.3` must have the plugins specified as default value for [the `controller.installPlugins` directive](https://github.com/jenkinsci/helm-charts/blob/main/charts/jenkins/VALUES_SUMMARY.md#jenkins-plugins) to ensure that the configuration side-car system works as expected. + +In case you are using a private registry you can use 'imagePullSecretName' to specify the name of the secret to use when pulling the image: + +```yaml +controller: + image: "registry/my-jenkins" + tag: "v1.2.3" + imagePullSecretName: registry-secret + installPlugins: false +``` + +### External URL Configuration + +If you are using the ingress definitions provided by this chart via the `controller.ingress` block the configured hostname will be the ingress hostname starting with `https://` or `http://` depending on the `tls` configuration. +The Protocol can be overwritten by specifying `controller.jenkinsUrlProtocol`. + +If you are not using the provided ingress you can specify `controller.jenkinsUrl` to change the URL definition. + +### Configuration as Code + +Jenkins Configuration as Code (JCasC) is now a standard component in the Jenkins project. +To allow JCasC's configuration from the helm values, the plugin [`configuration-as-code`](https://plugins.jenkins.io/configuration-as-code/) must be installed in the Jenkins Controller's Docker image (which is the case by default as specified by the [default value of the directive `controller.installPlugins`](https://github.com/jenkinsci/helm-charts/blob/main/charts/jenkins/VALUES_SUMMARY.md#jenkins-plugins)). + +JCasc configuration is passed through Helm values under the key `controller.JCasC`. +The section ["Jenkins Configuration as Code (JCasC)" of the page "VALUES_SUMMARY.md"](https://github.com/jenkinsci/helm-charts/blob/main/charts/jenkins/VALUES_SUMMARY.md#jenkins-configuration-as-code-jcasc) lists all the possible directives. + +In particular, you may specify custom JCasC scripts by adding sub-key under the `controller.JCasC.configScripts` for each configuration area where each corresponds to a plugin or section of the UI. + +The sub-keys (prior to `|` character) are only labels used to give the section a meaningful name. +The only restriction is they must conform to RFC 1123 definition of a DNS label, so they may only contain lowercase letters, numbers, and hyphens. + +Each key will become the name of a configuration yaml file on the controller in `/var/jenkins_home/casc_configs` (by default) and will be processed by the Configuration as Code Plugin during Jenkins startup. + +The lines after each `|` become the content of the configuration yaml file. + +The first line after this is a JCasC root element, e.g. jenkins, credentials, etc. + +Best reference is the Documentation link here: `https:///configuration-as-code`. + +The example below sets custom systemMessage: + +```yaml +controller: + JCasC: + configScripts: + welcome-message: | + jenkins: + systemMessage: Welcome to our CI\CD server. +``` + +More complex example that creates ldap settings: + +```yaml +controller: + JCasC: + configScripts: + ldap-settings: | + jenkins: + securityRealm: + ldap: + configurations: + - server: ldap.acme.com + rootDN: dc=acme,dc=uk + managerPasswordSecret: ${LDAP_PASSWORD} + groupMembershipStrategy: + fromUserRecord: + attributeName: "memberOf" +``` + +Keep in mind that default configuration file already contains some values that you won't be able to override under configScripts section. + +For example, you can not configure Jenkins URL and System Admin email address like this because of conflicting configuration error. + +Incorrect: + +```yaml +controller: + JCasC: + configScripts: + jenkins-url: | + unclassified: + location: + url: https://example.com/jenkins + adminAddress: example@mail.com +``` + +Correct: + +```yaml +controller: + jenkinsUrl: https://example.com/jenkins + jenkinsAdminEmail: example@mail.com +``` + +Further JCasC examples can be found [here](https://github.com/jenkinsci/configuration-as-code-plugin/tree/master/demos). + +#### Breaking out large Config as Code scripts + +Jenkins Config as Code scripts can become quite large, and maintaining all of your scripts within one yaml file can be difficult. The Config as Code plugin itself suggests updating the `CASC_JENKINS_CONFIG` environment variable to be a comma separated list of paths for the plugin to traverse, picking up the yaml files as needed. +However, under the Jenkins helm chart, this `CASC_JENKINS_CONFIG` value is maintained through the templates. A better solution is to split your `controller.JCasC.configScripts` into separate values files, and provide each file during the helm install. + +For example, you can have a values file (e.g values_main.yaml) that defines the values described in the `VALUES_SUMMARY.md` for your Jenkins configuration: + +```yaml +jenkins: + controller: + jenkinsUrlProtocol: https + installPlugins: false + ... +``` + +In a second file (e.g values_jenkins_casc.yaml), you can define a section of your config scripts: + +```yaml +jenkins: + controller: + JCasC: + configScripts: + jenkinsCasc: | + jenkins: + disableRememberMe: false + mode: NORMAL + ... +``` + +And keep extending your config scripts by creating more files (so not all config scripts are located in one yaml file for better maintenance): + +values_jenkins_unclassified.yaml + +```yaml +jenkins: + controller: + JCasC: + configScripts: + unclassifiedCasc: | + unclassified: + ... +``` + +When installing, you provide all relevant yaml files (e.g `helm install -f values_main.yaml -f values_jenkins_casc.yaml -f values_jenkins_unclassified.yaml ...`). Instead of updating the `CASC_JENKINS_CONFIG` environment variable to include multiple paths, multiple CasC yaml files will be created in the same path `var/jenkins_home/casc_configs`. + +#### Config as Code With or Without Auto-Reload + +Config as Code changes (to `controller.JCasC.configScripts`) can either force a new pod to be created and only be applied at next startup, or can be auto-reloaded on-the-fly. +If you set `controller.sidecars.configAutoReload.enabled` to `true`, a second, auxiliary container will be installed into the Jenkins controller pod, known as a "sidecar". +This watches for changes to configScripts, copies the content onto the Jenkins file-system and issues a POST to `http:///reload-configuration-as-code` with a pre-shared key. +You can monitor this sidecar's logs using command `kubectl logs -c config-reload -f`. +If you want to enable auto-reload then you also need to configure rbac as the container which triggers the reload needs to watch the config maps: + +```yaml +controller: + sidecars: + configAutoReload: + enabled: true +rbac: + create: true +``` + +### Allow Limited HTML Markup in User-Submitted Text + +Some third-party systems (e.g. GitHub) use HTML-formatted data in their payload sent to a Jenkins webhook (e.g. URL of a pull-request being built). +To display such data as processed HTML instead of raw text set `controller.enableRawHtmlMarkupFormatter` to true. +This option requires installation of the [OWASP Markup Formatter Plugin (antisamy-markup-formatter)](https://plugins.jenkins.io/antisamy-markup-formatter/). +This plugin is **not** installed by default but may be added to `controller.additionalPlugins`. + +### Change max connections to Kubernetes API +When using agents with containers other than JNLP, The kubernetes plugin will communicate with those containers using the Kubernetes API. this changes the maximum concurrent connections +```yaml +agent: + maxRequestsPerHostStr: "32" +``` +This will change the configuration of the kubernetes "cloud" (as called by jenkins) that is created automatically as part of this helm chart. + +### Change container cleanup timeout API +For tasks that use very large images, this timeout can be increased to avoid early termination of the task while the Kubernetes pod is still deploying. +```yaml +agent: + retentionTimeout: "32" +``` +This will change the configuration of the kubernetes "cloud" (as called by jenkins) that is created automatically as part of this helm chart. + +### Change seconds to wait for pod to be running +This will change how long Jenkins will wait (seconds) for pod to be in running state. +```yaml +agent: + waitForPodSec: "32" +``` +This will change the configuration of the kubernetes "cloud" (as called by jenkins) that is created automatically as part of this helm chart. + +### Mounting Volumes into Agent Pods + +Your Jenkins Agents will run as pods, and it's possible to inject volumes where needed: + +```yaml +agent: + volumes: + - type: Secret + secretName: jenkins-mysecrets + mountPath: /var/run/secrets/jenkins-mysecrets +``` + +The supported volume types are: `ConfigMap`, `EmptyDir`, `HostPath`, `Nfs`, `PVC`, `Secret`. +Each type supports a different set of configurable attributes, defined by [the corresponding Java class](https://github.com/jenkinsci/kubernetes-plugin/tree/master/src/main/java/org/csanchez/jenkins/plugins/kubernetes/volumes). + +### NetworkPolicy + +To make use of the NetworkPolicy resources created by default, install [a networking plugin that implements the Kubernetes NetworkPolicy spec](https://kubernetes.io/docs/tasks/administer-cluster/declare-network-policy#before-you-begin). + +[Install](#install-chart) helm chart with network policy enabled by setting `networkPolicy.enabled` to `true`. + +You can use `controller.networkPolicy.internalAgents` and `controller.networkPolicy.externalAgents` stanzas for fine-grained controls over where internal/external agents can connect from. +Internal ones are allowed based on pod labels and (optionally) namespaces, and external ones are allowed based on IP ranges. + +### Script approval list + +`controller.scriptApproval` allows to pass function signatures that will be allowed in pipelines. +Example: + +```yaml +controller: + scriptApproval: + - "method java.util.Base64$Decoder decode java.lang.String" + - "new java.lang.String byte[]" + - "staticMethod java.util.Base64 getDecoder" +``` + +### Custom Labels + +`controller.serviceLabels` can be used to add custom labels in `jenkins-controller-svc.yaml`. +For example: + +```yaml +ServiceLabels: + expose: true +``` + +### Persistence + +The Jenkins image stores persistence under `/var/jenkins_home` path of the container. +A dynamically managed Persistent Volume Claim is used to keep the data across deployments, by default. +This is known to work in GCE, AWS, and minikube. Alternatively, a previously configured Persistent Volume Claim can be used. + +It is possible to mount several volumes using `persistence.volumes` and `persistence.mounts` parameters. +See additional `persistence` values using [configuration commands](#configuration). + +#### Existing PersistentVolumeClaim + +1. Create the PersistentVolume +2. Create the PersistentVolumeClaim +3. [Install](#install-chart) the chart, setting `persistence.existingClaim` to `PVC_NAME` + +#### Long Volume Attach/Mount Times + +Certain volume type and filesystem format combinations may experience long +attach/mount times, [10 or more minutes][K8S_VOLUME_TIMEOUT], when using +`fsGroup`. This issue may result in the following entries in the pod's event +history: + +```console +Warning FailedMount 38m kubelet, aks-default-41587790-2 Unable to attach or mount volumes: unmounted volumes=[jenkins-home], unattached volumes=[plugins plugin-dir jenkins-token-rmq2g sc-config-volume tmp jenkins-home jenkins-config secrets-dir]: timed out waiting for the condition +``` + +In these cases, experiment with replacing `fsGroup` with +`supplementalGroups` in the pod's `securityContext`. This can be achieved by +setting the `controller.podSecurityContextOverride` Helm chart value to +something like: + +```yaml +controller: + podSecurityContextOverride: + runAsNonRoot: true + runAsUser: 1000 + supplementalGroups: [1000] +``` + +This issue has been reported on [azureDisk with ext4][K8S_VOLUME_TIMEOUT] and +on [Alibaba cloud][K8S_VOLUME_TIMEOUT_ALIBABA]. + +[K8S_VOLUME_TIMEOUT]: https://github.com/kubernetes/kubernetes/issues/67014 +[K8S_VOLUME_TIMEOUT_ALIBABA]: https://github.com/kubernetes/kubernetes/issues/67014#issuecomment-698770511 + +#### Storage Class + +It is possible to define which storage class to use, by setting `persistence.storageClass` to `[customStorageClass]`. +If set to a dash (`-`), dynamic provisioning is disabled. +If the storage class is set to null or left undefined (`""`), the default provisioner is used (gp2 on AWS, standard on GKE, AWS & OpenStack). + +### Additional Secrets + +Additional secrets and Additional Existing Secrets, +can be mounted into the Jenkins controller through the chart or created using `controller.additionalSecrets` or `controller.additionalExistingSecrets`. +A common use case might be identity provider credentials if using an external LDAP or OIDC-based identity provider. +The secret may then be referenced in JCasC configuration (see [JCasC configuration](#configuration-as-code)). + +`values.yaml` controller section, referencing mounted secrets: +```yaml +controller: + # the 'name' and 'keyName' are concatenated with a '-' in between, so for example: + # an existing secret "secret-credentials" and a key inside it named "github-password" should be used in Jcasc as ${secret-credentials-github-password} + # 'name' and 'keyName' must be lowercase RFC 1123 label must consist of lower case alphanumeric characters or '-', + # and must start and end with an alphanumeric character (e.g. 'my-name', or '123-abc') + # existingSecret existing secret "secret-credentials" and a key inside it named "github-username" should be used in Jcasc as ${github-username} + # When using existingSecret no need to specify the keyName under additionalExistingSecrets. + existingSecret: secret-credentials + + additionalExistingSecrets: + - name: secret-credentials + keyName: github-username + - name: secret-credentials + keyName: github-password + - name: secret-credentials + keyName: token + + additionalSecrets: + - name: client_id + value: abc123 + - name: client_secret + value: xyz999 + JCasC: + securityRealm: | + oic: + clientId: ${client_id} + clientSecret: ${client_secret} + ... + configScripts: + jenkins-casc-configs: | + credentials: + system: + domainCredentials: + - credentials: + - string: + description: "github access token" + id: "github_app_token" + scope: GLOBAL + secret: ${secret-credentials-token} + - usernamePassword: + description: "github access username password" + id: "github_username_pass" + password: ${secret-credentials-github-password} + scope: GLOBAL + username: ${secret-credentials-github-username} +``` + +For more information, see [JCasC documentation](https://github.com/jenkinsci/configuration-as-code-plugin/blob/master/docs/features/secrets.adoc#kubernetes-secrets). + +### Secret Claims from HashiCorp Vault + +It's possible for this chart to generate `SecretClaim` resources in order to automatically create and maintain Kubernetes `Secrets` from HashiCorp [Vault](https://www.vaultproject.io/) via [`kube-vault-controller`](https://github.com/roboll/kube-vault-controller) + +These `Secrets` can then be referenced in the same manner as Additional Secrets above. + +This can be achieved by defining required Secret Claims within `controller.secretClaims`, as follows: +```yaml +controller: + secretClaims: + - name: jenkins-secret + path: secret/path + - name: jenkins-short-ttl + path: secret/short-ttl-path + renew: 60 +``` + +### RBAC + +RBAC is enabled by default. If you want to disable it you will need to set `rbac.create` to `false`. + +### Adding Custom Pod Templates + +It is possible to add custom pod templates for the default configured kubernetes cloud. +Add a key under `agent.podTemplates` for each pod template. Each key (prior to `|` character) is just a label, and can be any value. +Keys are only used to give the pod template a meaningful name. The only restriction is they may only contain RFC 1123 \ DNS label characters: lowercase letters, numbers, and hyphens. Each pod template can contain multiple containers. +There's no need to add the _jnlp_ container since the kubernetes plugin will automatically inject it into the pod. +For this pod templates configuration to be loaded the following values must be set: + +```yaml +controller.JCasC.defaultConfig: true +``` + +The example below creates a python pod template in the kubernetes cloud: + +```yaml +agent: + podTemplates: + python: | + - name: python + label: jenkins-python + serviceAccount: jenkins + containers: + - name: python + image: python:3 + command: "/bin/sh -c" + args: "cat" + ttyEnabled: true + privileged: true + resourceRequestCpu: "400m" + resourceRequestMemory: "512Mi" + resourceLimitCpu: "1" + resourceLimitMemory: "1024Mi" +``` + +Best reference is `https:///configuration-as-code/reference#Cloud-kubernetes`. + +### Adding Pod Templates Using additionalAgents + +`additionalAgents` may be used to configure additional kubernetes pod templates. +Each additional agent corresponds to `agent` in terms of the configurable values and inherits all values from `agent` so you only need to specify values which differ. +For example: + +```yaml +agent: + podName: default + customJenkinsLabels: default + # set resources for additional agents to inherit + resources: + limits: + cpu: "1" + memory: "2048Mi" + +additionalAgents: + maven: + podName: maven + customJenkinsLabels: maven + # An example of overriding the jnlp container + # sideContainerName: jnlp + image: jenkins/jnlp-agent-maven + tag: latest + python: + podName: python + customJenkinsLabels: python + sideContainerName: python + image: python + tag: "3" + command: "/bin/sh -c" + args: "cat" + TTYEnabled: true +``` + +### Ingress Configuration + +This chart provides ingress resources configurable via the `controller.ingress` block. + +The simplest configuration looks like the following: + +```yaml +controller: + ingress: + enabled: true + paths: [] + apiVersion: "extensions/v1beta1" + hostName: jenkins.example.com +``` + +This snippet configures an ingress rule for exposing jenkins at `jenkins.example.com` + +You can define labels and annotations via `controller.ingress.labels` and `controller.ingress.annotations` respectively. +Additionally, you can configure the ingress tls via `controller.ingress.tls`. +By default, this ingress rule exposes all paths. +If needed this can be overwritten by specifying the wanted paths in `controller.ingress.paths` + +If you want to configure a secondary ingress e.g. you don't want the jenkins instance exposed but still want to receive webhooks you can configure `controller.secondaryingress`. +The secondaryingress doesn't expose anything by default and has to be configured via `controller.secondaryingress.paths`: + +```yaml +controller: + ingress: + enabled: true + apiVersion: "extensions/v1beta1" + hostName: "jenkins.internal.example.com" + annotations: + kubernetes.io/ingress.class: "internal" + secondaryingress: + enabled: true + apiVersion: "extensions/v1beta1" + hostName: "jenkins-scm.example.com" + annotations: + kubernetes.io/ingress.class: "public" + paths: + - /github-webhook +``` + +## Prometheus Metrics + +If you want to expose Prometheus metrics you need to install the [Jenkins Prometheus Metrics Plugin](https://github.com/jenkinsci/prometheus-plugin). +It will expose an endpoint (default `/prometheus`) with metrics where a Prometheus Server can scrape. + +If you have implemented [Prometheus Operator](https://github.com/prometheus-operator/prometheus-operator), you can set `controller.prometheus.enabled` to `true` to configure a `ServiceMonitor` and `PrometheusRule`. +If you want to further adjust alerting rules you can do so by configuring `controller.prometheus.alertingrules` + +If you have implemented Prometheus without using the operator, you can leave `controller.prometheus.enabled` set to `false`. + +### Running Behind a Forward Proxy + +The controller pod uses an Init Container to install plugins etc. If you are behind a corporate proxy it may be useful to set `controller.initContainerEnv` to add environment variables such as `http_proxy`, so that these can be downloaded. + +Additionally, you may want to add env vars for the init container, the Jenkins container, and the JVM (`controller.javaOpts`): + +```yaml +controller: + initContainerEnv: + - name: http_proxy + value: "http://192.168.64.1:3128" + - name: https_proxy + value: "http://192.168.64.1:3128" + - name: no_proxy + value: "" + - name: JAVA_OPTS + value: "-Dhttps.proxyHost=proxy_host_name_without_protocol -Dhttps.proxyPort=3128" + containerEnv: + - name: http_proxy + value: "http://192.168.64.1:3128" + - name: https_proxy + value: "http://192.168.64.1:3128" + javaOpts: >- + -Dhttp.proxyHost=192.168.64.1 + -Dhttp.proxyPort=3128 + -Dhttps.proxyHost=192.168.64.1 + -Dhttps.proxyPort=3128 +``` + +### HTTPS Keystore Configuration + +[This configuration](https://wiki.jenkins.io/pages/viewpage.action?pageId=135468777) enables jenkins to use keystore in order to serve HTTPS. +Here is the [value file section](https://wiki.jenkins.io/pages/viewpage.action?pageId=135468777#RunningJenkinswithnativeSSL/HTTPS-ConfigureJenkinstouseHTTPSandtheJKSkeystore) related to keystore configuration. +Keystore itself should be placed in front of `jenkinsKeyStoreBase64Encoded` key and in base64 encoded format. To achieve that after having `keystore.jks` file simply do this: `cat keystore.jks | base64` and paste the output in front of `jenkinsKeyStoreBase64Encoded`. +After enabling `httpsKeyStore.enable` make sure that `httpPort` and `targetPort` are not the same, as `targetPort` will serve HTTPS. +Do not set `controller.httpsKeyStore.httpPort` to `-1` because it will cause readiness and liveliness prob to fail. +If you already have a kubernetes secret that has keystore and its password you can specify its' name in front of `jenkinsHttpsJksSecretName`, You need to remember that your secret should have proper data key names `jenkins-jks-file` (or override the key name using `jenkinsHttpsJksSecretKey`) +and `https-jks-password` (or override the key name using `jenkinsHttpsJksPasswordSecretKey`; additionally you can make it get the password from a different secret using `jenkinsHttpsJksPasswordSecretName`). Example: + +```yaml +controller: + httpsKeyStore: + enable: true + jenkinsHttpsJksSecretName: '' + httpPort: 8081 + path: "/var/jenkins_keystore" + fileName: "keystore.jks" + password: "changeit" + jenkinsKeyStoreBase64Encoded: '' +``` +### AWS Security Group Policies + +To create SecurityGroupPolicies set `awsSecurityGroupPolicies.enabled` to true and add your policies. Each policy requires a `name`, array of `securityGroupIds` and a `podSelector`. Example: + +```yaml +awsSecurityGroupPolicies: + enabled: true + policies: + - name: "jenkins-controller" + securityGroupIds: + - sg-123456789 + podSelector: + matchExpressions: + - key: app.kubernetes.io/component + operator: In + values: + - jenkins-controller +``` + +### Agent Direct Connection + +Set `directConnection` to `true` to allow agents to connect directly to a given TCP port without having to negotiate a HTTP(S) connection. This can allow you to have agent connections without an external HTTP(S) port. Example: + +```yaml +agent: + jenkinsTunnel: "jenkinsci-agent:50000" + directConnection: true +``` + +## Migration Guide + +### From stable repository + +Upgrade an existing release from `stable/jenkins` to `jenkins/jenkins` seamlessly by ensuring you have the latest [repository info](#get-repository-info) and running the [upgrade commands](#upgrade-chart) specifying the `jenkins/jenkins` chart. + +### Major Version Upgrades + +Chart release versions follow [SemVer](../../CONTRIBUTING.md#versioning), where a MAJOR version change (example `1.0.0` -> `2.0.0`) indicates an incompatible breaking change needing manual actions. + +See [UPGRADING.md](./UPGRADING.md) for a list of breaking changes diff --git a/charts/jenkins/jenkins/5.4.2/UPGRADING.md b/charts/jenkins/jenkins/5.4.2/UPGRADING.md new file mode 100644 index 000000000..41e424dbd --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/UPGRADING.md @@ -0,0 +1,148 @@ +# Upgrade Notes + +## To 5.0.0 +- `controller.image`, `controller.tag`, and `controller.tagLabel` have been removed. If you want to overwrite the image you now need to configure any or all of: + - `controller.image.registry` + - `controller.image.repository` + - `controller.image.tag` + - `controller.image.tagLabel` +- `controller.imagePullPolicy` has been removed. If you want to overwrite the pull policy you now need to configure `controller.image.pullPolicy`. +- `controller.sidecars.configAutoReload.image` has been removed. If you want to overwrite the configAutoReload image you now need to configure any or all of: + - `controller.sidecars.configAutoReload.image.registry` + - `controller.sidecars.configAutoReload.image.repository` + - `controller.sidecars.configAutoReload.image.tag` +- `controller.sidecars.other` has been renamed to `controller.sidecars.additionalSidecarContainers`. +- `agent.image` and `agent.tag` have been removed. If you want to overwrite the agent image you now need to configure any or all of: + - `agent.image.repository` + - `agent.image.tag` + - The registry can still be overwritten by `agent.jnlpregistry` +- `agent.additionalContainers[*].image` has been renamed to `agent.additionalContainers[*].image.repository` +- `agent.additionalContainers[*].tag` has been renamed to `agent.additionalContainers[*].image.tag` +- `additionalAgents.*.image` has been renamed to `additionalAgents.*.image.repository` +- `additionalAgents.*.tag` has been renamed to `additionalAgents.*.image.tag` +- `additionalClouds.*.additionalAgents.*.image` has been renamed to `additionalClouds.*.additionalAgents.*.image.repository` +- `additionalClouds.*.additionalAgents.*.tag` has been renamed to `additionalClouds.*.additionalAgents.*.image.tag` +- `helmtest.bats.image` has been split up to: + - `helmtest.bats.image.registry` + - `helmtest.bats.image.repository` + - `helmtest.bats.image.tag` +- `controller.adminUsername` and `controller.adminPassword` have been renamed to `controller.admin.username` and `controller.admin.password` respectively +- `controller.adminSecret` has been renamed to `controller.admin.createSecret` +- `backup.*` was unmaintained and has thus been removed. See the following page for alternatives: [Kubernetes Backup and Migrations](https://nubenetes.com/kubernetes-backup-migrations/). + +## To 4.0.0 +Removes automatic `remotingSecurity` setting when using a container tag older than `2.326` (introduced in [`3.11.7`](./CHANGELOG.md#3117)). If you're using a version older than `2.326`, you should explicitly set `.controller.legacyRemotingSecurityEnabled` to `true`. + +## To 3.0.0 + +* Check `securityRealm` and `authorizationStrategy` and adjust it. + Otherwise, your configured users and permissions will be overridden. +* You need to use helm version 3 as the `Chart.yaml` uses `apiVersion: v2`. +* All XML configuration options have been removed. + In case those are still in use you need to migrate to configuration as code. + Upgrade guide to 2.0.0 contains pointers how to do that. +* Jenkins is now using a `StatefulSet` instead of a `Deployment` +* terminology has been adjusted that's also reflected in values.yaml + The following values from `values.yaml` have been renamed: + + * `master` => `controller` + * `master.useSecurity` => `controller.adminSecret` + * `master.slaveListenerPort` => `controller.agentListenerPort` + * `master.slaveHostPort` => `controller.agentListenerHostPort` + * `master.slaveKubernetesNamespace` => `agent.namespace` + * `master.slaveDefaultsProviderTemplate` => `agent.defaultsProviderTemplate` + * `master.slaveJenkinsUrl` => `agent.jenkinsUrl` + * `master.slaveJenkinsTunnel` => `agent.jenkinsTunnel` + * `master.slaveConnectTimeout` => `agent.kubernetesConnectTimeout` + * `master.slaveReadTimeout` => `agent.kubernetesReadTimeout` + * `master.slaveListenerServiceAnnotations` => `controller.agentListenerServiceAnnotations` + * `master.slaveListenerServiceType` => `controller.agentListenerServiceType` + * `master.slaveListenerLoadBalancerIP` => `controller.agentListenerLoadBalancerIP` + * `agent.slaveConnectTimeout` => `agent.connectTimeout` +* Removed values: + + * `master.imageTag`: use `controller.image` and `controller.tag` instead + * `slave.imageTag`: use `agent.image` and `agent.tag` instead + +## To 2.0.0 + +Configuration as Code is now default + container does not run as root anymore. + +### Configuration as Code new default + +Configuration is done via [Jenkins Configuration as Code Plugin](https://github.com/jenkinsci/configuration-as-code-plugin) by default. +That means that changes in values which result in a configuration change are always applied. +In contrast, the XML configuration was only applied during the first start and never altered. + +:exclamation::exclamation::exclamation: +Attention: +This also means if you manually altered configuration then this will most likely be reset to what was configured by default. +It also applies to `securityRealm` and `authorizationStrategy` as they are also configured using configuration as code. +:exclamation::exclamation::exclamation: + +### Image does not run as root anymore + +It's not recommended to run containers in Kubernetes as `root`. + +❗Attention: If you had not configured a different user before then you need to ensure that your image supports the user and group ID configured and also manually change permissions of all files so that Jenkins is still able to use them. + +### Summary of updated values + +As version 2.0.0 only updates default values and nothing else it's still possible to migrate to this version and opt out of some or all new defaults. +All you have to do is ensure the old values are set in your installation. + +Here we show which values have changed and the previous default values: + +```yaml +controller: + runAsUser: 1000 # was unset before + fsGroup: 1000 # was unset before + JCasC: + enabled: true # was false + defaultConfig: true # was false + sidecars: + configAutoReload: + enabled: true # was false +``` + +### Migration steps + +Migration instructions heavily depend on your current setup. +So think of the list below more as a general guideline of what should be done. + +- Ensure that the Jenkins image you are using contains a user with ID 1000 and a group with the same ID. + That's the case for `jenkins/jenkins:lts` image, which the chart uses by default +- Make a backup of your existing installation especially the persistent volume +- Ensure that you have the configuration as code plugin installed +- Export your current settings via the plugin: + `Manage Jenkins` -> `Configuration as Code` -> `Download Configuration` +- prepare your values file for the update e.g. add additional configuration as code setting that you need. + The export taken from above might be a good starting point for this. + In addition, the [demos](https://github.com/jenkinsci/configuration-as-code-plugin/tree/master/demos) from the plugin itself are quite useful. +- Test drive those setting on a separate installation +- Put Jenkins to Quiet Down mode so that it does not accept new jobs + `/quietDown` +- Change permissions of all files and folders to the new user and group id: + + ```console + kubectl exec -it -c jenkins /bin/bash + chown -R 1000:1000 /var/jenkins_home + ``` + +- Update Jenkins + +## To 1.0.0 + +Breaking changes: + +- Values have been renamed to follow [helm recommended naming conventions](https://helm.sh/docs/chart_best_practices/#naming-conventions) so that all variables start with a lowercase letter and words are separated with camelcase +- All resources are now using [helm recommended standard labels](https://helm.sh/docs/chart_best_practices/#standard-labels) + +As a result of the label changes also the selectors of the deployment have been updated. +Those are immutable so trying an updated will cause an error like: + +```console +Error: Deployment.apps "jenkins" is invalid: spec.selector: Invalid value: v1.LabelSelector{MatchLabels:map[string]string{"app.kubernetes.io/component":"jenkins-controller", "app.kubernetes.io/instance":"jenkins"}, MatchExpressions:[]v1.LabelSelectorRequirement(nil)}: field is immutable +``` + +In order to upgrade, [uninstall](./README.md#uninstall-chart) the Jenkins Deployment before upgrading: diff --git a/charts/jenkins/jenkins/5.4.2/VALUES.md b/charts/jenkins/jenkins/5.4.2/VALUES.md new file mode 100644 index 000000000..e005be9c5 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/VALUES.md @@ -0,0 +1,309 @@ +# Jenkins + +## Configuration + +The following tables list the configurable parameters of the Jenkins chart and their default values. + +## Values + +| Key | Type | Description | Default | +|:----|:-----|:---------|:------------| +| [additionalAgents](./values.yaml#L1165) | object | Configure additional | `{}` | +| [additionalClouds](./values.yaml#L1190) | object | | `{}` | +| [agent.TTYEnabled](./values.yaml#L1083) | bool | Allocate pseudo tty to the side container | `false` | +| [agent.additionalContainers](./values.yaml#L1118) | list | Add additional containers to the agents | `[]` | +| [agent.alwaysPullImage](./values.yaml#L976) | bool | Always pull agent container image before build | `false` | +| [agent.annotations](./values.yaml#L1114) | object | Annotations to apply to the pod | `{}` | +| [agent.args](./values.yaml#L1077) | string | Arguments passed to command to execute | `"${computer.jnlpmac} ${computer.name}"` | +| [agent.command](./values.yaml#L1075) | string | Command to execute when side container starts | `nil` | +| [agent.componentName](./values.yaml#L944) | string | | `"jenkins-agent"` | +| [agent.connectTimeout](./values.yaml#L1112) | int | Timeout in seconds for an agent to be online | `100` | +| [agent.containerCap](./values.yaml#L1085) | int | Max number of agents to launch | `10` | +| [agent.customJenkinsLabels](./values.yaml#L941) | list | Append Jenkins labels to the agent | `[]` | +| [agent.defaultsProviderTemplate](./values.yaml#L907) | string | The name of the pod template to use for providing default values | `""` | +| [agent.directConnection](./values.yaml#L947) | bool | | `false` | +| [agent.disableDefaultAgent](./values.yaml#L1136) | bool | Disable the default Jenkins Agent configuration | `false` | +| [agent.enabled](./values.yaml#L905) | bool | Enable Kubernetes plugin jnlp-agent podTemplate | `true` | +| [agent.envVars](./values.yaml#L1058) | list | Environment variables for the agent Pod | `[]` | +| [agent.hostNetworking](./values.yaml#L955) | bool | Enables the agent to use the host network | `false` | +| [agent.idleMinutes](./values.yaml#L1090) | int | Allows the Pod to remain active for reuse until the configured number of minutes has passed since the last step was executed on it | `0` | +| [agent.image.repository](./values.yaml#L934) | string | Repository to pull the agent jnlp image from | `"jenkins/inbound-agent"` | +| [agent.image.tag](./values.yaml#L936) | string | Tag of the image to pull | `"3256.v88a_f6e922152-1"` | +| [agent.imagePullSecretName](./values.yaml#L943) | string | Name of the secret to be used to pull the image | `nil` | +| [agent.inheritYamlMergeStrategy](./values.yaml#L1110) | bool | Controls whether the defined yaml merge strategy will be inherited if another defined pod template is configured to inherit from the current one | `false` | +| [agent.jenkinsTunnel](./values.yaml#L915) | string | Overrides the Kubernetes Jenkins tunnel | `nil` | +| [agent.jenkinsUrl](./values.yaml#L911) | string | Overrides the Kubernetes Jenkins URL | `nil` | +| [agent.jnlpregistry](./values.yaml#L931) | string | Custom registry used to pull the agent jnlp image from | `nil` | +| [agent.kubernetesConnectTimeout](./values.yaml#L917) | int | The connection timeout in seconds for connections to Kubernetes API. The minimum value is 5 | `5` | +| [agent.kubernetesReadTimeout](./values.yaml#L919) | int | The read timeout in seconds for connections to Kubernetes API. The minimum value is 15 | `15` | +| [agent.livenessProbe](./values.yaml#L966) | object | | `{}` | +| [agent.maxRequestsPerHostStr](./values.yaml#L921) | string | The maximum concurrent connections to Kubernetes API | `"32"` | +| [agent.namespace](./values.yaml#L927) | string | Namespace in which the Kubernetes agents should be launched | `nil` | +| [agent.nodeSelector](./values.yaml#L1069) | object | Node labels for pod assignment | `{}` | +| [agent.nodeUsageMode](./values.yaml#L939) | string | | `"NORMAL"` | +| [agent.podLabels](./values.yaml#L929) | object | Custom Pod labels (an object with `label-key: label-value` pairs) | `{}` | +| [agent.podName](./values.yaml#L1087) | string | Agent Pod base name | `"default"` | +| [agent.podRetention](./values.yaml#L985) | string | | `"Never"` | +| [agent.podTemplates](./values.yaml#L1146) | object | Configures extra pod templates for the default kubernetes cloud | `{}` | +| [agent.privileged](./values.yaml#L949) | bool | Agent privileged container | `false` | +| [agent.resources](./values.yaml#L957) | object | Resources allocation (Requests and Limits) | `{"limits":{"cpu":"512m","memory":"512Mi"},"requests":{"cpu":"512m","memory":"512Mi"}}` | +| [agent.restrictedPssSecurityContext](./values.yaml#L982) | bool | Set a restricted securityContext on jnlp containers | `false` | +| [agent.retentionTimeout](./values.yaml#L923) | int | Time in minutes after which the Kubernetes cloud plugin will clean up an idle worker that has not already terminated | `5` | +| [agent.runAsGroup](./values.yaml#L953) | string | Configure container group | `nil` | +| [agent.runAsUser](./values.yaml#L951) | string | Configure container user | `nil` | +| [agent.secretEnvVars](./values.yaml#L1062) | list | Mount a secret as environment variable | `[]` | +| [agent.showRawYaml](./values.yaml#L989) | bool | | `true` | +| [agent.sideContainerName](./values.yaml#L1079) | string | Side container name | `"jnlp"` | +| [agent.volumes](./values.yaml#L996) | list | Additional volumes | `[]` | +| [agent.waitForPodSec](./values.yaml#L925) | int | Seconds to wait for pod to be running | `600` | +| [agent.websocket](./values.yaml#L946) | bool | Enables agent communication via websockets | `false` | +| [agent.workingDir](./values.yaml#L938) | string | Configure working directory for default agent | `"/home/jenkins/agent"` | +| [agent.workspaceVolume](./values.yaml#L1031) | object | Workspace volume (defaults to EmptyDir) | `{}` | +| [agent.yamlMergeStrategy](./values.yaml#L1108) | string | Defines how the raw yaml field gets merged with yaml definitions from inherited pod templates. Possible values: "merge" or "override" | `"override"` | +| [agent.yamlTemplate](./values.yaml#L1097) | string | The raw yaml of a Pod API Object to merge into the agent spec | `""` | +| [awsSecurityGroupPolicies.enabled](./values.yaml#L1316) | bool | | `false` | +| [awsSecurityGroupPolicies.policies[0].name](./values.yaml#L1318) | string | | `""` | +| [awsSecurityGroupPolicies.policies[0].podSelector](./values.yaml#L1320) | object | | `{}` | +| [awsSecurityGroupPolicies.policies[0].securityGroupIds](./values.yaml#L1319) | list | | `[]` | +| [checkDeprecation](./values.yaml#L1313) | bool | Checks if any deprecated values are used | `true` | +| [clusterZone](./values.yaml#L21) | string | Override the cluster name for FQDN resolving | `"cluster.local"` | +| [controller.JCasC.authorizationStrategy](./values.yaml#L533) | string | Jenkins Config as Code Authorization Strategy-section | `"loggedInUsersCanDoAnything:\n allowAnonymousRead: false"` | +| [controller.JCasC.configMapAnnotations](./values.yaml#L538) | object | Annotations for the JCasC ConfigMap | `{}` | +| [controller.JCasC.configScripts](./values.yaml#L507) | object | List of Jenkins Config as Code scripts | `{}` | +| [controller.JCasC.configUrls](./values.yaml#L504) | list | Remote URLs for configuration files. | `[]` | +| [controller.JCasC.defaultConfig](./values.yaml#L498) | bool | Enables default Jenkins configuration via configuration as code plugin | `true` | +| [controller.JCasC.overwriteConfiguration](./values.yaml#L502) | bool | Whether Jenkins Config as Code should overwrite any existing configuration | `false` | +| [controller.JCasC.security](./values.yaml#L514) | object | Jenkins Config as Code security-section | `{"apiToken":{"creationOfLegacyTokenEnabled":false,"tokenGenerationOnCreationEnabled":false,"usageStatisticsEnabled":true}}` | +| [controller.JCasC.securityRealm](./values.yaml#L522) | string | Jenkins Config as Code Security Realm-section | `"local:\n allowsSignup: false\n enableCaptcha: false\n users:\n - id: \"${chart-admin-username}\"\n name: \"Jenkins Admin\"\n password: \"${chart-admin-password}\""` | +| [controller.additionalExistingSecrets](./values.yaml#L459) | list | List of additional existing secrets to mount | `[]` | +| [controller.additionalPlugins](./values.yaml#L409) | list | List of plugins to install in addition to those listed in controller.installPlugins | `[]` | +| [controller.additionalSecrets](./values.yaml#L468) | list | List of additional secrets to create and mount | `[]` | +| [controller.admin.createSecret](./values.yaml#L91) | bool | Create secret for admin user | `true` | +| [controller.admin.existingSecret](./values.yaml#L94) | string | The name of an existing secret containing the admin credentials | `""` | +| [controller.admin.password](./values.yaml#L81) | string | Admin password created as a secret if `controller.admin.createSecret` is true | `` | +| [controller.admin.passwordKey](./values.yaml#L86) | string | The key in the existing admin secret containing the password | `"jenkins-admin-password"` | +| [controller.admin.userKey](./values.yaml#L84) | string | The key in the existing admin secret containing the username | `"jenkins-admin-user"` | +| [controller.admin.username](./values.yaml#L78) | string | Admin username created as a secret if `controller.admin.createSecret` is true | `"admin"` | +| [controller.affinity](./values.yaml#L660) | object | Affinity settings | `{}` | +| [controller.agentListenerEnabled](./values.yaml#L318) | bool | Create Agent listener service | `true` | +| [controller.agentListenerExternalTrafficPolicy](./values.yaml#L328) | string | Traffic Policy of for the agentListener service | `nil` | +| [controller.agentListenerHostPort](./values.yaml#L322) | string | Host port to listen for agents | `nil` | +| [controller.agentListenerLoadBalancerIP](./values.yaml#L358) | string | Static IP for the agentListener LoadBalancer | `nil` | +| [controller.agentListenerLoadBalancerSourceRanges](./values.yaml#L330) | list | Allowed inbound IP for the agentListener service | `["0.0.0.0/0"]` | +| [controller.agentListenerNodePort](./values.yaml#L324) | string | Node port to listen for agents | `nil` | +| [controller.agentListenerPort](./values.yaml#L320) | int | Listening port for agents | `50000` | +| [controller.agentListenerServiceAnnotations](./values.yaml#L353) | object | Annotations for the agentListener service | `{}` | +| [controller.agentListenerServiceType](./values.yaml#L350) | string | Defines how to expose the agentListener service | `"ClusterIP"` | +| [controller.backendconfig.annotations](./values.yaml#L763) | object | backendconfig annotations | `{}` | +| [controller.backendconfig.apiVersion](./values.yaml#L757) | string | backendconfig API version | `"extensions/v1beta1"` | +| [controller.backendconfig.enabled](./values.yaml#L755) | bool | Enables backendconfig | `false` | +| [controller.backendconfig.labels](./values.yaml#L761) | object | backendconfig labels | `{}` | +| [controller.backendconfig.name](./values.yaml#L759) | string | backendconfig name | `nil` | +| [controller.backendconfig.spec](./values.yaml#L765) | object | backendconfig spec | `{}` | +| [controller.cloudName](./values.yaml#L487) | string | Name of default cloud configuration. | `"kubernetes"` | +| [controller.clusterIp](./values.yaml#L217) | string | k8s service clusterIP. Only used if serviceType is ClusterIP | `nil` | +| [controller.componentName](./values.yaml#L34) | string | Used for label app.kubernetes.io/component | `"jenkins-controller"` | +| [controller.containerEnv](./values.yaml#L150) | list | Environment variables for Jenkins Container | `[]` | +| [controller.containerEnvFrom](./values.yaml#L147) | list | Environment variable sources for Jenkins Container | `[]` | +| [controller.containerSecurityContext](./values.yaml#L205) | object | Allow controlling the securityContext for the jenkins container | `{"allowPrivilegeEscalation":false,"readOnlyRootFilesystem":true,"runAsGroup":1000,"runAsUser":1000}` | +| [controller.csrf.defaultCrumbIssuer.enabled](./values.yaml#L339) | bool | Enable the default CSRF Crumb issuer | `true` | +| [controller.csrf.defaultCrumbIssuer.proxyCompatability](./values.yaml#L341) | bool | Enable proxy compatibility | `true` | +| [controller.customInitContainers](./values.yaml#L541) | list | Custom init-container specification in raw-yaml format | `[]` | +| [controller.customJenkinsLabels](./values.yaml#L68) | list | Append Jenkins labels to the controller | `[]` | +| [controller.disableRememberMe](./values.yaml#L59) | bool | Disable use of remember me | `false` | +| [controller.disabledAgentProtocols](./values.yaml#L333) | list | Disabled agent protocols | `["JNLP-connect","JNLP2-connect"]` | +| [controller.enableRawHtmlMarkupFormatter](./values.yaml#L429) | bool | Enable HTML parsing using OWASP Markup Formatter Plugin (antisamy-markup-formatter) | `false` | +| [controller.executorMode](./values.yaml#L65) | string | Sets the executor mode of the Jenkins node. Possible values are "NORMAL" or "EXCLUSIVE" | `"NORMAL"` | +| [controller.existingSecret](./values.yaml#L456) | string | | `nil` | +| [controller.extraPorts](./values.yaml#L388) | list | Optionally configure other ports to expose in the controller container | `[]` | +| [controller.fsGroup](./values.yaml#L186) | int | Deprecated in favor of `controller.podSecurityContextOverride`. uid that will be used for persistent volume. | `1000` | +| [controller.googlePodMonitor.enabled](./values.yaml#L826) | bool | | `false` | +| [controller.googlePodMonitor.scrapeEndpoint](./values.yaml#L831) | string | | `"/prometheus"` | +| [controller.googlePodMonitor.scrapeInterval](./values.yaml#L829) | string | | `"60s"` | +| [controller.healthProbes](./values.yaml#L248) | bool | Enable Kubernetes Probes configuration configured in `controller.probes` | `true` | +| [controller.hostAliases](./values.yaml#L779) | list | Allows for adding entries to Pod /etc/hosts | `[]` | +| [controller.hostNetworking](./values.yaml#L70) | bool | | `false` | +| [controller.httpsKeyStore.disableSecretMount](./values.yaml#L847) | bool | | `false` | +| [controller.httpsKeyStore.enable](./values.yaml#L838) | bool | Enables HTTPS keystore on jenkins controller | `false` | +| [controller.httpsKeyStore.fileName](./values.yaml#L855) | string | Jenkins keystore filename which will appear under controller.httpsKeyStore.path | `"keystore.jks"` | +| [controller.httpsKeyStore.httpPort](./values.yaml#L851) | int | HTTP Port that Jenkins should listen to along with HTTPS, it also serves as the liveness and readiness probes port. | `8081` | +| [controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretKey](./values.yaml#L846) | string | Name of the key in the secret that contains the JKS password | `"https-jks-password"` | +| [controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretName](./values.yaml#L844) | string | Name of the secret that contains the JKS password, if it is not in the same secret as the JKS file | `""` | +| [controller.httpsKeyStore.jenkinsHttpsJksSecretKey](./values.yaml#L842) | string | Name of the key in the secret that already has ssl keystore | `"jenkins-jks-file"` | +| [controller.httpsKeyStore.jenkinsHttpsJksSecretName](./values.yaml#L840) | string | Name of the secret that already has ssl keystore | `""` | +| [controller.httpsKeyStore.jenkinsKeyStoreBase64Encoded](./values.yaml#L860) | string | Base64 encoded Keystore content. Keystore must be converted to base64 then being pasted here | `nil` | +| [controller.httpsKeyStore.password](./values.yaml#L857) | string | Jenkins keystore password | `"password"` | +| [controller.httpsKeyStore.path](./values.yaml#L853) | string | Path of HTTPS keystore file | `"/var/jenkins_keystore"` | +| [controller.image.pullPolicy](./values.yaml#L47) | string | Controller image pull policy | `"Always"` | +| [controller.image.registry](./values.yaml#L37) | string | Controller image registry | `"docker.io"` | +| [controller.image.repository](./values.yaml#L39) | string | Controller image repository | `"jenkins/jenkins"` | +| [controller.image.tag](./values.yaml#L42) | string | Controller image tag override; i.e., tag: "2.440.1-jdk17" | `nil` | +| [controller.image.tagLabel](./values.yaml#L45) | string | Controller image tag label | `"jdk17"` | +| [controller.imagePullSecretName](./values.yaml#L49) | string | Controller image pull secret | `nil` | +| [controller.ingress.annotations](./values.yaml#L702) | object | Ingress annotations | `{}` | +| [controller.ingress.apiVersion](./values.yaml#L698) | string | Ingress API version | `"extensions/v1beta1"` | +| [controller.ingress.enabled](./values.yaml#L681) | bool | Enables ingress | `false` | +| [controller.ingress.hostName](./values.yaml#L715) | string | Ingress hostname | `nil` | +| [controller.ingress.labels](./values.yaml#L700) | object | Ingress labels | `{}` | +| [controller.ingress.path](./values.yaml#L711) | string | Ingress path | `nil` | +| [controller.ingress.paths](./values.yaml#L685) | list | Override for the default Ingress paths | `[]` | +| [controller.ingress.resourceRootUrl](./values.yaml#L717) | string | Hostname to serve assets from | `nil` | +| [controller.ingress.tls](./values.yaml#L719) | list | Ingress TLS configuration | `[]` | +| [controller.initConfigMap](./values.yaml#L446) | string | Name of the existing ConfigMap that contains init scripts | `nil` | +| [controller.initContainerEnv](./values.yaml#L141) | list | Environment variables for Init Container | `[]` | +| [controller.initContainerEnvFrom](./values.yaml#L137) | list | Environment variable sources for Init Container | `[]` | +| [controller.initContainerResources](./values.yaml#L128) | object | Resources allocation (Requests and Limits) for Init Container | `{}` | +| [controller.initScripts](./values.yaml#L442) | object | Map of groovy init scripts to be executed during Jenkins controller start | `{}` | +| [controller.initializeOnce](./values.yaml#L414) | bool | Initialize only on first installation. Ensures plugins do not get updated inadvertently. Requires `persistence.enabled` to be set to `true` | `false` | +| [controller.installLatestPlugins](./values.yaml#L403) | bool | Download the minimum required version or latest version of all dependencies | `true` | +| [controller.installLatestSpecifiedPlugins](./values.yaml#L406) | bool | Set to true to download the latest version of any plugin that is requested to have the latest version | `false` | +| [controller.installPlugins](./values.yaml#L395) | list | List of Jenkins plugins to install. If you don't want to install plugins, set it to `false` | `["kubernetes:4253.v7700d91739e5","workflow-aggregator:600.vb_57cdd26fdd7","git:5.2.2","configuration-as-code:1810.v9b_c30a_249a_4c"]` | +| [controller.javaOpts](./values.yaml#L156) | string | Append to `JAVA_OPTS` env var | `nil` | +| [controller.jenkinsAdminEmail](./values.yaml#L96) | string | Email address for the administrator of the Jenkins instance | `nil` | +| [controller.jenkinsHome](./values.yaml#L101) | string | Custom Jenkins home path | `"/var/jenkins_home"` | +| [controller.jenkinsOpts](./values.yaml#L158) | string | Append to `JENKINS_OPTS` env var | `nil` | +| [controller.jenkinsRef](./values.yaml#L106) | string | Custom Jenkins reference path | `"/usr/share/jenkins/ref"` | +| [controller.jenkinsUriPrefix](./values.yaml#L173) | string | Root URI Jenkins will be served on | `nil` | +| [controller.jenkinsUrl](./values.yaml#L168) | string | Set Jenkins URL if you are not using the ingress definitions provided by the chart | `nil` | +| [controller.jenkinsUrlProtocol](./values.yaml#L165) | string | Set protocol for Jenkins URL; `https` if `controller.ingress.tls`, `http` otherwise | `nil` | +| [controller.jenkinsWar](./values.yaml#L109) | string | | `"/usr/share/jenkins/jenkins.war"` | +| [controller.jmxPort](./values.yaml#L385) | string | Open a port, for JMX stats | `nil` | +| [controller.legacyRemotingSecurityEnabled](./values.yaml#L361) | bool | Whether legacy remoting security should be enabled | `false` | +| [controller.lifecycle](./values.yaml#L51) | object | Lifecycle specification for controller-container | `{}` | +| [controller.loadBalancerIP](./values.yaml#L376) | string | Optionally assign a known public LB IP | `nil` | +| [controller.loadBalancerSourceRanges](./values.yaml#L372) | list | Allowed inbound IP addresses | `["0.0.0.0/0"]` | +| [controller.markupFormatter](./values.yaml#L433) | string | Yaml of the markup formatter to use | `"plainText"` | +| [controller.nodePort](./values.yaml#L223) | string | k8s node port. Only used if serviceType is NodePort | `nil` | +| [controller.nodeSelector](./values.yaml#L647) | object | Node labels for pod assignment | `{}` | +| [controller.numExecutors](./values.yaml#L62) | int | Set Number of executors | `0` | +| [controller.overwritePlugins](./values.yaml#L418) | bool | Overwrite installed plugins on start | `false` | +| [controller.overwritePluginsFromImage](./values.yaml#L422) | bool | Overwrite plugins that are already installed in the controller image | `true` | +| [controller.podAnnotations](./values.yaml#L668) | object | Annotations for controller pod | `{}` | +| [controller.podDisruptionBudget.annotations](./values.yaml#L312) | object | | `{}` | +| [controller.podDisruptionBudget.apiVersion](./values.yaml#L310) | string | Policy API version | `"policy/v1beta1"` | +| [controller.podDisruptionBudget.enabled](./values.yaml#L305) | bool | Enable Kubernetes Pod Disruption Budget configuration | `false` | +| [controller.podDisruptionBudget.labels](./values.yaml#L313) | object | | `{}` | +| [controller.podDisruptionBudget.maxUnavailable](./values.yaml#L315) | string | Number of pods that can be unavailable. Either an absolute number or a percentage | `"0"` | +| [controller.podLabels](./values.yaml#L241) | object | Custom Pod labels (an object with `label-key: label-value` pairs) | `{}` | +| [controller.podSecurityContextOverride](./values.yaml#L202) | string | Completely overwrites the contents of the pod security context, ignoring the values provided for `runAsUser`, `fsGroup`, and `securityContextCapabilities` | `nil` | +| [controller.priorityClassName](./values.yaml#L665) | string | The name of a `priorityClass` to apply to the controller pod | `nil` | +| [controller.probes.livenessProbe.failureThreshold](./values.yaml#L266) | int | Set the failure threshold for the liveness probe | `5` | +| [controller.probes.livenessProbe.httpGet.path](./values.yaml#L269) | string | Set the Pod's HTTP path for the liveness probe | `"{{ default \"\" .Values.controller.jenkinsUriPrefix }}/login"` | +| [controller.probes.livenessProbe.httpGet.port](./values.yaml#L271) | string | Set the Pod's HTTP port to use for the liveness probe | `"http"` | +| [controller.probes.livenessProbe.initialDelaySeconds](./values.yaml#L280) | string | Set the initial delay for the liveness probe in seconds | `nil` | +| [controller.probes.livenessProbe.periodSeconds](./values.yaml#L273) | int | Set the time interval between two liveness probes executions in seconds | `10` | +| [controller.probes.livenessProbe.timeoutSeconds](./values.yaml#L275) | int | Set the timeout for the liveness probe in seconds | `5` | +| [controller.probes.readinessProbe.failureThreshold](./values.yaml#L284) | int | Set the failure threshold for the readiness probe | `3` | +| [controller.probes.readinessProbe.httpGet.path](./values.yaml#L287) | string | Set the Pod's HTTP path for the liveness probe | `"{{ default \"\" .Values.controller.jenkinsUriPrefix }}/login"` | +| [controller.probes.readinessProbe.httpGet.port](./values.yaml#L289) | string | Set the Pod's HTTP port to use for the readiness probe | `"http"` | +| [controller.probes.readinessProbe.initialDelaySeconds](./values.yaml#L298) | string | Set the initial delay for the readiness probe in seconds | `nil` | +| [controller.probes.readinessProbe.periodSeconds](./values.yaml#L291) | int | Set the time interval between two readiness probes executions in seconds | `10` | +| [controller.probes.readinessProbe.timeoutSeconds](./values.yaml#L293) | int | Set the timeout for the readiness probe in seconds | `5` | +| [controller.probes.startupProbe.failureThreshold](./values.yaml#L253) | int | Set the failure threshold for the startup probe | `12` | +| [controller.probes.startupProbe.httpGet.path](./values.yaml#L256) | string | Set the Pod's HTTP path for the startup probe | `"{{ default \"\" .Values.controller.jenkinsUriPrefix }}/login"` | +| [controller.probes.startupProbe.httpGet.port](./values.yaml#L258) | string | Set the Pod's HTTP port to use for the startup probe | `"http"` | +| [controller.probes.startupProbe.periodSeconds](./values.yaml#L260) | int | Set the time interval between two startup probes executions in seconds | `10` | +| [controller.probes.startupProbe.timeoutSeconds](./values.yaml#L262) | int | Set the timeout for the startup probe in seconds | `5` | +| [controller.projectNamingStrategy](./values.yaml#L425) | string | | `"standard"` | +| [controller.prometheus.alertingRulesAdditionalLabels](./values.yaml#L812) | object | Additional labels to add to the PrometheusRule object | `{}` | +| [controller.prometheus.alertingrules](./values.yaml#L810) | list | Array of prometheus alerting rules | `[]` | +| [controller.prometheus.enabled](./values.yaml#L795) | bool | Enables prometheus service monitor | `false` | +| [controller.prometheus.metricRelabelings](./values.yaml#L822) | list | | `[]` | +| [controller.prometheus.prometheusRuleNamespace](./values.yaml#L814) | string | Set a custom namespace where to deploy PrometheusRule resource | `""` | +| [controller.prometheus.relabelings](./values.yaml#L820) | list | | `[]` | +| [controller.prometheus.scrapeEndpoint](./values.yaml#L805) | string | The endpoint prometheus should get metrics from | `"/prometheus"` | +| [controller.prometheus.scrapeInterval](./values.yaml#L801) | string | How often prometheus should scrape metrics | `"60s"` | +| [controller.prometheus.serviceMonitorAdditionalLabels](./values.yaml#L797) | object | Additional labels to add to the service monitor object | `{}` | +| [controller.prometheus.serviceMonitorNamespace](./values.yaml#L799) | string | Set a custom namespace where to deploy ServiceMonitor resource | `nil` | +| [controller.resources](./values.yaml#L115) | object | Resource allocation (Requests and Limits) | `{"limits":{"cpu":"2000m","memory":"4096Mi"},"requests":{"cpu":"50m","memory":"256Mi"}}` | +| [controller.route.annotations](./values.yaml#L774) | object | Route annotations | `{}` | +| [controller.route.enabled](./values.yaml#L770) | bool | Enables openshift route | `false` | +| [controller.route.labels](./values.yaml#L772) | object | Route labels | `{}` | +| [controller.route.path](./values.yaml#L776) | string | Route path | `nil` | +| [controller.runAsUser](./values.yaml#L183) | int | Deprecated in favor of `controller.podSecurityContextOverride`. uid that jenkins runs with. | `1000` | +| [controller.schedulerName](./values.yaml#L643) | string | Name of the Kubernetes scheduler to use | `""` | +| [controller.scriptApproval](./values.yaml#L437) | list | List of groovy functions to approve | `[]` | +| [controller.secondaryingress.annotations](./values.yaml#L737) | object | | `{}` | +| [controller.secondaryingress.apiVersion](./values.yaml#L735) | string | | `"extensions/v1beta1"` | +| [controller.secondaryingress.enabled](./values.yaml#L729) | bool | | `false` | +| [controller.secondaryingress.hostName](./values.yaml#L744) | string | | `nil` | +| [controller.secondaryingress.labels](./values.yaml#L736) | object | | `{}` | +| [controller.secondaryingress.paths](./values.yaml#L732) | list | | `[]` | +| [controller.secondaryingress.tls](./values.yaml#L745) | string | | `nil` | +| [controller.secretClaims](./values.yaml#L480) | list | List of `SecretClaim` resources to create | `[]` | +| [controller.securityContextCapabilities](./values.yaml#L192) | object | | `{}` | +| [controller.serviceAnnotations](./values.yaml#L230) | object | Jenkins controller service annotations | `{}` | +| [controller.serviceExternalTrafficPolicy](./values.yaml#L227) | string | | `nil` | +| [controller.serviceLabels](./values.yaml#L236) | object | Labels for the Jenkins controller-service | `{}` | +| [controller.servicePort](./values.yaml#L219) | int | k8s service port | `8080` | +| [controller.serviceType](./values.yaml#L214) | string | k8s service type | `"ClusterIP"` | +| [controller.shareProcessNamespace](./values.yaml#L124) | bool | | `false` | +| [controller.sidecars.additionalSidecarContainers](./values.yaml#L625) | list | Configures additional sidecar container(s) for the Jenkins controller | `[]` | +| [controller.sidecars.configAutoReload.additionalVolumeMounts](./values.yaml#L571) | list | Enables additional volume mounts for the config auto-reload container | `[]` | +| [controller.sidecars.configAutoReload.containerSecurityContext](./values.yaml#L620) | object | Enable container security context | `{"allowPrivilegeEscalation":false,"readOnlyRootFilesystem":true}` | +| [controller.sidecars.configAutoReload.enabled](./values.yaml#L554) | bool | Enables Jenkins Config as Code auto-reload | `true` | +| [controller.sidecars.configAutoReload.env](./values.yaml#L602) | object | Environment variables for the Jenkins Config as Code auto-reload container | `{}` | +| [controller.sidecars.configAutoReload.envFrom](./values.yaml#L600) | list | Environment variable sources for the Jenkins Config as Code auto-reload container | `[]` | +| [controller.sidecars.configAutoReload.folder](./values.yaml#L613) | string | | `"/var/jenkins_home/casc_configs"` | +| [controller.sidecars.configAutoReload.image.registry](./values.yaml#L557) | string | Registry for the image that triggers the reload | `"docker.io"` | +| [controller.sidecars.configAutoReload.image.repository](./values.yaml#L559) | string | Repository of the image that triggers the reload | `"kiwigrid/k8s-sidecar"` | +| [controller.sidecars.configAutoReload.image.tag](./values.yaml#L561) | string | Tag for the image that triggers the reload | `"1.27.5"` | +| [controller.sidecars.configAutoReload.imagePullPolicy](./values.yaml#L562) | string | | `"IfNotPresent"` | +| [controller.sidecars.configAutoReload.logging](./values.yaml#L577) | object | Config auto-reload logging settings | `{"configuration":{"backupCount":3,"formatter":"JSON","logLevel":"INFO","logToConsole":true,"logToFile":false,"maxBytes":1024,"override":false}}` | +| [controller.sidecars.configAutoReload.logging.configuration.override](./values.yaml#L581) | bool | Enables custom log config utilizing using the settings below. | `false` | +| [controller.sidecars.configAutoReload.reqRetryConnect](./values.yaml#L595) | int | How many connection-related errors to retry on | `10` | +| [controller.sidecars.configAutoReload.resources](./values.yaml#L563) | object | | `{}` | +| [controller.sidecars.configAutoReload.scheme](./values.yaml#L590) | string | The scheme to use when connecting to the Jenkins configuration as code endpoint | `"http"` | +| [controller.sidecars.configAutoReload.skipTlsVerify](./values.yaml#L592) | bool | Skip TLS verification when connecting to the Jenkins configuration as code endpoint | `false` | +| [controller.sidecars.configAutoReload.sleepTime](./values.yaml#L597) | string | How many seconds to wait before updating config-maps/secrets (sets METHOD=SLEEP on the sidecar) | `nil` | +| [controller.sidecars.configAutoReload.sshTcpPort](./values.yaml#L611) | int | | `1044` | +| [controller.statefulSetAnnotations](./values.yaml#L670) | object | Annotations for controller StatefulSet | `{}` | +| [controller.statefulSetLabels](./values.yaml#L232) | object | Jenkins controller custom labels for the StatefulSet | `{}` | +| [controller.targetPort](./values.yaml#L221) | int | k8s target port | `8080` | +| [controller.terminationGracePeriodSeconds](./values.yaml#L653) | string | Set TerminationGracePeriodSeconds | `nil` | +| [controller.terminationMessagePath](./values.yaml#L655) | string | Set the termination message path | `nil` | +| [controller.terminationMessagePolicy](./values.yaml#L657) | string | Set the termination message policy | `nil` | +| [controller.testEnabled](./values.yaml#L834) | bool | Can be used to disable rendering controller test resources when using helm template | `true` | +| [controller.tolerations](./values.yaml#L651) | list | Toleration labels for pod assignment | `[]` | +| [controller.topologySpreadConstraints](./values.yaml#L677) | object | Topology spread constraints | `{}` | +| [controller.updateStrategy](./values.yaml#L674) | object | Update strategy for StatefulSet | `{}` | +| [controller.usePodSecurityContext](./values.yaml#L176) | bool | Enable pod security context (must be `true` if podSecurityContextOverride, runAsUser or fsGroup are set) | `true` | +| [credentialsId](./values.yaml#L27) | string | The Jenkins credentials to access the Kubernetes API server. For the default cluster it is not needed. | `nil` | +| [fullnameOverride](./values.yaml#L13) | string | Override the full resource names | `jenkins-(release-name)` or `jenkins` if the release-name is `jenkins` | +| [helmtest.bats.image.registry](./values.yaml#L1329) | string | Registry of the image used to test the framework | `"docker.io"` | +| [helmtest.bats.image.repository](./values.yaml#L1331) | string | Repository of the image used to test the framework | `"bats/bats"` | +| [helmtest.bats.image.tag](./values.yaml#L1333) | string | Tag of the image to test the framework | `"1.11.0"` | +| [kubernetesURL](./values.yaml#L24) | string | The URL of the Kubernetes API server | `"https://kubernetes.default"` | +| [nameOverride](./values.yaml#L10) | string | Override the resource name prefix | `Chart.Name` | +| [namespaceOverride](./values.yaml#L16) | string | Override the deployment namespace | `Release.Namespace` | +| [networkPolicy.apiVersion](./values.yaml#L1259) | string | NetworkPolicy ApiVersion | `"networking.k8s.io/v1"` | +| [networkPolicy.enabled](./values.yaml#L1254) | bool | Enable the creation of NetworkPolicy resources | `false` | +| [networkPolicy.externalAgents.except](./values.yaml#L1273) | list | A list of IP sub-ranges to be excluded from the allowlisted IP range | `[]` | +| [networkPolicy.externalAgents.ipCIDR](./values.yaml#L1271) | string | The IP range from which external agents are allowed to connect to controller, i.e., 172.17.0.0/16 | `nil` | +| [networkPolicy.internalAgents.allowed](./values.yaml#L1263) | bool | Allow internal agents (from the same cluster) to connect to controller. Agent pods will be filtered based on PodLabels | `true` | +| [networkPolicy.internalAgents.namespaceLabels](./values.yaml#L1267) | object | A map of labels (keys/values) that agents namespaces must have to be able to connect to controller | `{}` | +| [networkPolicy.internalAgents.podLabels](./values.yaml#L1265) | object | A map of labels (keys/values) that agent pods must have to be able to connect to controller | `{}` | +| [persistence.accessMode](./values.yaml#L1229) | string | The PVC access mode | `"ReadWriteOnce"` | +| [persistence.annotations](./values.yaml#L1225) | object | Annotations for the PVC | `{}` | +| [persistence.dataSource](./values.yaml#L1235) | object | Existing data source to clone PVC from | `{}` | +| [persistence.enabled](./values.yaml#L1209) | bool | Enable the use of a Jenkins PVC | `true` | +| [persistence.existingClaim](./values.yaml#L1215) | string | Provide the name of a PVC | `nil` | +| [persistence.labels](./values.yaml#L1227) | object | Labels for the PVC | `{}` | +| [persistence.mounts](./values.yaml#L1247) | list | Additional mounts | `[]` | +| [persistence.size](./values.yaml#L1231) | string | The size of the PVC | `"8Gi"` | +| [persistence.storageClass](./values.yaml#L1223) | string | Storage class for the PVC | `nil` | +| [persistence.subPath](./values.yaml#L1240) | string | SubPath for jenkins-home mount | `nil` | +| [persistence.volumes](./values.yaml#L1242) | list | Additional volumes | `[]` | +| [rbac.create](./values.yaml#L1279) | bool | Whether RBAC resources are created | `true` | +| [rbac.readSecrets](./values.yaml#L1281) | bool | Whether the Jenkins service account should be able to read Kubernetes secrets | `false` | +| [renderHelmLabels](./values.yaml#L30) | bool | Enables rendering of the helm.sh/chart label to the annotations | `true` | +| [serviceAccount.annotations](./values.yaml#L1291) | object | Configures annotations for the ServiceAccount | `{}` | +| [serviceAccount.create](./values.yaml#L1285) | bool | Configures if a ServiceAccount with this name should be created | `true` | +| [serviceAccount.extraLabels](./values.yaml#L1293) | object | Configures extra labels for the ServiceAccount | `{}` | +| [serviceAccount.imagePullSecretName](./values.yaml#L1295) | string | Controller ServiceAccount image pull secret | `nil` | +| [serviceAccount.name](./values.yaml#L1289) | string | | `nil` | +| [serviceAccountAgent.annotations](./values.yaml#L1306) | object | Configures annotations for the agent ServiceAccount | `{}` | +| [serviceAccountAgent.create](./values.yaml#L1300) | bool | Configures if an agent ServiceAccount should be created | `false` | +| [serviceAccountAgent.extraLabels](./values.yaml#L1308) | object | Configures extra labels for the agent ServiceAccount | `{}` | +| [serviceAccountAgent.imagePullSecretName](./values.yaml#L1310) | string | Agent ServiceAccount image pull secret | `nil` | +| [serviceAccountAgent.name](./values.yaml#L1304) | string | The name of the agent ServiceAccount to be used by access-controlled resources | `nil` | diff --git a/charts/jenkins/jenkins/5.4.2/VALUES.md.gotmpl b/charts/jenkins/jenkins/5.4.2/VALUES.md.gotmpl new file mode 100644 index 000000000..21080e35a --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/VALUES.md.gotmpl @@ -0,0 +1,28 @@ +# Jenkins + +## Configuration + +The following tables list the configurable parameters of the Jenkins chart and their default values. + +{{- define "chart.valueDefaultColumnRender" -}} +{{- $defaultValue := (trimAll "`" (default .Default .AutoDefault) | replace "\n" "") -}} +`{{- $defaultValue | replace "\n" "" -}}` +{{- end -}} + +{{- define "chart.typeColumnRender" -}} +{{- .Type -}} +{{- end -}} + +{{- define "chart.valueDescription" -}} +{{- default .Description .AutoDescription }} +{{- end -}} + +{{- define "chart.valuesTable" -}} +| Key | Type | Description | Default | +|:----|:-----|:---------|:------------| +{{- range .Values }} +| [{{ .Key }}](./values.yaml#L{{ .LineNumber }}) | {{ template "chart.typeColumnRender" . }} | {{ template "chart.valueDescription" . }} | {{ template "chart.valueDefaultColumnRender" . }} | +{{- end }} +{{- end }} + +{{ template "chart.valuesSection" . }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/NOTES.txt b/charts/jenkins/jenkins/5.4.2/templates/NOTES.txt new file mode 100644 index 000000000..953dd2606 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/NOTES.txt @@ -0,0 +1,68 @@ +{{- $prefix := .Values.controller.jenkinsUriPrefix | default "" -}} +{{- $url := "" -}} +1. Get your '{{ .Values.controller.admin.username }}' user password by running: + kubectl exec --namespace {{ template "jenkins.namespace" . }} -it svc/{{ template "jenkins.fullname" . }} -c jenkins -- /bin/cat /run/secrets/additional/chart-admin-password && echo +{{- if .Values.controller.ingress.hostName -}} +{{- if .Values.controller.ingress.tls -}} +{{- $url = print "https://" .Values.controller.ingress.hostName $prefix -}} +{{- else -}} +{{- $url = print "http://" .Values.controller.ingress.hostName $prefix -}} +{{- end }} +2. Visit {{ $url }} +{{- else }} +2. Get the Jenkins URL to visit by running these commands in the same shell: +{{- if contains "NodePort" .Values.controller.serviceType }} + export NODE_PORT=$(kubectl get --namespace {{ template "jenkins.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "jenkins.fullname" . }}) + export NODE_IP=$(kubectl get nodes --namespace {{ template "jenkins.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}") +{{- if .Values.controller.httpsKeyStore.enable -}} +{{- $url = print "https://$NODE_IP:$NODE_PORT" $prefix -}} +{{- else -}} +{{- $url = print "http://$NODE_IP:$NODE_PORT" $prefix -}} +{{- end }} + echo {{ $url }} + +{{- else if contains "LoadBalancer" .Values.controller.serviceType }} + NOTE: It may take a few minutes for the LoadBalancer IP to be available. + You can watch the status of by running 'kubectl get svc --namespace {{ template "jenkins.namespace" . }} -w {{ template "jenkins.fullname" . }}' + export SERVICE_IP=$(kubectl get svc --namespace {{ template "jenkins.namespace" . }} {{ template "jenkins.fullname" . }} --template "{{ "{{ range (index .status.loadBalancer.ingress 0) }}{{ . }}{{ end }}" }}") +{{- if .Values.controller.httpsKeyStore.enable -}} +{{- $url = print "https://$SERVICE_IP:" .Values.controller.servicePort $prefix -}} +{{- else -}} +{{- $url = print "http://$SERVICE_IP:" .Values.controller.servicePort $prefix -}} +{{- end }} + echo {{ $url }} + +{{- else if contains "ClusterIP" .Values.controller.serviceType -}} +{{- if .Values.controller.httpsKeyStore.enable -}} +{{- $url = print "https://127.0.0.1:" .Values.controller.servicePort $prefix -}} +{{- else -}} +{{- $url = print "http://127.0.0.1:" .Values.controller.servicePort $prefix -}} +{{- end }} + echo {{ $url }} + kubectl --namespace {{ template "jenkins.namespace" . }} port-forward svc/{{template "jenkins.fullname" . }} {{ .Values.controller.servicePort }}:{{ .Values.controller.servicePort }} +{{- end }} +{{- end }} + +3. Login with the password from step 1 and the username: {{ .Values.controller.admin.username }} +4. Configure security realm and authorization strategy +5. Use Jenkins Configuration as Code by specifying configScripts in your values.yaml file, see documentation: {{ $url }}/configuration-as-code and examples: https://github.com/jenkinsci/configuration-as-code-plugin/tree/master/demos + +For more information on running Jenkins on Kubernetes, visit: +https://cloud.google.com/solutions/jenkins-on-container-engine + +For more information about Jenkins Configuration as Code, visit: +https://jenkins.io/projects/jcasc/ + +{{ if and (eq .Values.controller.image.repository "jenkins/jenkins") (eq .Values.controller.image.registry "docker.io") }} +NOTE: Consider using a custom image with pre-installed plugins +{{- else if .Values.controller.installPlugins }} +NOTE: Consider disabling `installPlugins` if your image already contains plugins. +{{- end }} + +{{- if .Values.persistence.enabled }} +{{- else }} +################################################################################# +###### WARNING: Persistence is disabled!!! You will lose your data when ##### +###### the Jenkins pod is terminated. ##### +################################################################################# +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/_helpers.tpl b/charts/jenkins/jenkins/5.4.2/templates/_helpers.tpl new file mode 100644 index 000000000..e334e5848 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/_helpers.tpl @@ -0,0 +1,669 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "jenkins.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Expand the label of the chart. +*/}} +{{- define "jenkins.label" -}} +{{- printf "%s-%s" (include "jenkins.name" .) .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" -}} +{{- end -}} + + +{{/* +Allow the release namespace to be overridden for multi-namespace deployments in combined charts. +*/}} +{{- define "jenkins.namespace" -}} + {{- if .Values.namespaceOverride -}} + {{- .Values.namespaceOverride -}} + {{- else -}} + {{- .Release.Namespace -}} + {{- end -}} +{{- end -}} + +{{- define "jenkins.agent.namespace" -}} + {{- if .Values.agent.namespace -}} + {{- tpl .Values.agent.namespace . -}} + {{- else -}} + {{- if .Values.namespaceOverride -}} + {{- .Values.namespaceOverride -}} + {{- else -}} + {{- .Release.Namespace -}} + {{- 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 "jenkins.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 -}} + +{{/* +Returns the admin password +https://github.com/helm/charts/issues/5167#issuecomment-619137759 +*/}} +{{- define "jenkins.password" -}} + {{- if .Values.controller.admin.password -}} + {{- .Values.controller.admin.password | b64enc | quote }} + {{- else -}} + {{- $secret := (lookup "v1" "Secret" .Release.Namespace (include "jenkins.fullname" .)).data -}} + {{- if $secret -}} + {{/* + Reusing current password since secret exists + */}} + {{- index $secret ( .Values.controller.admin.passwordKey | default "jenkins-admin-password" ) -}} + {{- else -}} + {{/* + Generate new password + */}} + {{- randAlphaNum 22 | b64enc | quote }} + {{- end -}} + {{- end -}} +{{- end -}} + +{{/* +Returns the Jenkins URL +*/}} +{{- define "jenkins.url" -}} +{{- if .Values.controller.jenkinsUrl }} + {{- .Values.controller.jenkinsUrl }} +{{- else }} + {{- if .Values.controller.ingress.hostName }} + {{- if .Values.controller.ingress.tls }} + {{- default "https" .Values.controller.jenkinsUrlProtocol }}://{{ tpl .Values.controller.ingress.hostName $ }}{{ default "" .Values.controller.jenkinsUriPrefix }} + {{- else }} + {{- default "http" .Values.controller.jenkinsUrlProtocol }}://{{ tpl .Values.controller.ingress.hostName $ }}{{ default "" .Values.controller.jenkinsUriPrefix }} + {{- end }} + {{- else }} + {{- default "http" .Values.controller.jenkinsUrlProtocol }}://{{ template "jenkins.fullname" . }}:{{.Values.controller.servicePort}}{{ default "" .Values.controller.jenkinsUriPrefix }} + {{- end}} +{{- end}} +{{- end -}} + +{{/* +Returns configuration as code default config +*/}} +{{- define "jenkins.casc.defaults" -}} +jenkins: + {{- $configScripts := toYaml .Values.controller.JCasC.configScripts }} + {{- if and (.Values.controller.JCasC.authorizationStrategy) (not (contains "authorizationStrategy:" $configScripts)) }} + authorizationStrategy: + {{- tpl .Values.controller.JCasC.authorizationStrategy . | nindent 4 }} + {{- end }} + {{- if and (.Values.controller.JCasC.securityRealm) (not (contains "securityRealm:" $configScripts)) }} + securityRealm: + {{- tpl .Values.controller.JCasC.securityRealm . | nindent 4 }} + {{- end }} + disableRememberMe: {{ .Values.controller.disableRememberMe }} + {{- if .Values.controller.legacyRemotingSecurityEnabled }} + remotingSecurity: + enabled: true + {{- end }} + mode: {{ .Values.controller.executorMode }} + numExecutors: {{ .Values.controller.numExecutors }} + {{- if not (kindIs "invalid" .Values.controller.customJenkinsLabels) }} + labelString: "{{ join " " .Values.controller.customJenkinsLabels }}" + {{- end }} + {{- if .Values.controller.projectNamingStrategy }} + {{- if kindIs "string" .Values.controller.projectNamingStrategy }} + projectNamingStrategy: "{{ .Values.controller.projectNamingStrategy }}" + {{- else }} + projectNamingStrategy: + {{- toYaml .Values.controller.projectNamingStrategy | nindent 4 }} + {{- end }} + {{- end }} + markupFormatter: + {{- if .Values.controller.enableRawHtmlMarkupFormatter }} + rawHtml: + disableSyntaxHighlighting: true + {{- else }} + {{- toYaml .Values.controller.markupFormatter | nindent 4 }} + {{- end }} + clouds: + - kubernetes: + containerCapStr: "{{ .Values.agent.containerCap }}" + {{- if .Values.agent.jnlpregistry }} + jnlpregistry: "{{ .Values.agent.jnlpregistry }}" + {{- end }} + defaultsProviderTemplate: "{{ .Values.agent.defaultsProviderTemplate }}" + connectTimeout: "{{ .Values.agent.kubernetesConnectTimeout }}" + readTimeout: "{{ .Values.agent.kubernetesReadTimeout }}" + {{- if .Values.agent.directConnection }} + directConnection: true + {{- else }} + {{- if .Values.agent.jenkinsUrl }} + jenkinsUrl: "{{ tpl .Values.agent.jenkinsUrl . }}" + {{- else }} + jenkinsUrl: "http://{{ template "jenkins.fullname" . }}.{{ template "jenkins.namespace" . }}.svc.{{.Values.clusterZone}}:{{.Values.controller.servicePort}}{{ default "" .Values.controller.jenkinsUriPrefix }}" + {{- end }} + {{- if not .Values.agent.websocket }} + {{- if .Values.agent.jenkinsTunnel }} + jenkinsTunnel: "{{ tpl .Values.agent.jenkinsTunnel . }}" + {{- else }} + jenkinsTunnel: "{{ template "jenkins.fullname" . }}-agent.{{ template "jenkins.namespace" . }}.svc.{{.Values.clusterZone}}:{{ .Values.controller.agentListenerPort }}" + {{- end }} + {{- else }} + webSocket: true + {{- end }} + {{- end }} + maxRequestsPerHostStr: {{ .Values.agent.maxRequestsPerHostStr | quote }} + retentionTimeout: {{ .Values.agent.retentionTimeout | quote }} + waitForPodSec: {{ .Values.agent.waitForPodSec | quote }} + name: "{{ .Values.controller.cloudName }}" + namespace: "{{ template "jenkins.agent.namespace" . }}" + restrictedPssSecurityContext: {{ .Values.agent.restrictedPssSecurityContext }} + serverUrl: "{{ .Values.kubernetesURL }}" + credentialsId: "{{ .Values.credentialsId }}" + {{- if .Values.agent.enabled }} + podLabels: + - key: "jenkins/{{ .Release.Name }}-{{ .Values.agent.componentName }}" + value: "true" + {{- range $key, $val := .Values.agent.podLabels }} + - key: {{ $key | quote }} + value: {{ $val | quote }} + {{- end }} + templates: + {{- if not .Values.agent.disableDefaultAgent }} + {{- include "jenkins.casc.podTemplate" . | nindent 8 }} + {{- end }} + {{- if .Values.additionalAgents }} + {{- /* save .Values.agent */}} + {{- $agent := .Values.agent }} + {{- range $name, $additionalAgent := .Values.additionalAgents }} + {{- $additionalContainersEmpty := and (hasKey $additionalAgent "additionalContainers") (empty $additionalAgent.additionalContainers) }} + {{- /* merge original .Values.agent into additional agent to ensure it at least has the default values */}} + {{- $additionalAgent := merge $additionalAgent $agent }} + {{- /* clear list of additional containers in case it is configured empty for this agent (merge might have overwritten that) */}} + {{- if $additionalContainersEmpty }} + {{- $_ := set $additionalAgent "additionalContainers" list }} + {{- end }} + {{- /* set .Values.agent to $additionalAgent */}} + {{- $_ := set $.Values "agent" $additionalAgent }} + {{- include "jenkins.casc.podTemplate" $ | nindent 8 }} + {{- end }} + {{- /* restore .Values.agent */}} + {{- $_ := set .Values "agent" $agent }} + {{- end }} + {{- if .Values.agent.podTemplates }} + {{- range $key, $val := .Values.agent.podTemplates }} + {{- tpl $val $ | nindent 8 }} + {{- end }} + {{- end }} + {{- end }} + {{- if .Values.additionalClouds }} + {{- /* save root */}} + {{- $oldRoot := deepCopy $ }} + {{- range $name, $additionalCloud := .Values.additionalClouds }} + {{- $newRoot := deepCopy $ }} + {{- /* clear additionalAgents from the copy if override set to `true` */}} + {{- if .additionalAgentsOverride }} + {{- $_ := set $newRoot.Values "additionalAgents" list}} + {{- end}} + {{- $newValues := merge $additionalCloud $newRoot.Values }} + {{- $_ := set $newRoot "Values" $newValues }} + {{- /* clear additionalClouds from the copy */}} + {{- $_ := set $newRoot.Values "additionalClouds" list }} + {{- with $newRoot}} + - kubernetes: + containerCapStr: "{{ .Values.agent.containerCap }}" + {{- if .Values.agent.jnlpregistry }} + jnlpregistry: "{{ .Values.agent.jnlpregistry }}" + {{- end }} + defaultsProviderTemplate: "{{ .Values.agent.defaultsProviderTemplate }}" + connectTimeout: "{{ .Values.agent.kubernetesConnectTimeout }}" + readTimeout: "{{ .Values.agent.kubernetesReadTimeout }}" + {{- if .Values.agent.directConnection }} + directConnection: true + {{- else }} + {{- if .Values.agent.jenkinsUrl }} + jenkinsUrl: "{{ tpl .Values.agent.jenkinsUrl . }}" + {{- else }} + jenkinsUrl: "http://{{ template "jenkins.fullname" . }}.{{ template "jenkins.namespace" . }}.svc.{{.Values.clusterZone}}:{{.Values.controller.servicePort}}{{ default "" .Values.controller.jenkinsUriPrefix }}" + {{- end }} + {{- if not .Values.agent.websocket }} + {{- if .Values.agent.jenkinsTunnel }} + jenkinsTunnel: "{{ tpl .Values.agent.jenkinsTunnel . }}" + {{- else }} + jenkinsTunnel: "{{ template "jenkins.fullname" . }}-agent.{{ template "jenkins.namespace" . }}.svc.{{.Values.clusterZone}}:{{ .Values.controller.agentListenerPort }}" + {{- end }} + {{- else }} + webSocket: true + {{- end }} + {{- end }} + maxRequestsPerHostStr: {{ .Values.agent.maxRequestsPerHostStr | quote }} + retentionTimeout: {{ .Values.agent.retentionTimeout | quote }} + waitForPodSec: {{ .Values.agent.waitForPodSec | quote }} + name: {{ $name | quote }} + namespace: "{{ template "jenkins.agent.namespace" . }}" + restrictedPssSecurityContext: {{ .Values.agent.restrictedPssSecurityContext }} + serverUrl: "{{ .Values.kubernetesURL }}" + credentialsId: "{{ .Values.credentialsId }}" + {{- if .Values.agent.enabled }} + podLabels: + - key: "jenkins/{{ .Release.Name }}-{{ .Values.agent.componentName }}" + value: "true" + {{- range $key, $val := .Values.agent.podLabels }} + - key: {{ $key | quote }} + value: {{ $val | quote }} + {{- end }} + templates: + {{- if not .Values.agent.disableDefaultAgent }} + {{- include "jenkins.casc.podTemplate" . | nindent 8 }} + {{- end }} + {{- if .Values.additionalAgents }} + {{- /* save .Values.agent */}} + {{- $agent := .Values.agent }} + {{- range $name, $additionalAgent := .Values.additionalAgents }} + {{- $additionalContainersEmpty := and (hasKey $additionalAgent "additionalContainers") (empty $additionalAgent.additionalContainers) }} + {{- /* merge original .Values.agent into additional agent to ensure it at least has the default values */}} + {{- $additionalAgent := merge $additionalAgent $agent }} + {{- /* clear list of additional containers in case it is configured empty for this agent (merge might have overwritten that) */}} + {{- if $additionalContainersEmpty }} + {{- $_ := set $additionalAgent "additionalContainers" list }} + {{- end }} + {{- /* set .Values.agent to $additionalAgent */}} + {{- $_ := set $.Values "agent" $additionalAgent }} + {{- include "jenkins.casc.podTemplate" $ | nindent 8 }} + {{- end }} + {{- /* restore .Values.agent */}} + {{- $_ := set .Values "agent" $agent }} + {{- end }} + {{- with .Values.agent.podTemplates }} + {{- range $key, $val := . }} + {{- tpl $val $ | nindent 8 }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- /* restore root */}} + {{- $_ := set $ "Values" $oldRoot.Values }} + {{- end }} + {{- if .Values.controller.csrf.defaultCrumbIssuer.enabled }} + crumbIssuer: + standard: + excludeClientIPFromCrumb: {{ if .Values.controller.csrf.defaultCrumbIssuer.proxyCompatability }}true{{ else }}false{{- end }} + {{- end }} +{{- include "jenkins.casc.security" . }} +{{- with .Values.controller.scriptApproval }} + scriptApproval: + approvedSignatures: + {{- range $key, $val := . }} + - "{{ $val }}" + {{- end }} +{{- end }} +unclassified: + location: + {{- with .Values.controller.jenkinsAdminEmail }} + adminAddress: {{ . }} + {{- end }} + url: {{ template "jenkins.url" . }} +{{- end -}} + +{{/* +Returns a name template to be used for jcasc configmaps, using +suffix passed in at call as index 0 +*/}} +{{- define "jenkins.casc.configName" -}} +{{- $name := index . 0 -}} +{{- $root := index . 1 -}} +"{{- include "jenkins.fullname" $root -}}-jenkins-{{ $name }}" +{{- end -}} + +{{/* +Returns kubernetes pod template configuration as code +*/}} +{{- define "jenkins.casc.podTemplate" -}} +- name: "{{ .Values.agent.podName }}" + namespace: "{{ template "jenkins.agent.namespace" . }}" +{{- if .Values.agent.annotations }} + annotations: + {{- range $key, $value := .Values.agent.annotations }} + - key: {{ $key }} + value: {{ $value | quote }} + {{- end }} +{{- end }} + id: {{ sha256sum (toYaml .Values.agent) }} + containers: + - name: "{{ .Values.agent.sideContainerName }}" + alwaysPullImage: {{ .Values.agent.alwaysPullImage }} + args: "{{ .Values.agent.args | replace "$" "^$" }}" + {{- with .Values.agent.command }} + command: {{ . }} + {{- end }} + envVars: + - envVar: + {{- if .Values.agent.directConnection }} + key: "JENKINS_DIRECT_CONNECTION" + {{- if .Values.agent.jenkinsTunnel }} + value: "{{ tpl .Values.agent.jenkinsTunnel . }}" + {{- else }} + value: "{{ template "jenkins.fullname" . }}-agent.{{ template "jenkins.namespace" . }}.svc.{{.Values.clusterZone}}:{{ .Values.controller.agentListenerPort }}" + {{- end }} + {{- else }} + key: "JENKINS_URL" + {{- if .Values.agent.jenkinsUrl }} + value: {{ tpl .Values.agent.jenkinsUrl . }} + {{- else }} + value: "http://{{ template "jenkins.fullname" . }}.{{ template "jenkins.namespace" . }}.svc.{{.Values.clusterZone}}:{{.Values.controller.servicePort}}{{ default "/" .Values.controller.jenkinsUriPrefix }}" + {{- end }} + {{- end }} + image: "{{ .Values.agent.image.repository }}:{{ .Values.agent.image.tag }}" + {{- if .Values.agent.livenessProbe }} + livenessProbe: + execArgs: {{.Values.agent.livenessProbe.execArgs | quote}} + failureThreshold: {{.Values.agent.livenessProbe.failureThreshold}} + initialDelaySeconds: {{.Values.agent.livenessProbe.initialDelaySeconds}} + periodSeconds: {{.Values.agent.livenessProbe.periodSeconds}} + successThreshold: {{.Values.agent.livenessProbe.successThreshold}} + timeoutSeconds: {{.Values.agent.livenessProbe.timeoutSeconds}} + {{- end }} + privileged: "{{- if .Values.agent.privileged }}true{{- else }}false{{- end }}" + resourceLimitCpu: {{.Values.agent.resources.limits.cpu}} + resourceLimitMemory: {{.Values.agent.resources.limits.memory}} + {{- with .Values.agent.resources.limits.ephemeralStorage }} + resourceLimitEphemeralStorage: {{.}} + {{- end }} + resourceRequestCpu: {{.Values.agent.resources.requests.cpu}} + resourceRequestMemory: {{.Values.agent.resources.requests.memory}} + {{- with .Values.agent.resources.requests.ephemeralStorage }} + resourceRequestEphemeralStorage: {{.}} + {{- end }} + {{- with .Values.agent.runAsUser }} + runAsUser: {{ . }} + {{- end }} + {{- with .Values.agent.runAsGroup }} + runAsGroup: {{ . }} + {{- end }} + ttyEnabled: {{ .Values.agent.TTYEnabled }} + workingDir: {{ .Values.agent.workingDir }} +{{- range $additionalContainers := .Values.agent.additionalContainers }} + - name: "{{ $additionalContainers.sideContainerName }}" + alwaysPullImage: {{ $additionalContainers.alwaysPullImage | default $.Values.agent.alwaysPullImage }} + args: "{{ $additionalContainers.args | replace "$" "^$" }}" + {{- with $additionalContainers.command }} + command: {{ . }} + {{- end }} + envVars: + - envVar: + key: "JENKINS_URL" + {{- if $additionalContainers.jenkinsUrl }} + value: {{ tpl ($additionalContainers.jenkinsUrl) . }} + {{- else }} + value: "http://{{ template "jenkins.fullname" $ }}.{{ template "jenkins.namespace" $ }}.svc.{{ $.Values.clusterZone }}:{{ $.Values.controller.servicePort }}{{ default "/" $.Values.controller.jenkinsUriPrefix }}" + {{- end }} + image: "{{ $additionalContainers.image.repository }}:{{ $additionalContainers.image.tag }}" + {{- if $additionalContainers.livenessProbe }} + livenessProbe: + execArgs: {{$additionalContainers.livenessProbe.execArgs | quote}} + failureThreshold: {{$additionalContainers.livenessProbe.failureThreshold}} + initialDelaySeconds: {{$additionalContainers.livenessProbe.initialDelaySeconds}} + periodSeconds: {{$additionalContainers.livenessProbe.periodSeconds}} + successThreshold: {{$additionalContainers.livenessProbe.successThreshold}} + timeoutSeconds: {{$additionalContainers.livenessProbe.timeoutSeconds}} + {{- end }} + privileged: "{{- if $additionalContainers.privileged }}true{{- else }}false{{- end }}" + resourceLimitCpu: {{ if $additionalContainers.resources }}{{ $additionalContainers.resources.limits.cpu }}{{ else }}{{ $.Values.agent.resources.limits.cpu }}{{ end }} + resourceLimitMemory: {{ if $additionalContainers.resources }}{{ $additionalContainers.resources.limits.memory }}{{ else }}{{ $.Values.agent.resources.limits.memory }}{{ end }} + resourceRequestCpu: {{ if $additionalContainers.resources }}{{ $additionalContainers.resources.requests.cpu }}{{ else }}{{ $.Values.agent.resources.requests.cpu }}{{ end }} + resourceRequestMemory: {{ if $additionalContainers.resources }}{{ $additionalContainers.resources.requests.memory }}{{ else }}{{ $.Values.agent.resources.requests.memory }}{{ end }} + {{- if or $additionalContainers.runAsUser $.Values.agent.runAsUser }} + runAsUser: {{ $additionalContainers.runAsUser | default $.Values.agent.runAsUser }} + {{- end }} + {{- if or $additionalContainers.runAsGroup $.Values.agent.runAsGroup }} + runAsGroup: {{ $additionalContainers.runAsGroup | default $.Values.agent.runAsGroup }} + {{- end }} + ttyEnabled: {{ $additionalContainers.TTYEnabled | default $.Values.agent.TTYEnabled }} + workingDir: {{ $additionalContainers.workingDir | default $.Values.agent.workingDir }} +{{- end }} +{{- if or .Values.agent.envVars .Values.agent.secretEnvVars }} + envVars: + {{- range $index, $var := .Values.agent.envVars }} + - envVar: + key: {{ $var.name }} + value: {{ tpl $var.value $ }} + {{- end }} + {{- range $index, $var := .Values.agent.secretEnvVars }} + - secretEnvVar: + key: {{ $var.key }} + secretName: {{ $var.secretName }} + secretKey: {{ $var.secretKey }} + optional: {{ $var.optional | default false }} + {{- end }} +{{- end }} + idleMinutes: {{ .Values.agent.idleMinutes }} + instanceCap: 2147483647 + {{- if .Values.agent.hostNetworking }} + hostNetwork: {{ .Values.agent.hostNetworking }} + {{- end }} + {{- if .Values.agent.imagePullSecretName }} + imagePullSecrets: + - name: {{ .Values.agent.imagePullSecretName }} + {{- end }} + label: "{{ .Release.Name }}-{{ .Values.agent.componentName }} {{ .Values.agent.customJenkinsLabels | join " " }}" +{{- if .Values.agent.nodeSelector }} + nodeSelector: + {{- $local := dict "first" true }} + {{- range $key, $value := .Values.agent.nodeSelector }} + {{- if $local.first }} {{ else }},{{ end }} + {{- $key }}={{ tpl $value $ }} + {{- $_ := set $local "first" false }} + {{- end }} +{{- end }} + nodeUsageMode: {{ quote .Values.agent.nodeUsageMode }} + podRetention: {{ .Values.agent.podRetention }} + showRawYaml: {{ .Values.agent.showRawYaml }} + serviceAccount: "{{ include "jenkins.serviceAccountAgentName" . }}" + slaveConnectTimeoutStr: "{{ .Values.agent.connectTimeout }}" +{{- if .Values.agent.volumes }} + volumes: + {{- range $index, $volume := .Values.agent.volumes }} + -{{- if (eq $volume.type "ConfigMap") }} configMapVolume: + {{- else if (eq $volume.type "EmptyDir") }} emptyDirVolume: + {{- else if (eq $volume.type "EphemeralVolume") }} genericEphemeralVolume: + {{- else if (eq $volume.type "HostPath") }} hostPathVolume: + {{- else if (eq $volume.type "Nfs") }} nfsVolume: + {{- else if (eq $volume.type "PVC") }} persistentVolumeClaim: + {{- else if (eq $volume.type "Secret") }} secretVolume: + {{- else }} {{ $volume.type }}: + {{- end }} + {{- range $key, $value := $volume }} + {{- if not (eq $key "type") }} + {{ $key }}: {{ if kindIs "string" $value }}{{ tpl $value $ | quote }}{{ else }}{{ $value }}{{ end }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} +{{- if .Values.agent.workspaceVolume }} + workspaceVolume: + {{- if (eq .Values.agent.workspaceVolume.type "DynamicPVC") }} + dynamicPVC: + {{- else if (eq .Values.agent.workspaceVolume.type "EmptyDir") }} + emptyDirWorkspaceVolume: + {{- else if (eq .Values.agent.workspaceVolume.type "EphemeralVolume") }} + genericEphemeralVolume: + {{- else if (eq .Values.agent.workspaceVolume.type "HostPath") }} + hostPathWorkspaceVolume: + {{- else if (eq .Values.agent.workspaceVolume.type "Nfs") }} + nfsWorkspaceVolume: + {{- else if (eq .Values.agent.workspaceVolume.type "PVC") }} + persistentVolumeClaimWorkspaceVolume: + {{- else }} + {{ .Values.agent.workspaceVolume.type }}: + {{- end }} + {{- range $key, $value := .Values.agent.workspaceVolume }} + {{- if not (eq $key "type") }} + {{ $key }}: {{ if kindIs "string" $value }}{{ tpl $value $ | quote }}{{ else }}{{ $value }}{{ end }} + {{- end }} + {{- end }} +{{- end }} +{{- if .Values.agent.yamlTemplate }} + yaml: |- + {{- tpl (trim .Values.agent.yamlTemplate) . | nindent 4 }} +{{- end }} + yamlMergeStrategy: {{ .Values.agent.yamlMergeStrategy }} + inheritYamlMergeStrategy: {{ .Values.agent.inheritYamlMergeStrategy }} +{{- end -}} + +{{- define "jenkins.kubernetes-version" -}} + {{- if .Values.controller.installPlugins -}} + {{- range .Values.controller.installPlugins -}} + {{- if hasPrefix "kubernetes:" . }} + {{- $split := splitList ":" . }} + {{- printf "%s" (index $split 1 ) -}} + {{- end -}} + {{- end -}} + {{- end -}} +{{- end -}} + +{{- define "jenkins.casc.security" }} +security: +{{- with .Values.controller.JCasC }} +{{- if .security }} + {{- .security | toYaml | nindent 2 }} +{{- end }} +{{- end }} +{{- end -}} + +{{/* +Create the name of the service account to use +*/}} +{{- define "jenkins.serviceAccountName" -}} +{{- if .Values.serviceAccount.create -}} + {{ default (include "jenkins.fullname" .) .Values.serviceAccount.name }} +{{- else -}} + {{ default "default" .Values.serviceAccount.name }} +{{- end -}} +{{- end -}} + +{{/* +Create the name of the service account for Jenkins agents to use +*/}} +{{- define "jenkins.serviceAccountAgentName" -}} +{{- if .Values.serviceAccountAgent.create -}} + {{ default (printf "%s-%s" (include "jenkins.fullname" .) "agent") .Values.serviceAccountAgent.name }} +{{- else -}} + {{ default "default" .Values.serviceAccountAgent.name }} +{{- end -}} +{{- end -}} + +{{/* +Create a full tag name for controller image +*/}} +{{- define "controller.image.tag" -}} +{{- if .Values.controller.image.tagLabel -}} + {{- default (printf "%s-%s" .Chart.AppVersion .Values.controller.image.tagLabel) .Values.controller.image.tag -}} +{{- else -}} + {{- default .Chart.AppVersion .Values.controller.image.tag -}} +{{- end -}} +{{- end -}} + +{{/* +Create the HTTP port for interacting with the controller +*/}} +{{- define "controller.httpPort" -}} +{{- if .Values.controller.httpsKeyStore.enable -}} + {{- .Values.controller.httpsKeyStore.httpPort -}} +{{- else -}} + {{- .Values.controller.targetPort -}} +{{- end -}} +{{- end -}} + +{{- define "jenkins.configReloadContainer" -}} +{{- $root := index . 0 -}} +{{- $containerName := index . 1 -}} +{{- $containerType := index . 2 -}} +- name: {{ $containerName }} + image: "{{ $root.Values.controller.sidecars.configAutoReload.image.registry }}/{{ $root.Values.controller.sidecars.configAutoReload.image.repository }}:{{ $root.Values.controller.sidecars.configAutoReload.image.tag }}" + imagePullPolicy: {{ $root.Values.controller.sidecars.configAutoReload.imagePullPolicy }} + {{- if $root.Values.controller.sidecars.configAutoReload.containerSecurityContext }} + securityContext: {{- toYaml $root.Values.controller.sidecars.configAutoReload.containerSecurityContext | nindent 4 }} + {{- end }} + {{- if $root.Values.controller.sidecars.configAutoReload.envFrom }} + envFrom: +{{ (tpl (toYaml $root.Values.controller.sidecars.configAutoReload.envFrom) $root) | indent 4 }} + {{- end }} + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: LABEL + value: "{{ template "jenkins.fullname" $root }}-jenkins-config" + - name: FOLDER + value: "{{ $root.Values.controller.sidecars.configAutoReload.folder }}" + - name: NAMESPACE + value: '{{ $root.Values.controller.sidecars.configAutoReload.searchNamespace | default (include "jenkins.namespace" $root) }}' + {{- if eq $containerType "init" }} + - name: METHOD + value: "LIST" + {{- else if $root.Values.controller.sidecars.configAutoReload.sleepTime }} + - name: METHOD + value: "SLEEP" + - name: SLEEP_TIME + value: "{{ $root.Values.controller.sidecars.configAutoReload.sleepTime }}" + {{- end }} + {{- if eq $containerType "sidecar" }} + - name: REQ_URL + value: "{{- default "http" $root.Values.controller.sidecars.configAutoReload.scheme }}://localhost:{{- include "controller.httpPort" $root -}}{{- $root.Values.controller.jenkinsUriPrefix -}}/reload-configuration-as-code/?casc-reload-token=$(POD_NAME)" + - name: REQ_METHOD + value: "POST" + - name: REQ_RETRY_CONNECT + value: "{{ $root.Values.controller.sidecars.configAutoReload.reqRetryConnect }}" + {{- if $root.Values.controller.sidecars.configAutoReload.skipTlsVerify }} + - name: REQ_SKIP_TLS_VERIFY + value: "true" + {{- end }} + {{- end }} + + {{- if $root.Values.controller.sidecars.configAutoReload.env }} + {{- range $envVarItem := $root.Values.controller.sidecars.configAutoReload.env -}} + {{- if or (ne $containerType "init") (ne .name "METHOD") }} +{{- (tpl (toYaml (list $envVarItem)) $root) | nindent 4 }} + {{- end -}} + {{- end -}} + {{- end }} + {{- if $root.Values.controller.sidecars.configAutoReload.logging.configuration.override }} + - name: LOG_CONFIG + value: "{{ $root.Values.controller.jenkinsHome }}/auto-reload/auto-reload-config.yaml" + {{- end }} + + resources: +{{ toYaml $root.Values.controller.sidecars.configAutoReload.resources | indent 4 }} + volumeMounts: + - name: sc-config-volume + mountPath: {{ $root.Values.controller.sidecars.configAutoReload.folder | quote }} + - name: jenkins-home + mountPath: {{ $root.Values.controller.jenkinsHome }} + {{- if $root.Values.persistence.subPath }} + subPath: {{ $root.Values.persistence.subPath }} + {{- end }} + {{- if $root.Values.controller.sidecars.configAutoReload.logging.configuration.override }} + - name: auto-reload-config + mountPath: {{ $root.Values.controller.jenkinsHome }}/auto-reload + - name: auto-reload-config-logs + mountPath: {{ $root.Values.controller.jenkinsHome }}/auto-reload-logs + {{- end }} + {{- if $root.Values.controller.sidecars.configAutoReload.additionalVolumeMounts }} +{{ (tpl (toYaml $root.Values.controller.sidecars.configAutoReload.additionalVolumeMounts) $root) | indent 4 }} + {{- end }} + +{{- end -}} diff --git a/charts/jenkins/jenkins/5.4.2/templates/auto-reload-config.yaml b/charts/jenkins/jenkins/5.4.2/templates/auto-reload-config.yaml new file mode 100644 index 000000000..8c177d7f3 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/auto-reload-config.yaml @@ -0,0 +1,60 @@ +{{- if .Values.controller.sidecars.configAutoReload.logging.configuration.override }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "jenkins.fullname" . }}-auto-reload-config + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": {{ template "jenkins.name" . }} + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ .Chart.Name }}-{{ .Chart.Version }}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ $.Release.Service }}" + "app.kubernetes.io/instance": "{{ $.Release.Name }}" + "app.kubernetes.io/component": "{{ $.Values.controller.componentName }}" +data: + auto-reload-config.yaml: |- + version: 1 + disable_existing_loggers: false + root: + level: {{ .Values.controller.sidecars.configAutoReload.logging.configuration.logLevel }} + handlers: + {{- if .Values.controller.sidecars.configAutoReload.logging.configuration.logToConsole}} + - console + {{- end }} + {{- if .Values.controller.sidecars.configAutoReload.logging.configuration.logToFile }} + - file + {{- end }} + handlers: + {{- if .Values.controller.sidecars.configAutoReload.logging.configuration.logToConsole}} + console: + class: logging.StreamHandler + level: {{ .Values.controller.sidecars.configAutoReload.logging.configuration.logLevel }} + formatter: {{ .Values.controller.sidecars.configAutoReload.logging.configuration.formatter }} + {{- end }} + {{- if .Values.controller.sidecars.configAutoReload.logging.configuration.logToFile }} + file: + class : logging.handlers.RotatingFileHandler + formatter: {{ .Values.controller.sidecars.configAutoReload.logging.configuration.formatter }} + filename: {{ .Values.controller.jenkinsHome }}/auto-reload-logs/file.log + maxBytes: {{ .Values.controller.sidecars.configAutoReload.logging.configuration.maxBytes }} + backupCount: {{ .Values.controller.sidecars.configAutoReload.logging.configuration.backupCount }} + {{- end }} + formatters: + JSON: + "()": logger.JsonFormatter + format: "%(levelname)s %(message)s" + rename_fields: + message: msg + levelname: level + LOGFMT: + "()": logger.LogfmtFormatter + keys: + - time + - level + - msg + mapping: + time: asctime + level: levelname + msg: message + {{- end }} \ No newline at end of file diff --git a/charts/jenkins/jenkins/5.4.2/templates/config-init-scripts.yaml b/charts/jenkins/jenkins/5.4.2/templates/config-init-scripts.yaml new file mode 100644 index 000000000..7dd253cc3 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/config-init-scripts.yaml @@ -0,0 +1,18 @@ +{{- if .Values.controller.initScripts -}} + +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "jenkins.fullname" . }}-init-scripts + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +data: +{{- range $key, $val := .Values.controller.initScripts }} + init{{ $key }}.groovy: |- +{{ tpl $val $ | indent 4 }} +{{- end }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/config.yaml b/charts/jenkins/jenkins/5.4.2/templates/config.yaml new file mode 100644 index 000000000..5de0b9f72 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/config.yaml @@ -0,0 +1,92 @@ +{{- $jenkinsHome := .Values.controller.jenkinsHome -}} + +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "jenkins.fullname" . }} + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +data: + apply_config.sh: |- + set -e +{{- if .Values.controller.initializeOnce }} + if [ -f {{ .Values.controller.jenkinsHome }}/initialization-completed ]; then + echo "controller was previously initialized, refusing to re-initialize" + exit 0 + fi +{{- end }} + echo "disable Setup Wizard" + # Prevent Setup Wizard when JCasC is enabled + echo $JENKINS_VERSION > {{ .Values.controller.jenkinsHome }}/jenkins.install.UpgradeWizard.state + echo $JENKINS_VERSION > {{ .Values.controller.jenkinsHome }}/jenkins.install.InstallUtil.lastExecVersion +{{- if .Values.controller.overwritePlugins }} + echo "remove all plugins from shared volume" + # remove all plugins from shared volume + rm -rf {{ .Values.controller.jenkinsHome }}/plugins/* +{{- end }} +{{- if .Values.controller.JCasC.overwriteConfiguration }} + echo "deleting all XML config files" + rm -f {{ .Values.controller.jenkinsHome }}/config.xml + rm -f {{ .Values.controller.jenkinsHome }}/*plugins*.xml + find {{ .Values.controller.jenkinsHome }} -maxdepth 1 -type f -iname '*configuration*.xml' -exec rm -f {} \; +{{- end }} +{{- if .Values.controller.installPlugins }} + echo "download plugins" + # Install missing plugins + cp /var/jenkins_config/plugins.txt {{ .Values.controller.jenkinsHome }}; + rm -rf {{ .Values.controller.jenkinsRef }}/plugins/*.lock + version () { echo "$@" | awk -F. '{ printf("%d%03d%03d%03d\n", $1,$2,$3,$4); }'; } + if [ -f "{{ .Values.controller.jenkinsWar }}" ] && [ -n "$(command -v jenkins-plugin-cli)" 2>/dev/null ] && [ $(version $(jenkins-plugin-cli --version)) -ge $(version "2.1.1") ]; then + jenkins-plugin-cli --verbose --war "{{ .Values.controller.jenkinsWar }}" --plugin-file "{{ .Values.controller.jenkinsHome }}/plugins.txt" --latest {{ .Values.controller.installLatestPlugins }}{{- if .Values.controller.installLatestSpecifiedPlugins }} --latest-specified{{- end }}; + else + /usr/local/bin/install-plugins.sh `echo $(cat {{ .Values.controller.jenkinsHome }}/plugins.txt)`; + fi + echo "copy plugins to shared volume" + # Copy plugins to shared volume + yes n | cp -i {{ .Values.controller.jenkinsRef }}/plugins/* /var/jenkins_plugins/; +{{- end }} + {{- if not .Values.controller.sidecars.configAutoReload.enabled }} + echo "copy configuration as code files" + mkdir -p {{ .Values.controller.jenkinsHome }}/casc_configs; + rm -rf {{ .Values.controller.jenkinsHome }}/casc_configs/* + {{- if or .Values.controller.JCasC.defaultConfig .Values.controller.JCasC.configScripts }} + cp -v /var/jenkins_config/*.yaml {{ .Values.controller.jenkinsHome }}/casc_configs + {{- end }} + {{- end }} + echo "finished initialization" +{{- if .Values.controller.initializeOnce }} + touch {{ .Values.controller.jenkinsHome }}/initialization-completed +{{- end }} + {{- if not .Values.controller.sidecars.configAutoReload.enabled }} +# Only add config to this script if we aren't auto-reloading otherwise the pod will restart upon each config change: +{{- if .Values.controller.JCasC.defaultConfig }} + jcasc-default-config.yaml: |- + {{- include "jenkins.casc.defaults" . |nindent 4}} +{{- end }} +{{- range $key, $val := .Values.controller.JCasC.configScripts }} + {{ $key }}.yaml: |- +{{ tpl $val $| indent 4 }} +{{- end }} +{{- end }} + plugins.txt: |- +{{- if .Values.controller.installPlugins }} + {{- range $installPlugin := .Values.controller.installPlugins }} + {{- $installPlugin | nindent 4 }} + {{- end }} + {{- range $addlPlugin := .Values.controller.additionalPlugins }} + {{- /* duplicate plugin check */}} + {{- range $installPlugin := $.Values.controller.installPlugins }} + {{- if eq (splitList ":" $addlPlugin | first) (splitList ":" $installPlugin | first) }} + {{- $message := print "[PLUGIN CONFLICT] controller.additionalPlugins contains '" $addlPlugin "'" }} + {{- $message := print $message " but controller.installPlugins already contains '" $installPlugin "'." }} + {{- $message := print $message " Override controller.installPlugins to use '" $addlPlugin "' plugin." }} + {{- fail $message }} + {{- end }} + {{- end }} + {{- $addlPlugin | nindent 4 }} + {{- end }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/deprecation.yaml b/charts/jenkins/jenkins/5.4.2/templates/deprecation.yaml new file mode 100644 index 000000000..f54017ce4 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/deprecation.yaml @@ -0,0 +1,151 @@ +{{- if .Values.checkDeprecation }} + {{- if .Values.master }} + {{ fail "`master` does no longer exist. It has been renamed to `controller`" }} + {{- end }} + + {{- if .Values.controller.imageTag }} + {{ fail "`controller.imageTag` does no longer exist. Please use `controller.image.tag` instead" }} + {{- end }} + + {{- if .Values.controller.slaveListenerPort }} + {{ fail "`controller.slaveListenerPort` does no longer exist. It has been renamed to `controller.agentListenerPort`" }} + {{- end }} + + {{- if .Values.controller.slaveHostPort }} + {{ fail "`controller.slaveHostPort` does no longer exist. It has been renamed to `controller.agentListenerHostPort`" }} + {{- end }} + + {{- if .Values.controller.slaveKubernetesNamespace }} + {{ fail "`controller.slaveKubernetesNamespace` does no longer exist. It has been renamed to `agent.namespace`" }} + {{- end }} + + {{- if .Values.controller.slaveDefaultsProviderTemplate }} + {{ fail "`controller.slaveDefaultsProviderTemplate` does no longer exist. It has been renamed to `agent.defaultsProviderTemplate`" }} + {{- end }} + + {{- if .Values.controller.useSecurity }} + {{ fail "`controller.useSecurity` does no longer exist. It has been renamed to `controller.adminSecret`" }} + {{- end }} + + {{- if .Values.controller.slaveJenkinsUrl }} + {{ fail "`controller.slaveJenkinsUrl` does no longer exist. It has been renamed to `agent.jenkinsUrl`" }} + {{- end }} + + {{- if .Values.controller.slaveJenkinsTunnel }} + {{ fail "`controller.slaveJenkinsTunnel` does no longer exist. It has been renamed to `agent.jenkinsTunnel`" }} + {{- end }} + + {{- if .Values.controller.slaveConnectTimeout }} + {{ fail "`controller.slaveConnectTimeout` does no longer exist. It has been renamed to `agent.kubernetesConnectTimeout`" }} + {{- end }} + + {{- if .Values.controller.slaveReadTimeout }} + {{ fail "`controller.slaveReadTimeout` does no longer exist. It has been renamed to `agent.kubernetesReadTimeout`" }} + {{- end }} + + {{- if .Values.controller.slaveListenerServiceType }} + {{ fail "`controller.slaveListenerServiceType` does no longer exist. It has been renamed to `controller.agentListenerServiceType`" }} + {{- end }} + + {{- if .Values.controller.slaveListenerLoadBalancerIP }} + {{ fail "`controller.slaveListenerLoadBalancerIP` does no longer exist. It has been renamed to `controller.agentListenerLoadBalancerIP`" }} + {{- end }} + + {{- if .Values.controller.slaveListenerServiceAnnotations }} + {{ fail "`controller.slaveListenerServiceAnnotations` does no longer exist. It has been renamed to `controller.agentListenerServiceAnnotations`" }} + {{- end }} + + {{- if .Values.agent.slaveConnectTimeout }} + {{ fail "`agent.slaveConnectTimeout` does no longer exist. It has been renamed to `agent.connectTimeout`" }} + {{- end }} + + {{- if .Values.NetworkPolicy }} + + {{- if .Values.NetworkPolicy.Enabled }} + {{ fail "`NetworkPolicy.Enabled` does no longer exist. It has been renamed to `networkPolicy.enabled`" }} + {{- end }} + + {{- if .Values.NetworkPolicy.ApiVersion }} + {{ fail "`NetworkPolicy.ApiVersion` does no longer exist. It has been renamed to `networkPolicy.apiVersion`" }} + {{- end }} + + {{ fail "NetworkPolicy.* values have been renamed, please check the documentation" }} + {{- end }} + + + {{- if .Values.rbac.install }} + {{ fail "`rbac.install` does no longer exist. It has been renamed to `rbac.create` and is enabled by default!" }} + {{- end }} + + {{- if .Values.rbac.serviceAccountName }} + {{ fail "`rbac.serviceAccountName` does no longer exist. It has been renamed to `serviceAccount.name`" }} + {{- end }} + + {{- if .Values.rbac.serviceAccountAnnotations }} + {{ fail "`rbac.serviceAccountAnnotations` does no longer exist. It has been renamed to `serviceAccount.annotations`" }} + {{- end }} + + {{- if .Values.rbac.roleRef }} + {{ fail "`rbac.roleRef` does no longer exist. RBAC roles are now generated, please check the documentation" }} + {{- end }} + + {{- if .Values.rbac.roleKind }} + {{ fail "`rbac.roleKind` does no longer exist. RBAC roles are now generated, please check the documentation" }} + {{- end }} + + {{- if .Values.rbac.roleBindingKind }} + {{ fail "`rbac.roleBindingKind` does no longer exist. RBAC roles are now generated, please check the documentation" }} + {{- end }} + + {{- if .Values.controller.JCasC.pluginVersion }} + {{ fail "controller.JCasC.pluginVersion has been deprecated, please use controller.installPlugins instead" }} + {{- end }} + + {{- if .Values.controller.deploymentLabels }} + {{ fail "`controller.deploymentLabels` does no longer exist. It has been renamed to `controller.statefulSetLabels`" }} + {{- end }} + + {{- if .Values.controller.deploymentAnnotations }} + {{ fail "`controller.deploymentAnnotations` does no longer exist. It has been renamed to `controller.statefulSetAnnotations`" }} + {{- end }} + + {{- if .Values.controller.rollingUpdate }} + {{ fail "`controller.rollingUpdate` does no longer exist. It is no longer relevant, since a StatefulSet is used for the Jenkins controller" }} + {{- end }} + + {{- if .Values.controller.tag }} + {{ fail "`controller.tag` no longer exists. It has been renamed to `controller.image.tag'" }} + {{- end }} + + {{- if .Values.controller.tagLabel }} + {{ fail "`controller.tagLabel` no longer exists. It has been renamed to `controller.image.tagLabel`" }} + {{- end }} + + {{- if .Values.controller.adminSecret }} + {{ fail "`controller.adminSecret` no longer exists. It has been renamed to `controller.admin.createSecret`" }} + {{- end }} + + {{- if .Values.controller.adminUser }} + {{ fail "`controller.adminUser` no longer exists. It has been renamed to `controller.admin.username`" }} + {{- end }} + + {{- if .Values.controller.adminPassword }} + {{ fail "`controller.adminPassword` no longer exists. It has been renamed to `controller.admin.password`" }} + {{- end }} + + {{- if .Values.controller.sidecars.other }} + {{ fail "`controller.sidecars.other` no longer exists. It has been renamed to `controller.sidecars.additionalSidecarContainers`" }} + {{- end }} + + {{- if .Values.agent.tag }} + {{ fail "`controller.agent.tag` no longer exists. It has been renamed to `controller.agent.image.tag`" }} + {{- end }} + + {{- if .Values.backup }} + {{ fail "`controller.backup` no longer exists." }} + {{- end }} + + {{- if .Values.helmtest.bats.tag }} + {{ fail "`helmtest.bats.tag` no longer exists. It has been renamed to `helmtest.bats.image.tag`" }} + {{- end }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/home-pvc.yaml b/charts/jenkins/jenkins/5.4.2/templates/home-pvc.yaml new file mode 100644 index 000000000..f417d23ad --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/home-pvc.yaml @@ -0,0 +1,41 @@ +{{- if not (contains "jenkins-home" (quote .Values.persistence.volumes)) }} +{{- if and .Values.persistence.enabled (not .Values.persistence.existingClaim) -}} +kind: PersistentVolumeClaim +apiVersion: v1 +metadata: +{{- if .Values.persistence.annotations }} + annotations: +{{ toYaml .Values.persistence.annotations | indent 4 }} +{{- end }} + name: {{ template "jenkins.fullname" . }} + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- if .Values.persistence.labels }} +{{ toYaml .Values.persistence.labels | indent 4 }} +{{- end }} +spec: +{{- if .Values.persistence.dataSource }} + dataSource: +{{ toYaml .Values.persistence.dataSource | indent 4 }} +{{- end }} + accessModes: + - {{ .Values.persistence.accessMode | quote }} + resources: + requests: + storage: {{ .Values.persistence.size | quote }} +{{- if .Values.persistence.storageClass }} +{{- if (eq "-" .Values.persistence.storageClass) }} + storageClassName: "" +{{- else }} + storageClassName: "{{ .Values.persistence.storageClass }}" +{{- end }} +{{- end }} +{{- end }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jcasc-config.yaml b/charts/jenkins/jenkins/5.4.2/templates/jcasc-config.yaml new file mode 100644 index 000000000..f51444525 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jcasc-config.yaml @@ -0,0 +1,53 @@ +{{- $root := . }} +{{- if .Values.controller.sidecars.configAutoReload.enabled }} +{{- range $key, $val := .Values.controller.JCasC.configScripts }} +{{- if $val }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "jenkins.casc.configName" (list (printf "config-%s" $key) $ )}} + namespace: {{ template "jenkins.namespace" $root }} + labels: + "app.kubernetes.io/name": {{ template "jenkins.name" $root}} + {{- if $root.Values.renderHelmLabels }} + "helm.sh/chart": "{{ $root.Chart.Name }}-{{ $root.Chart.Version }}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ $.Release.Service }}" + "app.kubernetes.io/instance": "{{ $.Release.Name }}" + "app.kubernetes.io/component": "{{ $.Values.controller.componentName }}" + {{ template "jenkins.fullname" $root }}-jenkins-config: "true" +{{- if $root.Values.controller.JCasC.configMapAnnotations }} + annotations: +{{ toYaml $root.Values.controller.JCasC.configMapAnnotations | indent 4 }} +{{- end }} +data: + {{ $key }}.yaml: |- +{{ tpl $val $| indent 4 }} +{{- end }} +{{- end }} +{{- if .Values.controller.JCasC.defaultConfig }} +--- +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "jenkins.casc.configName" (list "jcasc-config" $ )}} + namespace: {{ template "jenkins.namespace" $root }} + labels: + "app.kubernetes.io/name": {{ template "jenkins.name" $root}} + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ $root.Chart.Name }}-{{ $root.Chart.Version }}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ $.Release.Service }}" + "app.kubernetes.io/instance": "{{ $.Release.Name }}" + "app.kubernetes.io/component": "{{ $.Values.controller.componentName }}" + {{ template "jenkins.fullname" $root }}-jenkins-config: "true" +{{- if $root.Values.controller.JCasC.configMapAnnotations }} + annotations: +{{ toYaml $root.Values.controller.JCasC.configMapAnnotations | indent 4 }} +{{- end }} +data: + jcasc-default-config.yaml: |- + {{- include "jenkins.casc.defaults" . | nindent 4 }} +{{- end}} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-agent-svc.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-agent-svc.yaml new file mode 100644 index 000000000..4440b91f8 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-agent-svc.yaml @@ -0,0 +1,43 @@ +{{- if .Values.controller.agentListenerEnabled }} +apiVersion: v1 +kind: Service +metadata: + name: {{ template "jenkins.fullname" . }}-agent + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- if .Values.controller.agentListenerServiceAnnotations }} + annotations: + {{- toYaml .Values.controller.agentListenerServiceAnnotations | nindent 4 }} + {{- end }} +spec: + {{- if .Values.controller.agentListenerExternalTrafficPolicy }} + externalTrafficPolicy: {{.Values.controller.agentListenerExternalTrafficPolicy}} + {{- end }} + ports: + - port: {{ .Values.controller.agentListenerPort }} + targetPort: {{ .Values.controller.agentListenerPort }} + {{- if (and (eq .Values.controller.agentListenerServiceType "NodePort") (not (empty .Values.controller.agentListenerNodePort))) }} + nodePort: {{ .Values.controller.agentListenerNodePort }} + {{- end }} + name: agent-listener + selector: + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + type: {{ .Values.controller.agentListenerServiceType }} + {{if eq .Values.controller.agentListenerServiceType "LoadBalancer"}} +{{- if .Values.controller.agentListenerLoadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{ toYaml .Values.controller.agentListenerLoadBalancerSourceRanges | indent 4 }} +{{- end }} + {{- end }} + {{- if and (eq .Values.controller.agentListenerServiceType "LoadBalancer") (.Values.controller.agentListenerLoadBalancerIP) }} + loadBalancerIP: {{ .Values.controller.agentListenerLoadBalancerIP }} + {{- end }} + {{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-aws-security-group-policies.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-aws-security-group-policies.yaml new file mode 100644 index 000000000..2f6e7a13d --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-aws-security-group-policies.yaml @@ -0,0 +1,16 @@ +{{- if .Values.awsSecurityGroupPolicies.enabled -}} +{{- range .Values.awsSecurityGroupPolicies.policies -}} +apiVersion: vpcresources.k8s.aws/v1beta1 +kind: SecurityGroupPolicy +metadata: + name: {{ .name }} + namespace: {{ template "jenkins.namespace" $ }} +spec: + podSelector: + {{- toYaml .podSelector | nindent 6}} + securityGroups: + groupIds: + {{- toYaml .securityGroupIds | nindent 6}} +--- +{{- end -}} +{{- end -}} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-alerting-rules.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-alerting-rules.yaml new file mode 100644 index 000000000..3fd806172 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-alerting-rules.yaml @@ -0,0 +1,26 @@ +{{- if and .Values.controller.prometheus.enabled .Values.controller.prometheus.alertingrules }} +--- +apiVersion: monitoring.coreos.com/v1 +kind: PrometheusRule +metadata: + name: {{ template "jenkins.fullname" . }} +{{- if .Values.controller.prometheus.prometheusRuleNamespace }} + namespace: {{ .Values.controller.prometheus.prometheusRuleNamespace }} +{{- else }} + namespace: {{ template "jenkins.namespace" . }} +{{- end }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- range $key, $val := .Values.controller.prometheus.alertingRulesAdditionalLabels }} + {{ $key }}: {{ $val | quote }} + {{- end}} +spec: + groups: +{{ toYaml .Values.controller.prometheus.alertingrules | indent 2 }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-backendconfig.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-backendconfig.yaml new file mode 100644 index 000000000..0e8a566fc --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-backendconfig.yaml @@ -0,0 +1,24 @@ +{{- if .Values.controller.backendconfig.enabled }} +apiVersion: {{ .Values.controller.backendconfig.apiVersion }} +kind: BackendConfig +metadata: + name: {{ .Values.controller.backendconfig.name }} + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- if .Values.controller.backendconfig.labels }} +{{ toYaml .Values.controller.backendconfig.labels | indent 4 }} +{{- end }} +{{- if .Values.controller.backendconfig.annotations }} + annotations: +{{ toYaml .Values.controller.backendconfig.annotations | indent 4 }} +{{- end }} +spec: +{{ toYaml .Values.controller.backendconfig.spec | indent 2 }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-ingress.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-ingress.yaml new file mode 100644 index 000000000..b3b344ff8 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-ingress.yaml @@ -0,0 +1,77 @@ +{{- $kubeTargetVersion := default .Capabilities.KubeVersion.GitVersion .Values.kubeTargetVersionOverride }} +{{- if .Values.controller.ingress.enabled }} +{{- if semverCompare ">=1.19-0" $kubeTargetVersion -}} +apiVersion: networking.k8s.io/v1 +{{- else if semverCompare ">=1.14-0" $kubeTargetVersion -}} +apiVersion: networking.k8s.io/v1beta1 +{{- else -}} +apiVersion: {{ .Values.controller.ingress.apiVersion }} +{{- end }} +kind: Ingress +metadata: + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- if .Values.controller.ingress.labels }} +{{ toYaml .Values.controller.ingress.labels | indent 4 }} +{{- end }} +{{- if .Values.controller.ingress.annotations }} + annotations: +{{ toYaml .Values.controller.ingress.annotations | indent 4 }} +{{- end }} + name: {{ template "jenkins.fullname" . }} +spec: +{{- if .Values.controller.ingress.ingressClassName }} + ingressClassName: {{ .Values.controller.ingress.ingressClassName | quote }} +{{- end }} + rules: + - http: + paths: +{{- if empty (.Values.controller.ingress.paths) }} + - backend: +{{- if semverCompare ">=1.19-0" $kubeTargetVersion }} + service: + name: {{ template "jenkins.fullname" . }} + port: + number: {{ .Values.controller.servicePort }} + pathType: ImplementationSpecific +{{- else }} + serviceName: {{ template "jenkins.fullname" . }} + servicePort: {{ .Values.controller.servicePort }} +{{- end }} +{{- if .Values.controller.ingress.path }} + path: {{ .Values.controller.ingress.path }} +{{- end -}} +{{- else }} +{{ tpl (toYaml .Values.controller.ingress.paths | indent 6) . }} +{{- end -}} +{{- if .Values.controller.ingress.hostName }} + host: {{ tpl .Values.controller.ingress.hostName . | quote }} +{{- end }} +{{- if .Values.controller.ingress.resourceRootUrl }} + - http: + paths: + - backend: +{{- if semverCompare ">=1.19-0" $kubeTargetVersion }} + service: + name: {{ template "jenkins.fullname" . }} + port: + number: {{ .Values.controller.servicePort }} + pathType: ImplementationSpecific +{{- else }} + serviceName: {{ template "jenkins.fullname" . }} + servicePort: {{ .Values.controller.servicePort }} +{{- end }} + host: {{ tpl .Values.controller.ingress.resourceRootUrl . | quote }} +{{- end }} +{{- if .Values.controller.ingress.tls }} + tls: +{{ tpl (toYaml .Values.controller.ingress.tls ) . | indent 4 }} +{{- end -}} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-networkpolicy.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-networkpolicy.yaml new file mode 100644 index 000000000..82835f2bd --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-networkpolicy.yaml @@ -0,0 +1,76 @@ +{{- if .Values.networkPolicy.enabled }} +kind: NetworkPolicy +apiVersion: {{ .Values.networkPolicy.apiVersion }} +metadata: + name: "{{ .Release.Name }}-{{ .Values.controller.componentName }}" + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +spec: + podSelector: + matchLabels: + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + ingress: + # Allow web access to the UI + - ports: + - port: {{ .Values.controller.targetPort }} + {{- if .Values.controller.agentListenerEnabled }} + # Allow inbound connections from agents + - from: + {{- if .Values.networkPolicy.internalAgents.allowed }} + - podSelector: + matchLabels: + "jenkins/{{ .Release.Name }}-{{ .Values.agent.componentName }}": "true" + {{- range $k,$v:= .Values.networkPolicy.internalAgents.podLabels }} + {{ $k }}: {{ $v }} + {{- end }} + {{- if .Values.networkPolicy.internalAgents.namespaceLabels }} + namespaceSelector: + matchLabels: + {{- range $k,$v:= .Values.networkPolicy.internalAgents.namespaceLabels }} + {{ $k }}: {{ $v }} + {{- end }} + {{- end }} + {{- end }} + {{- if or .Values.networkPolicy.externalAgents.ipCIDR .Values.networkPolicy.externalAgents.except }} + - ipBlock: + cidr: {{ required "ipCIDR is required if you wish to allow external agents to connect to Jenkins Controller." .Values.networkPolicy.externalAgents.ipCIDR }} + {{- if .Values.networkPolicy.externalAgents.except }} + except: + {{- range .Values.networkPolicy.externalAgents.except }} + - {{ . }} + {{- end }} + {{- end }} + {{- end }} + ports: + - port: {{ .Values.controller.agentListenerPort }} + {{- end }} +{{- if .Values.agent.enabled }} +--- +kind: NetworkPolicy +apiVersion: {{ .Values.networkPolicy.apiVersion }} +metadata: + name: "{{ .Release.Name }}-{{ .Values.agent.componentName }}" + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +spec: + podSelector: + matchLabels: + # DefaultDeny + "jenkins/{{ .Release.Name }}-{{ .Values.agent.componentName }}": "true" +{{- end }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-pdb.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-pdb.yaml new file mode 100644 index 000000000..9dc1fafe2 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-pdb.yaml @@ -0,0 +1,34 @@ +{{- if .Values.controller.podDisruptionBudget.enabled }} +{{- $kubeTargetVersion := default .Capabilities.KubeVersion.GitVersion .Values.kubeTargetVersionOverride }} +{{- if semverCompare ">=1.21-0" $kubeTargetVersion -}} +apiVersion: policy/v1 +{{- else if semverCompare ">=1.5-0" $kubeTargetVersion -}} +apiVersion: policy/v1beta1 +{{- else -}} +apiVersion: {{ .Values.controller.podDisruptionBudget.apiVersion }} +{{- end }} +kind: PodDisruptionBudget +metadata: + name: {{ template "jenkins.fullname" . }}-pdb + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- if .Values.controller.podDisruptionBudget.labels -}} + {{ toYaml .Values.controller.podDisruptionBudget.labels | nindent 4 }} + {{- end }} + {{- if .Values.controller.podDisruptionBudget.annotations }} + annotations: {{ toYaml .Values.controller.podDisruptionBudget.annotations | nindent 4 }} + {{- end }} +spec: + maxUnavailable: {{ .Values.controller.podDisruptionBudget.maxUnavailable }} + selector: + matchLabels: + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-podmonitor.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-podmonitor.yaml new file mode 100644 index 000000000..9a04019c3 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-podmonitor.yaml @@ -0,0 +1,30 @@ +{{- if .Values.controller.googlePodMonitor.enabled }} +apiVersion: monitoring.googleapis.com/v1 +kind: PodMonitoring + +metadata: + name: {{ template "jenkins.fullname" . }} +{{- if .Values.controller.googlePodMonitor.serviceMonitorNamespace }} + namespace: {{ .Values.controller.googlePodMonitor.serviceMonitorNamespace }} +{{- else }} + namespace: {{ template "jenkins.namespace" . }} +{{- end }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + +spec: + endpoints: + - interval: {{ .Values.controller.googlePodMonitor.scrapeInterval }} + port: http + path: {{ .Values.controller.jenkinsUriPrefix }}{{ .Values.controller.googlePodMonitor.scrapeEndpoint }} + selector: + matchLabels: + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-route.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-route.yaml new file mode 100644 index 000000000..3550380ee --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-route.yaml @@ -0,0 +1,34 @@ +{{- if .Values.controller.route.enabled }} +apiVersion: route.openshift.io/v1 +kind: Route +metadata: + namespace: {{ template "jenkins.namespace" . }} + labels: + app: {{ template "jenkins.fullname" . }} + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + release: "{{ .Release.Name }}" + heritage: "{{ .Release.Service }}" + component: "{{ .Release.Name }}-{{ .Values.controller.componentName }}" +{{- if .Values.controller.route.labels }} +{{ toYaml .Values.controller.route.labels | indent 4 }} +{{- end }} +{{- if .Values.controller.route.annotations }} + annotations: +{{ toYaml .Values.controller.route.annotations | indent 4 }} +{{- end }} + name: {{ template "jenkins.fullname" . }} +spec: + host: {{ .Values.controller.route.path }} + port: + targetPort: http + tls: + insecureEdgeTerminationPolicy: Redirect + termination: edge + to: + kind: Service + name: {{ template "jenkins.fullname" . }} + weight: 100 + wildcardPolicy: None +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-secondary-ingress.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-secondary-ingress.yaml new file mode 100644 index 000000000..c63e48229 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-secondary-ingress.yaml @@ -0,0 +1,56 @@ +{{- if .Values.controller.secondaryingress.enabled }} +{{- $kubeTargetVersion := default .Capabilities.KubeVersion.GitVersion .Values.kubeTargetVersionOverride }} +{{- $serviceName := include "jenkins.fullname" . -}} +{{- $servicePort := .Values.controller.servicePort -}} +{{- if semverCompare ">=1.19-0" $kubeTargetVersion -}} +apiVersion: networking.k8s.io/v1 +{{- else if semverCompare ">=1.14-0" $kubeTargetVersion -}} +apiVersion: networking.k8s.io/v1beta1 +{{- else -}} +apiVersion: {{ .Values.controller.secondaryingress.apiVersion }} +{{- end }} +kind: Ingress +metadata: + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- if .Values.controller.secondaryingress.labels -}} + {{ toYaml .Values.controller.secondaryingress.labels | nindent 4 }} + {{- end }} + {{- if .Values.controller.secondaryingress.annotations }} + annotations: {{ toYaml .Values.controller.secondaryingress.annotations | nindent 4 }} + {{- end }} + name: {{ template "jenkins.fullname" . }}-secondary +spec: +{{- if .Values.controller.secondaryingress.ingressClassName }} + ingressClassName: {{ .Values.controller.secondaryingress.ingressClassName | quote }} +{{- end }} + rules: + - host: {{ .Values.controller.secondaryingress.hostName }} + http: + paths: + {{- range .Values.controller.secondaryingress.paths }} + - path: {{ . | quote }} + backend: +{{ if semverCompare ">=1.19-0" $kubeTargetVersion }} + service: + name: {{ $serviceName }} + port: + number: {{ $servicePort }} + pathType: ImplementationSpecific +{{ else }} + serviceName: {{ $serviceName }} + servicePort: {{ $servicePort }} +{{ end }} + {{- end}} +{{- if .Values.controller.secondaryingress.tls }} + tls: +{{ toYaml .Values.controller.secondaryingress.tls | indent 4 }} +{{- end -}} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-servicemonitor.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-servicemonitor.yaml new file mode 100644 index 000000000..8710b2bc9 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-servicemonitor.yaml @@ -0,0 +1,45 @@ +{{- if and .Values.controller.prometheus.enabled }} +apiVersion: monitoring.coreos.com/v1 +kind: ServiceMonitor + +metadata: + name: {{ template "jenkins.fullname" . }} +{{- if .Values.controller.prometheus.serviceMonitorNamespace }} + namespace: {{ .Values.controller.prometheus.serviceMonitorNamespace }} +{{- else }} + namespace: {{ template "jenkins.namespace" . }} +{{- end }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- range $key, $val := .Values.controller.prometheus.serviceMonitorAdditionalLabels }} + {{ $key }}: {{ $val | quote }} + {{- end}} + +spec: + endpoints: + - interval: {{ .Values.controller.prometheus.scrapeInterval }} + port: http + path: {{ .Values.controller.jenkinsUriPrefix }}{{ .Values.controller.prometheus.scrapeEndpoint }} + {{- with .Values.controller.prometheus.relabelings }} + relabelings: + {{- toYaml . | nindent 6 }} + {{- end }} + {{- with .Values.controller.prometheus.metricRelabelings }} + metricRelabelings: + {{- toYaml . | nindent 6 }} + {{- end }} + jobLabel: {{ template "jenkins.fullname" . }} + namespaceSelector: + matchNames: + - "{{ template "jenkins.namespace" $ }}" + selector: + matchLabels: + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-statefulset.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-statefulset.yaml new file mode 100644 index 000000000..50e61acf1 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-statefulset.yaml @@ -0,0 +1,424 @@ +{{- if .Capabilities.APIVersions.Has "apps/v1" }} +apiVersion: apps/v1 +{{- else }} +apiVersion: apps/v1beta1 +{{- end }} +kind: StatefulSet +metadata: + name: {{ template "jenkins.fullname" . }} + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- range $key, $val := .Values.controller.statefulSetLabels }} + {{ $key }}: {{ $val | quote }} + {{- end}} + {{- if .Values.controller.statefulSetAnnotations }} + annotations: +{{ toYaml .Values.controller.statefulSetAnnotations | indent 4 }} + {{- end }} +spec: + serviceName: {{ template "jenkins.fullname" . }} + replicas: 1 + selector: + matchLabels: + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + {{- if .Values.controller.updateStrategy }} + updateStrategy: +{{ toYaml .Values.controller.updateStrategy | indent 4 }} + {{- end }} + template: + metadata: + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- range $key, $val := .Values.controller.podLabels }} + {{ $key }}: {{ $val | quote }} + {{- end}} + annotations: + checksum/config: {{ include (print $.Template.BasePath "/config.yaml") . | sha256sum }} + {{- if .Values.controller.initScripts }} + checksum/config-init-scripts: {{ include (print $.Template.BasePath "/config-init-scripts.yaml") . | sha256sum }} + {{- end }} + {{- if .Values.controller.podAnnotations }} +{{ tpl (toYaml .Values.controller.podAnnotations | indent 8) . }} + {{- end }} + spec: + {{- if .Values.controller.schedulerName }} + schedulerName: {{ .Values.controller.schedulerName }} + {{- end }} + {{- if .Values.controller.nodeSelector }} + nodeSelector: +{{ toYaml .Values.controller.nodeSelector | indent 8 }} + {{- end }} + {{- if .Values.controller.tolerations }} + tolerations: +{{ toYaml .Values.controller.tolerations | indent 8 }} + {{- end }} + {{- if .Values.controller.affinity }} + affinity: +{{ toYaml .Values.controller.affinity | indent 8 }} + {{- end }} + {{- if .Values.controller.topologySpreadConstraints }} + topologySpreadConstraints: +{{ toYaml .Values.controller.topologySpreadConstraints | indent 8 }} + {{- end }} + {{- if quote .Values.controller.terminationGracePeriodSeconds }} + terminationGracePeriodSeconds: {{ .Values.controller.terminationGracePeriodSeconds }} + {{- end }} + {{- if .Values.controller.priorityClassName }} + priorityClassName: {{ .Values.controller.priorityClassName }} + {{- end }} + {{- if .Values.controller.shareProcessNamespace }} + shareProcessNamespace: true + {{- end }} +{{- if .Values.controller.usePodSecurityContext }} + securityContext: + {{- if kindIs "map" .Values.controller.podSecurityContextOverride }} + {{- tpl (toYaml .Values.controller.podSecurityContextOverride | nindent 8) . -}} + {{- else }} + {{/* The rest of this section should be replaced with the contents of this comment one the runAsUser, fsGroup, and securityContextCapabilities Helm chart values have been removed: + runAsUser: 1000 + fsGroup: 1000 + runAsNonRoot: true + */}} + runAsUser: {{ default 0 .Values.controller.runAsUser }} + {{- if and (.Values.controller.runAsUser) (.Values.controller.fsGroup) }} + {{- if not (eq (int .Values.controller.runAsUser) 0) }} + fsGroup: {{ .Values.controller.fsGroup }} + runAsNonRoot: true + {{- end }} + {{- if .Values.controller.securityContextCapabilities }} + capabilities: + {{- toYaml .Values.controller.securityContextCapabilities | nindent 10 }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} + serviceAccountName: "{{ template "jenkins.serviceAccountName" . }}" +{{- if .Values.controller.hostNetworking }} + hostNetwork: true + dnsPolicy: ClusterFirstWithHostNet +{{- end }} + {{- if .Values.controller.hostAliases }} + hostAliases: + {{- toYaml .Values.controller.hostAliases | nindent 8 }} + {{- end }} + initContainers: +{{- if .Values.controller.customInitContainers }} +{{ tpl (toYaml .Values.controller.customInitContainers) . | indent 8 }} +{{- end }} + +{{- if .Values.controller.sidecars.configAutoReload.enabled }} +{{- include "jenkins.configReloadContainer" (list $ "config-reload-init" "init") | nindent 8 }} +{{- end}} + + - name: "init" + image: "{{ .Values.controller.image.registry }}/{{ .Values.controller.image.repository }}:{{- include "controller.image.tag" . -}}" + imagePullPolicy: "{{ .Values.controller.image.pullPolicy }}" + {{- if .Values.controller.containerSecurityContext }} + securityContext: {{- toYaml .Values.controller.containerSecurityContext | nindent 12 }} + {{- end }} + command: [ "sh", "/var/jenkins_config/apply_config.sh" ] + {{- if .Values.controller.initContainerEnvFrom }} + envFrom: +{{ (tpl (toYaml .Values.controller.initContainerEnvFrom) .) | indent 12 }} + {{- end }} + {{- if .Values.controller.initContainerEnv }} + env: +{{ (tpl (toYaml .Values.controller.initContainerEnv) .) | indent 12 }} + {{- end }} + resources: +{{- if .Values.controller.initContainerResources }} +{{ toYaml .Values.controller.initContainerResources | indent 12 }} +{{- else }} +{{ toYaml .Values.controller.resources | indent 12 }} +{{- end }} + volumeMounts: + {{- if .Values.persistence.mounts }} +{{ toYaml .Values.persistence.mounts | indent 12 }} + {{- end }} + - mountPath: {{ .Values.controller.jenkinsHome }} + name: jenkins-home + {{- if .Values.persistence.subPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + - mountPath: /var/jenkins_config + name: jenkins-config + {{- if .Values.controller.installPlugins }} + {{- if .Values.controller.overwritePluginsFromImage }} + - mountPath: {{ .Values.controller.jenkinsRef }}/plugins + name: plugins + {{- end }} + - mountPath: /var/jenkins_plugins + name: plugin-dir + - mountPath: /tmp + name: tmp-volume + {{- end }} + {{- if or .Values.controller.initScripts .Values.controller.initConfigMap }} + - mountPath: {{ .Values.controller.jenkinsHome }}/init.groovy.d + name: init-scripts + {{- end }} + {{- if and .Values.controller.httpsKeyStore.enable (not .Values.controller.httpsKeyStore.disableSecretMount) }} + {{- $httpsJKSDirPath := printf "%s" .Values.controller.httpsKeyStore.path }} + - mountPath: {{ $httpsJKSDirPath }} + name: jenkins-https-keystore + {{- end }} + containers: + - name: jenkins + image: "{{ .Values.controller.image.registry }}/{{ .Values.controller.image.repository }}:{{- include "controller.image.tag" . -}}" + imagePullPolicy: "{{ .Values.controller.image.pullPolicy }}" + {{- if .Values.controller.containerSecurityContext }} + securityContext: {{- toYaml .Values.controller.containerSecurityContext | nindent 12 }} + {{- end }} + {{- if .Values.controller.overrideArgs }} + args: [ + {{- range $overrideArg := .Values.controller.overrideArgs }} + "{{- tpl $overrideArg $ }}", + {{- end }} + ] + {{- else if .Values.controller.httpsKeyStore.enable }} + {{- $httpsJKSFilePath := printf "%s/%s" .Values.controller.httpsKeyStore.path .Values.controller.httpsKeyStore.fileName }} + args: [ "--httpPort={{.Values.controller.httpsKeyStore.httpPort}}", "--httpsPort={{.Values.controller.targetPort}}", '--httpsKeyStore={{ $httpsJKSFilePath }}', "--httpsKeyStorePassword=$(JENKINS_HTTPS_KEYSTORE_PASSWORD)" ] + {{- else }} + args: [ "--httpPort={{.Values.controller.targetPort}}"] + {{- end }} + {{- if .Values.controller.lifecycle }} + lifecycle: +{{ toYaml .Values.controller.lifecycle | indent 12 }} + {{- end }} +{{- if .Values.controller.terminationMessagePath }} + terminationMessagePath: {{ .Values.controller.terminationMessagePath }} +{{- end }} +{{- if .Values.controller.terminationMessagePolicy }} + terminationMessagePolicy: {{ .Values.controller.terminationMessagePolicy }} +{{- end }} + {{- if .Values.controller.containerEnvFrom }} + envFrom: +{{ (tpl ( toYaml .Values.controller.containerEnvFrom) .) | indent 12 }} + {{- end }} + env: + {{- if .Values.controller.containerEnv }} +{{ (tpl ( toYaml .Values.controller.containerEnv) .) | indent 12 }} + {{- end }} + {{- if or .Values.controller.additionalSecrets .Values.controller.existingSecret .Values.controller.additionalExistingSecrets .Values.controller.admin.createSecret }} + - name: SECRETS + value: /run/secrets/additional + {{- end }} + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: JAVA_OPTS + value: >- + {{ if .Values.controller.sidecars.configAutoReload.enabled }} -Dcasc.reload.token=$(POD_NAME) {{ end }}{{ default "" .Values.controller.javaOpts }} + - name: JENKINS_OPTS + value: >- + {{ if .Values.controller.jenkinsUriPrefix }}--prefix={{ .Values.controller.jenkinsUriPrefix }} {{ end }} --webroot=/var/jenkins_cache/war {{ default "" .Values.controller.jenkinsOpts}} + - name: JENKINS_SLAVE_AGENT_PORT + value: "{{ .Values.controller.agentListenerPort }}" + {{- if .Values.controller.httpsKeyStore.enable }} + - name: JENKINS_HTTPS_KEYSTORE_PASSWORD + {{- if not .Values.controller.httpsKeyStore.disableSecretMount }} + valueFrom: + secretKeyRef: + name: {{ if .Values.controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretName }} {{ .Values.controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretName }} {{ else if .Values.controller.httpsKeyStore.jenkinsHttpsJksSecretName }} {{ .Values.controller.httpsKeyStore.jenkinsHttpsJksSecretName }} {{ else }} {{ template "jenkins.fullname" . }}-https-jks {{ end }} + key: "{{ .Values.controller.httpsKeyStore.jenkinsHttpsJksPasswordSecretKey }}" + {{- else }} + value: {{ .Values.controller.httpsKeyStore.password }} + {{- end }} + {{- end }} + + - name: CASC_JENKINS_CONFIG + value: {{ .Values.controller.sidecars.configAutoReload.folder | default (printf "%s/casc_configs" (.Values.controller.jenkinsRef)) }}{{- if .Values.controller.JCasC.configUrls }},{{ join "," .Values.controller.JCasC.configUrls }}{{- end }} + ports: + {{- if .Values.controller.httpsKeyStore.enable }} + - containerPort: {{.Values.controller.httpsKeyStore.httpPort}} + {{- else }} + - containerPort: {{.Values.controller.targetPort}} + {{- end }} + name: http + - containerPort: {{ .Values.controller.agentListenerPort }} + name: agent-listener + {{- if .Values.controller.agentListenerHostPort }} + hostPort: {{ .Values.controller.agentListenerHostPort }} + {{- end }} + {{- if .Values.controller.jmxPort }} + - containerPort: {{ .Values.controller.jmxPort }} + name: jmx + {{- end }} +{{- range $index, $port := .Values.controller.extraPorts }} + - containerPort: {{ $port.port }} + name: {{ $port.name }} +{{- end }} +{{- if and .Values.controller.healthProbes .Values.controller.probes}} + {{- if semverCompare ">=1.16-0" .Capabilities.KubeVersion.GitVersion }} + startupProbe: +{{ tpl (toYaml .Values.controller.probes.startupProbe | indent 12) .}} + {{- end }} + livenessProbe: +{{ tpl (toYaml .Values.controller.probes.livenessProbe | indent 12) .}} + readinessProbe: +{{ tpl (toYaml .Values.controller.probes.readinessProbe | indent 12) .}} +{{- end }} + resources: +{{ toYaml .Values.controller.resources | indent 12 }} + volumeMounts: +{{- if .Values.persistence.mounts }} +{{ toYaml .Values.persistence.mounts | indent 12 }} +{{- end }} + {{- if and .Values.controller.httpsKeyStore.enable (not .Values.controller.httpsKeyStore.disableSecretMount) }} + {{- $httpsJKSDirPath := printf "%s" .Values.controller.httpsKeyStore.path }} + - mountPath: {{ $httpsJKSDirPath }} + name: jenkins-https-keystore + {{- end }} + - mountPath: {{ .Values.controller.jenkinsHome }} + name: jenkins-home + readOnly: false + {{- if .Values.persistence.subPath }} + subPath: {{ .Values.persistence.subPath }} + {{- end }} + - mountPath: /var/jenkins_config + name: jenkins-config + readOnly: true + {{- if .Values.controller.installPlugins }} + - mountPath: {{ .Values.controller.jenkinsRef }}/plugins/ + name: plugin-dir + readOnly: false + {{- end }} + {{- if or .Values.controller.initScripts .Values.controller.initConfigMap }} + - mountPath: {{ .Values.controller.jenkinsHome }}/init.groovy.d + name: init-scripts + {{- end }} + {{- if .Values.controller.sidecars.configAutoReload.enabled }} + - name: sc-config-volume + mountPath: {{ .Values.controller.sidecars.configAutoReload.folder | default (printf "%s/casc_configs" (.Values.controller.jenkinsRef)) }} + {{- end }} + {{- if or .Values.controller.additionalSecrets .Values.controller.existingSecret .Values.controller.additionalExistingSecrets .Values.controller.admin.createSecret }} + - name: jenkins-secrets + mountPath: /run/secrets/additional + readOnly: true + {{- end }} + - name: jenkins-cache + mountPath: /var/jenkins_cache + - mountPath: /tmp + name: tmp-volume + +{{- if .Values.controller.sidecars.configAutoReload.enabled }} +{{- include "jenkins.configReloadContainer" (list $ "config-reload" "sidecar") | nindent 8 }} +{{- end}} + + +{{- if .Values.controller.sidecars.additionalSidecarContainers}} +{{ tpl (toYaml .Values.controller.sidecars.additionalSidecarContainers | indent 8) .}} +{{- end }} + + volumes: +{{- if .Values.persistence.volumes }} +{{ tpl (toYaml .Values.persistence.volumes | indent 6) . }} +{{- end }} + {{- if .Values.controller.sidecars.configAutoReload.logging.configuration.override }} + - name: auto-reload-config + configMap: + name: {{ template "jenkins.fullname" . }}-auto-reload-config + - name: auto-reload-config-logs + emptyDir: {} + {{- end }} + {{- if .Values.controller.installPlugins }} + {{- if .Values.controller.overwritePluginsFromImage }} + - name: plugins + emptyDir: {} + {{- end }} + {{- end }} + {{- if and .Values.controller.initScripts .Values.controller.initConfigMap }} + - name: init-scripts + projected: + sources: + - configMap: + name: {{ template "jenkins.fullname" . }}-init-scripts + - configMap: + name: {{ .Values.controller.initConfigMap }} + {{- else if .Values.controller.initConfigMap }} + - name: init-scripts + configMap: + name: {{ .Values.controller.initConfigMap }} + {{- else if .Values.controller.initScripts }} + - name: init-scripts + configMap: + name: {{ template "jenkins.fullname" . }}-init-scripts + {{- end }} + - name: jenkins-config + configMap: + name: {{ template "jenkins.fullname" . }} + {{- if .Values.controller.installPlugins }} + - name: plugin-dir + emptyDir: {} + {{- end }} + {{- if or .Values.controller.additionalSecrets .Values.controller.existingSecret .Values.controller.additionalExistingSecrets .Values.controller.admin.createSecret }} + - name: jenkins-secrets + projected: + sources: + {{- if .Values.controller.additionalSecrets }} + - secret: + name: {{ template "jenkins.fullname" . }}-additional-secrets + {{- end }} + {{- if .Values.controller.additionalExistingSecrets }} + {{- range $key, $value := .Values.controller.additionalExistingSecrets }} + - secret: + name: {{ tpl $value.name $ }} + items: + - key: {{ tpl $value.keyName $ }} + path: {{ tpl $value.name $ }}-{{ tpl $value.keyName $ }} + {{- end }} + {{- end }} + {{- if .Values.controller.admin.createSecret }} + - secret: + name: {{ .Values.controller.admin.existingSecret | default (include "jenkins.fullname" .) }} + items: + - key: {{ .Values.controller.admin.userKey | default "jenkins-admin-user" }} + path: chart-admin-username + - key: {{ .Values.controller.admin.passwordKey | default "jenkins-admin-password" }} + path: chart-admin-password + {{- end }} + {{- if .Values.controller.existingSecret }} + - secret: + name: {{ .Values.controller.existingSecret }} + {{- end }} + {{- end }} + - name: jenkins-cache + emptyDir: {} + {{- if not (contains "jenkins-home" (quote .Values.persistence.volumes)) }} + - name: jenkins-home + {{- if .Values.persistence.enabled }} + persistentVolumeClaim: + claimName: {{ .Values.persistence.existingClaim | default (include "jenkins.fullname" .) }} + {{- else }} + emptyDir: {} + {{- end -}} + {{- end }} + - name: sc-config-volume + emptyDir: {} + - name: tmp-volume + emptyDir: {} + + {{- if and .Values.controller.httpsKeyStore.enable (not .Values.controller.httpsKeyStore.disableSecretMount) }} + - name: jenkins-https-keystore + secret: + secretName: {{ if .Values.controller.httpsKeyStore.jenkinsHttpsJksSecretName }} {{ .Values.controller.httpsKeyStore.jenkinsHttpsJksSecretName }} {{ else }} {{ template "jenkins.fullname" . }}-https-jks {{ end }} + items: + - key: {{ .Values.controller.httpsKeyStore.jenkinsHttpsJksSecretKey }} + path: {{ .Values.controller.httpsKeyStore.fileName }} + {{- end }} + +{{- if .Values.controller.imagePullSecretName }} + imagePullSecrets: + - name: {{ .Values.controller.imagePullSecretName }} +{{- end -}} diff --git a/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-svc.yaml b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-svc.yaml new file mode 100644 index 000000000..a83466ce3 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/jenkins-controller-svc.yaml @@ -0,0 +1,56 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{template "jenkins.fullname" . }} + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + {{- if .Values.controller.serviceLabels }} +{{ toYaml .Values.controller.serviceLabels | indent 4 }} + {{- end }} +{{- if .Values.controller.serviceAnnotations }} + annotations: +{{ toYaml .Values.controller.serviceAnnotations | indent 4 }} +{{- end }} +spec: + {{- if .Values.controller.serviceExternalTrafficPolicy }} + externalTrafficPolicy: {{.Values.controller.serviceExternalTrafficPolicy}} + {{- end }} + {{- if (and (eq .Values.controller.serviceType "ClusterIP") (not (empty .Values.controller.clusterIP))) }} + clusterIP: {{.Values.controller.clusterIP}} + {{- end }} + ports: + - port: {{.Values.controller.servicePort}} + name: http + targetPort: {{ .Values.controller.targetPort }} + {{- if (and (eq .Values.controller.serviceType "NodePort") (not (empty .Values.controller.nodePort))) }} + nodePort: {{.Values.controller.nodePort}} + {{- end }} +{{- range $index, $port := .Values.controller.extraPorts }} + - port: {{ $port.port }} + name: {{ $port.name }} + {{- if $port.targetPort }} + targetPort: {{ $port.targetPort }} + {{- else }} + targetPort: {{ $port.port }} + {{- end -}} +{{- end }} + selector: + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + type: {{.Values.controller.serviceType}} + {{if eq .Values.controller.serviceType "LoadBalancer"}} +{{- if .Values.controller.loadBalancerSourceRanges }} + loadBalancerSourceRanges: +{{ toYaml .Values.controller.loadBalancerSourceRanges | indent 4 }} +{{- end }} + {{if .Values.controller.loadBalancerIP}} + loadBalancerIP: {{.Values.controller.loadBalancerIP}} + {{end}} + {{end}} diff --git a/charts/jenkins/jenkins/5.4.2/templates/rbac.yaml b/charts/jenkins/jenkins/5.4.2/templates/rbac.yaml new file mode 100644 index 000000000..581cb8d48 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/rbac.yaml @@ -0,0 +1,149 @@ +{{ if .Values.rbac.create }} +{{- $serviceName := include "jenkins.fullname" . -}} + +# This role is used to allow Jenkins scheduling of agents via Kubernetes plugin. +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ $serviceName }}-schedule-agents + namespace: {{ template "jenkins.agent.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +rules: +- apiGroups: [""] + resources: ["pods", "pods/exec", "pods/log", "persistentvolumeclaims", "events"] + verbs: ["get", "list", "watch"] +- apiGroups: [""] + resources: ["pods", "pods/exec", "persistentvolumeclaims"] + verbs: ["create", "delete", "deletecollection", "patch", "update"] + +--- + +# We bind the role to the Jenkins service account. The role binding is created in the namespace +# where the agents are supposed to run. +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ $serviceName }}-schedule-agents + namespace: {{ template "jenkins.agent.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ $serviceName }}-schedule-agents +subjects: +- kind: ServiceAccount + name: {{ template "jenkins.serviceAccountName" .}} + namespace: {{ template "jenkins.namespace" . }} + +--- + +{{- if .Values.rbac.readSecrets }} +# This is needed if you want to use https://jenkinsci.github.io/kubernetes-credentials-provider-plugin/ +# as it needs permissions to get/watch/list Secrets +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ template "jenkins.fullname" . }}-read-secrets + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +rules: + - apiGroups: [""] + resources: ["secrets"] + verbs: ["get", "watch", "list"] + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ $serviceName }}-read-secrets + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ template "jenkins.fullname" . }}-read-secrets +subjects: + - kind: ServiceAccount + name: {{ template "jenkins.serviceAccountName" . }} + namespace: {{ template "jenkins.namespace" . }} + +--- +{{- end}} + +{{- if .Values.controller.sidecars.configAutoReload.enabled }} +# The sidecar container which is responsible for reloading configuration changes +# needs permissions to watch ConfigMaps +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ template "jenkins.fullname" . }}-casc-reload + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +rules: +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["get", "watch", "list"] + +--- + +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ $serviceName }}-watch-configmaps + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ template "jenkins.fullname" . }}-casc-reload +subjects: +- kind: ServiceAccount + name: {{ template "jenkins.serviceAccountName" . }} + namespace: {{ template "jenkins.namespace" . }} + +{{- end}} + +{{ end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/secret-additional.yaml b/charts/jenkins/jenkins/5.4.2/templates/secret-additional.yaml new file mode 100644 index 000000000..d1908aa9b --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/secret-additional.yaml @@ -0,0 +1,21 @@ +{{- if .Values.controller.additionalSecrets -}} +--- +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "jenkins.fullname" . }}-additional-secrets + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +type: Opaque +data: +{{- range .Values.controller.additionalSecrets }} + {{ .name }}: {{ .value | b64enc }} +{{- end }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/secret-claims.yaml b/charts/jenkins/jenkins/5.4.2/templates/secret-claims.yaml new file mode 100644 index 000000000..e8b6d6c8e --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/secret-claims.yaml @@ -0,0 +1,29 @@ +{{- if .Values.controller.secretClaims -}} +{{- $r := .Release -}} +{{- $v := .Values -}} +{{- $chart := printf "%s-%s" .Chart.Name .Chart.Version -}} +{{- $namespace := include "jenkins.namespace" . -}} +{{- $serviceName := include "jenkins.fullname" . -}} +{{ range .Values.controller.secretClaims }} +--- +kind: SecretClaim +apiVersion: vaultproject.io/v1 +metadata: + name: {{ $serviceName }}-{{ .name | default .path | lower }} + namespace: {{ $namespace }} + labels: + "app.kubernetes.io/name": '{{ $serviceName }}' + {{- if $v.renderHelmLabels }} + "helm.sh/chart": "{{ $chart }}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ $r.Service }}" + "app.kubernetes.io/instance": "{{ $r.Name }}" + "app.kubernetes.io/component": "{{ $v.controller.componentName }}" +spec: + type: {{ .type | default "Opaque" }} + path: {{ .path }} +{{- if .renew }} + renew: {{ .renew }} +{{- end }} +{{- end }} +{{- end }} \ No newline at end of file diff --git a/charts/jenkins/jenkins/5.4.2/templates/secret-https-jks.yaml b/charts/jenkins/jenkins/5.4.2/templates/secret-https-jks.yaml new file mode 100644 index 000000000..5348de41e --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/secret-https-jks.yaml @@ -0,0 +1,20 @@ +{{- if and .Values.controller.httpsKeyStore.enable ( not .Values.controller.httpsKeyStore.jenkinsHttpsJksSecretName ) (not .Values.controller.httpsKeyStore.disableSecretMount) -}} +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "jenkins.fullname" . }}-https-jks + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +type: Opaque +data: + jenkins-jks-file: | +{{ .Values.controller.httpsKeyStore.jenkinsKeyStoreBase64Encoded | indent 4 }} + https-jks-password: {{ .Values.controller.httpsKeyStore.password | b64enc }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/secret.yaml b/charts/jenkins/jenkins/5.4.2/templates/secret.yaml new file mode 100644 index 000000000..cc6ace179 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/secret.yaml @@ -0,0 +1,20 @@ +{{- if and (not .Values.controller.admin.existingSecret) (.Values.controller.admin.createSecret) -}} + +apiVersion: v1 +kind: Secret +metadata: + name: {{ template "jenkins.fullname" . }} + namespace: {{ template "jenkins.namespace" . }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +type: Opaque +data: + jenkins-admin-password: {{ template "jenkins.password" . }} + jenkins-admin-user: {{ .Values.controller.admin.username | b64enc | quote }} +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/service-account-agent.yaml b/charts/jenkins/jenkins/5.4.2/templates/service-account-agent.yaml new file mode 100644 index 000000000..48f08ba6c --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/service-account-agent.yaml @@ -0,0 +1,26 @@ +{{ if .Values.serviceAccountAgent.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "jenkins.serviceAccountAgentName" . }} + namespace: {{ template "jenkins.agent.namespace" . }} +{{- if .Values.serviceAccountAgent.annotations }} + annotations: +{{ tpl (toYaml .Values.serviceAccountAgent.annotations) . | indent 4 }} +{{- end }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- if .Values.serviceAccountAgent.extraLabels }} +{{ tpl (toYaml .Values.serviceAccountAgent.extraLabels) . | indent 4 }} +{{- end }} +{{- if .Values.serviceAccountAgent.imagePullSecretName }} +imagePullSecrets: + - name: {{ .Values.serviceAccountAgent.imagePullSecretName }} +{{- end -}} +{{ end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/service-account.yaml b/charts/jenkins/jenkins/5.4.2/templates/service-account.yaml new file mode 100644 index 000000000..b44eb488c --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/service-account.yaml @@ -0,0 +1,26 @@ +{{ if .Values.serviceAccount.create }} +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ include "jenkins.serviceAccountName" . }} + namespace: {{ template "jenkins.namespace" . }} +{{- if .Values.serviceAccount.annotations }} + annotations: +{{ tpl (toYaml .Values.serviceAccount.annotations) . | indent 4 }} +{{- end }} + labels: + "app.kubernetes.io/name": '{{ template "jenkins.name" .}}' + {{- if .Values.renderHelmLabels }} + "helm.sh/chart": "{{ template "jenkins.label" .}}" + {{- end }} + "app.kubernetes.io/managed-by": "{{ .Release.Service }}" + "app.kubernetes.io/instance": "{{ .Release.Name }}" + "app.kubernetes.io/component": "{{ .Values.controller.componentName }}" +{{- if .Values.serviceAccount.extraLabels }} +{{ tpl (toYaml .Values.serviceAccount.extraLabels) . | indent 4 }} +{{- end }} +{{- if .Values.serviceAccount.imagePullSecretName }} +imagePullSecrets: + - name: {{ .Values.serviceAccount.imagePullSecretName }} +{{- end -}} +{{ end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/tests/jenkins-test.yaml b/charts/jenkins/jenkins/5.4.2/templates/tests/jenkins-test.yaml new file mode 100644 index 000000000..12a935ecc --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/tests/jenkins-test.yaml @@ -0,0 +1,49 @@ +{{- if .Values.controller.testEnabled }} +apiVersion: v1 +kind: Pod +metadata: + name: "{{ .Release.Name }}-ui-test-{{ randAlphaNum 5 | lower }}" + namespace: {{ template "jenkins.namespace" . }} + annotations: + "helm.sh/hook": test-success +spec: + {{- if .Values.controller.nodeSelector }} + nodeSelector: +{{ toYaml .Values.controller.nodeSelector | indent 4 }} + {{- end }} + {{- if .Values.controller.tolerations }} + tolerations: +{{ toYaml .Values.controller.tolerations | indent 4 }} + {{- end }} + initContainers: + - name: "test-framework" + image: "{{ .Values.helmtest.bats.image.registry }}/{{ .Values.helmtest.bats.image.repository }}:{{ .Values.helmtest.bats.image.tag }}" + command: + - "bash" + - "-c" + args: + - | + # copy bats to tools dir + set -ex + cp -R /opt/bats /tools/bats/ + volumeMounts: + - mountPath: /tools + name: tools + containers: + - name: {{ .Release.Name }}-ui-test + image: "{{ .Values.controller.image.registry }}/{{ .Values.controller.image.repository }}:{{- include "controller.image.tag" . -}}" + command: ["/tools/bats/bin/bats", "-t", "/tests/run.sh"] + volumeMounts: + - mountPath: /tests + name: tests + readOnly: true + - mountPath: /tools + name: tools + volumes: + - name: tests + configMap: + name: {{ template "jenkins.fullname" . }}-tests + - name: tools + emptyDir: {} + restartPolicy: Never +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/templates/tests/test-config.yaml b/charts/jenkins/jenkins/5.4.2/templates/tests/test-config.yaml new file mode 100644 index 000000000..12c5b3a0d --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/templates/tests/test-config.yaml @@ -0,0 +1,14 @@ +{{- if .Values.controller.testEnabled }} +apiVersion: v1 +kind: ConfigMap +metadata: + name: {{ template "jenkins.fullname" . }}-tests + namespace: {{ template "jenkins.namespace" . }} + annotations: + "helm.sh/hook": test +data: + run.sh: |- + @test "Testing Jenkins UI is accessible" { + curl --retry 48 --retry-delay 10 {{ template "jenkins.fullname" . }}:{{ .Values.controller.servicePort }}{{ default "" .Values.controller.jenkinsUriPrefix }}/login + } +{{- end }} diff --git a/charts/jenkins/jenkins/5.4.2/values.yaml b/charts/jenkins/jenkins/5.4.2/values.yaml new file mode 100644 index 000000000..293df1a86 --- /dev/null +++ b/charts/jenkins/jenkins/5.4.2/values.yaml @@ -0,0 +1,1333 @@ +# Default values for jenkins. +# This is a YAML-formatted file. +# Declare name/value pairs to be passed into your templates. +# name: value + +## Overrides for generated resource names +# See templates/_helpers.tpl +# -- Override the resource name prefix +# @default -- `Chart.Name` +nameOverride: +# -- Override the full resource names +# @default -- `jenkins-(release-name)` or `jenkins` if the release-name is `jenkins` +fullnameOverride: +# -- Override the deployment namespace +# @default -- `Release.Namespace` +namespaceOverride: + +# For FQDN resolving of the controller service. Change this value to match your existing configuration. +# ref: https://github.com/kubernetes/dns/blob/master/docs/specification.md +# -- Override the cluster name for FQDN resolving +clusterZone: "cluster.local" + +# -- The URL of the Kubernetes API server +kubernetesURL: "https://kubernetes.default" + +# -- The Jenkins credentials to access the Kubernetes API server. For the default cluster it is not needed. +credentialsId: + +# -- Enables rendering of the helm.sh/chart label to the annotations +renderHelmLabels: true + +controller: + # -- Used for label app.kubernetes.io/component + componentName: "jenkins-controller" + image: + # -- Controller image registry + registry: "docker.io" + # -- Controller image repository + repository: "jenkins/jenkins" + + # -- Controller image tag override; i.e., tag: "2.440.1-jdk17" + tag: + + # -- Controller image tag label + tagLabel: jdk17 + # -- Controller image pull policy + pullPolicy: "Always" + # -- Controller image pull secret + imagePullSecretName: + # -- Lifecycle specification for controller-container + lifecycle: {} + # postStart: + # exec: + # command: + # - "uname" + # - "-a" + + # -- Disable use of remember me + disableRememberMe: false + + # -- Set Number of executors + numExecutors: 0 + + # -- Sets the executor mode of the Jenkins node. Possible values are "NORMAL" or "EXCLUSIVE" + executorMode: "NORMAL" + + # -- Append Jenkins labels to the controller + customJenkinsLabels: [] + + hostNetworking: false + + # When enabling LDAP or another non-Jenkins identity source, the built-in admin account will no longer exist. + # If you disable the non-Jenkins identity store and instead use the Jenkins internal one, + # you should revert controller.admin.username to your preferred admin user: + admin: + + # -- Admin username created as a secret if `controller.admin.createSecret` is true + username: "admin" + # -- Admin password created as a secret if `controller.admin.createSecret` is true + # @default -- + password: + + # -- The key in the existing admin secret containing the username + userKey: jenkins-admin-user + # -- The key in the existing admin secret containing the password + passwordKey: jenkins-admin-password + + # The default configuration uses this secret to configure an admin user + # If you don't need that user or use a different security realm, then you can disable it + # -- Create secret for admin user + createSecret: true + + # -- The name of an existing secret containing the admin credentials + existingSecret: "" + # -- Email address for the administrator of the Jenkins instance + jenkinsAdminEmail: + + # This value should not be changed unless you use your custom image of jenkins or any derived from. + # If you want to use Cloudbees Jenkins Distribution docker, you should set jenkinsHome: "/var/cloudbees-jenkins-distribution" + # -- Custom Jenkins home path + jenkinsHome: "/var/jenkins_home" + + # This value should not be changed unless you use your custom image of jenkins or any derived from. + # If you want to use Cloudbees Jenkins Distribution docker, you should set jenkinsRef: "/usr/share/cloudbees-jenkins-distribution/ref" + # -- Custom Jenkins reference path + jenkinsRef: "/usr/share/jenkins/ref" + + # Path to the jenkins war file which is used by jenkins-plugin-cli. + jenkinsWar: "/usr/share/jenkins/jenkins.war" + # Override the default arguments passed to the war + # overrideArgs: + # - --httpPort=8080 + + # -- Resource allocation (Requests and Limits) + resources: + requests: + cpu: "50m" + memory: "256Mi" + limits: + cpu: "2000m" + memory: "4096Mi" + + # Share process namespace to allow sidecar containers to interact with processes in other containers in the same pod + shareProcessNamespace: false + + # Overrides the init container default values + # -- Resources allocation (Requests and Limits) for Init Container + initContainerResources: {} + # initContainerResources: + # requests: + # cpu: "50m" + # memory: "256Mi" + # limits: + # cpu: "2000m" + # memory: "4096Mi" + # -- Environment variable sources for Init Container + initContainerEnvFrom: [] + + # useful for i.e., http_proxy + # -- Environment variables for Init Container + initContainerEnv: [] + # initContainerEnv: + # - name: http_proxy + # value: "http://192.168.64.1:3128" + + # -- Environment variable sources for Jenkins Container + containerEnvFrom: [] + + # -- Environment variables for Jenkins Container + containerEnv: [] + # - name: http_proxy + # value: "http://192.168.64.1:3128" + + # Set min/max heap here if needed with "-Xms512m -Xmx512m" + # -- Append to `JAVA_OPTS` env var + javaOpts: + # -- Append to `JENKINS_OPTS` env var + jenkinsOpts: + + # If you are using the ingress definitions provided by this chart via the `controller.ingress` block, + # the configured hostname will be the ingress hostname starting with `https://` + # or `http://` depending on the `tls` configuration. + # The Protocol can be overwritten by specifying `controller.jenkinsUrlProtocol`. + # -- Set protocol for Jenkins URL; `https` if `controller.ingress.tls`, `http` otherwise + jenkinsUrlProtocol: + + # -- Set Jenkins URL if you are not using the ingress definitions provided by the chart + jenkinsUrl: + + # If you set this prefix and use ingress controller, then you might want to set the ingress path below + # I.e., "/jenkins" + # -- Root URI Jenkins will be served on + jenkinsUriPrefix: + + # -- Enable pod security context (must be `true` if podSecurityContextOverride, runAsUser or fsGroup are set) + usePodSecurityContext: true + + # Note that `runAsUser`, `fsGroup`, and `securityContextCapabilities` are + # being deprecated and replaced by `podSecurityContextOverride`. + # Set runAsUser to 1000 to let Jenkins run as non-root user 'jenkins', which exists in 'jenkins/jenkins' docker image. + # When configuring runAsUser to a different value than 0 also set fsGroup to the same value: + # -- Deprecated in favor of `controller.podSecurityContextOverride`. uid that jenkins runs with. + runAsUser: 1000 + + # -- Deprecated in favor of `controller.podSecurityContextOverride`. uid that will be used for persistent volume. + fsGroup: 1000 + + # If you have PodSecurityPolicies that require dropping of capabilities as suggested by CIS K8s benchmark, put them here + # securityContextCapabilities: + # drop: + # - NET_RAW + securityContextCapabilities: {} + + # In the case of mounting an ext4 filesystem, it might be desirable to use `supplementalGroups` instead of `fsGroup` in + # the `securityContext` block: https://github.com/kubernetes/kubernetes/issues/67014#issuecomment-589915496 + # podSecurityContextOverride: + # runAsUser: 1000 + # runAsNonRoot: true + # supplementalGroups: [1000] + # capabilities: {} + # -- Completely overwrites the contents of the pod security context, ignoring the values provided for `runAsUser`, `fsGroup`, and `securityContextCapabilities` + podSecurityContextOverride: ~ + + # -- Allow controlling the securityContext for the jenkins container + containerSecurityContext: + runAsUser: 1000 + runAsGroup: 1000 + readOnlyRootFilesystem: true + allowPrivilegeEscalation: false + + # For minikube, set this to NodePort, elsewhere uses LoadBalancer + # Use ClusterIP if your setup includes ingress controller + # -- k8s service type + serviceType: ClusterIP + + # -- k8s service clusterIP. Only used if serviceType is ClusterIP + clusterIp: + # -- k8s service port + servicePort: 8080 + # -- k8s target port + targetPort: 8080 + # -- k8s node port. Only used if serviceType is NodePort + nodePort: + + # Use Local to preserve the client source IP and avoids a second hop for LoadBalancer and NodePort type services, + # but risks potentially imbalanced traffic spreading. + serviceExternalTrafficPolicy: + + # -- Jenkins controller service annotations + serviceAnnotations: {} + # -- Jenkins controller custom labels for the StatefulSet + statefulSetLabels: {} + # foo: bar + # bar: foo + # -- Labels for the Jenkins controller-service + serviceLabels: {} + # service.beta.kubernetes.io/aws-load-balancer-backend-protocol: https + + # Put labels on Jenkins controller pod + # -- Custom Pod labels (an object with `label-key: label-value` pairs) + podLabels: {} + + # Enable Kubernetes Startup, Liveness and Readiness Probes + # if Startup Probe is supported, enable it too + # ~ 2 minutes to allow Jenkins to restart when upgrading plugins. Set ReadinessTimeout to be shorter than LivenessTimeout. + # ref: https://kubernetes.io/docs/tasks/configure-pod-container/configure-liveness-readiness-startup-probes + # -- Enable Kubernetes Probes configuration configured in `controller.probes` + healthProbes: true + + probes: + startupProbe: + # -- Set the failure threshold for the startup probe + failureThreshold: 12 + httpGet: + # -- Set the Pod's HTTP path for the startup probe + path: '{{ default "" .Values.controller.jenkinsUriPrefix }}/login' + # -- Set the Pod's HTTP port to use for the startup probe + port: http + # -- Set the time interval between two startup probes executions in seconds + periodSeconds: 10 + # -- Set the timeout for the startup probe in seconds + timeoutSeconds: 5 + + livenessProbe: + # -- Set the failure threshold for the liveness probe + failureThreshold: 5 + httpGet: + # -- Set the Pod's HTTP path for the liveness probe + path: '{{ default "" .Values.controller.jenkinsUriPrefix }}/login' + # -- Set the Pod's HTTP port to use for the liveness probe + port: http + # -- Set the time interval between two liveness probes executions in seconds + periodSeconds: 10 + # -- Set the timeout for the liveness probe in seconds + timeoutSeconds: 5 + + # If Startup Probe is not supported on your Kubernetes cluster, you might want to use "initialDelaySeconds" instead. + # It delays the initial liveness probe while Jenkins is starting + # -- Set the initial delay for the liveness probe in seconds + initialDelaySeconds: + + readinessProbe: + # -- Set the failure threshold for the readiness probe + failureThreshold: 3 + httpGet: + # -- Set the Pod's HTTP path for the liveness probe + path: '{{ default "" .Values.controller.jenkinsUriPrefix }}/login' + # -- Set the Pod's HTTP port to use for the readiness probe + port: http + # -- Set the time interval between two readiness probes executions in seconds + periodSeconds: 10 + # -- Set the timeout for the readiness probe in seconds + timeoutSeconds: 5 + + # If Startup Probe is not supported on your Kubernetes cluster, you might want to use "initialDelaySeconds" instead. + # It delays the initial readiness probe while Jenkins is starting + # -- Set the initial delay for the readiness probe in seconds + initialDelaySeconds: + + # PodDisruptionBudget config + podDisruptionBudget: + # ref: https://kubernetes.io/docs/tasks/run-application/configure-pdb/ + + # -- Enable Kubernetes Pod Disruption Budget configuration + enabled: false + + # For Kubernetes v1.5+, use 'policy/v1beta1' + # For Kubernetes v1.21+, use 'policy/v1' + # -- Policy API version + apiVersion: "policy/v1beta1" + + annotations: {} + labels: {} + # -- Number of pods that can be unavailable. Either an absolute number or a percentage + maxUnavailable: "0" + + # -- Create Agent listener service + agentListenerEnabled: true + # -- Listening port for agents + agentListenerPort: 50000 + # -- Host port to listen for agents + agentListenerHostPort: + # -- Node port to listen for agents + agentListenerNodePort: + + # ref: https://kubernetes.io/docs/concepts/services-networking/service/#traffic-policies + # -- Traffic Policy of for the agentListener service + agentListenerExternalTrafficPolicy: + # -- Allowed inbound IP for the agentListener service + agentListenerLoadBalancerSourceRanges: + - 0.0.0.0/0 + # -- Disabled agent protocols + disabledAgentProtocols: + - JNLP-connect + - JNLP2-connect + csrf: + defaultCrumbIssuer: + # -- Enable the default CSRF Crumb issuer + enabled: true + # -- Enable proxy compatibility + proxyCompatability: true + + # Kubernetes service type for the JNLP agent service + # agentListenerServiceType is the Kubernetes Service type for the JNLP agent service, + # either 'LoadBalancer', 'NodePort', or 'ClusterIP' + # Note if you set this to 'LoadBalancer', you *must* define annotations to secure it. By default, + # this will be an external load balancer and allowing inbound 0.0.0.0/0, a HUGE + # security risk: https://github.com/kubernetes/charts/issues/1341 + # -- Defines how to expose the agentListener service + agentListenerServiceType: "ClusterIP" + + # -- Annotations for the agentListener service + agentListenerServiceAnnotations: {} + + # Optionally, assign an IP to the LoadBalancer agentListenerService LoadBalancer + # GKE users: only regional static IPs will work for Service Load balancer. + # -- Static IP for the agentListener LoadBalancer + agentListenerLoadBalancerIP: + + # -- Whether legacy remoting security should be enabled + legacyRemotingSecurityEnabled: false + + # Example of a 'LoadBalancer'-type agent listener with annotations securing it + # agentListenerServiceType: LoadBalancer + # agentListenerServiceAnnotations: + # service.beta.kubernetes.io/aws-load-balancer-internal: "True" + # service.beta.kubernetes.io/load-balancer-source-ranges: "172.0.0.0/8, 10.0.0.0/8" + + # LoadBalancerSourcesRange is a list of allowed CIDR values, which are combined with ServicePort to + # set allowed inbound rules on the security group assigned to the controller load balancer + # -- Allowed inbound IP addresses + loadBalancerSourceRanges: + - 0.0.0.0/0 + + # -- Optionally assign a known public LB IP + loadBalancerIP: + + # Optionally configure a JMX port. This requires additional javaOpts, for example, + # javaOpts: > + # -Dcom.sun.management.jmxremote.port=4000 + # -Dcom.sun.management.jmxremote.authenticate=false + # -Dcom.sun.management.jmxremote.ssl=false + # jmxPort: 4000 + # -- Open a port, for JMX stats + jmxPort: + + # -- Optionally configure other ports to expose in the controller container + extraPorts: [] + # - name: BuildInfoProxy + # port: 9000 + # targetPort: 9010 (Optional: Use to explicitly set targetPort if different from port) + + # Plugins will be installed during Jenkins controller start + # -- List of Jenkins plugins to install. If you don't want to install plugins, set it to `false` + installPlugins: + - kubernetes:4253.v7700d91739e5 + - workflow-aggregator:600.vb_57cdd26fdd7 + - git:5.2.2 + - configuration-as-code:1810.v9b_c30a_249a_4c + + # If set to false, Jenkins will download the minimum required version of all dependencies. + # -- Download the minimum required version or latest version of all dependencies + installLatestPlugins: true + + # -- Set to true to download the latest version of any plugin that is requested to have the latest version + installLatestSpecifiedPlugins: false + + # -- List of plugins to install in addition to those listed in controller.installPlugins + additionalPlugins: [] + + # Without this; whenever the controller gets restarted (Evicted, etc.) it will fetch plugin updates that have the potential to cause breakage. + # Note that for this to work, `persistence.enabled` needs to be set to `true` + # -- Initialize only on first installation. Ensures plugins do not get updated inadvertently. Requires `persistence.enabled` to be set to `true` + initializeOnce: false + + # Enable to always override the installed plugins with the values of 'controller.installPlugins' on upgrade or redeployment. + # -- Overwrite installed plugins on start + overwritePlugins: false + + # Configures if plugins bundled with `controller.image` should be overwritten with the values of 'controller.installPlugins' on upgrade or redeployment. + # -- Overwrite plugins that are already installed in the controller image + overwritePluginsFromImage: true + + # Configures the restrictions for naming projects. Set this key to null or empty to skip it in the default config. + projectNamingStrategy: standard + + # Useful with ghprb plugin. The OWASP plugin is not installed by default, please update controller.installPlugins. + # -- Enable HTML parsing using OWASP Markup Formatter Plugin (antisamy-markup-formatter) + enableRawHtmlMarkupFormatter: false + + # This is ignored if enableRawHtmlMarkupFormatter is true + # -- Yaml of the markup formatter to use + markupFormatter: plainText + + # Used to approve a list of groovy functions in pipelines used the script-security plugin. Can be viewed under /scriptApproval + # -- List of groovy functions to approve + scriptApproval: [] + # - "method groovy.json.JsonSlurperClassic parseText java.lang.String" + # - "new groovy.json.JsonSlurperClassic" + + # -- Map of groovy init scripts to be executed during Jenkins controller start + initScripts: {} + # test: |- + # print 'adding global pipeline libraries, register properties, bootstrap jobs...' + # -- Name of the existing ConfigMap that contains init scripts + initConfigMap: + + # 'name' is a name of an existing secret in the same namespace as jenkins, + # 'keyName' is the name of one of the keys inside the current secret. + # the 'name' and 'keyName' are concatenated with a '-' in between, so for example: + # an existing secret "secret-credentials" and a key inside it named "github-password" should be used in JCasC as ${secret-credentials-github-password} + # 'name' and 'keyName' must be lowercase RFC 1123 label must consist of lower case alphanumeric characters or '-', + # and must start and end with an alphanumeric character (e.g. 'my-name', or '123-abc') + # existingSecret existing secret "secret-credentials" and a key inside it named "github-username" should be used in JCasC as ${github-username} + # When using existingSecret no need to specify the keyName under additionalExistingSecrets. + existingSecret: + + # -- List of additional existing secrets to mount + additionalExistingSecrets: [] + # ref: https://github.com/jenkinsci/configuration-as-code-plugin/blob/master/docs/features/secrets.adoc#kubernetes-secrets + # additionalExistingSecrets: + # - name: secret-name-1 + # keyName: username + # - name: secret-name-1 + # keyName: password + + # -- List of additional secrets to create and mount + additionalSecrets: [] + # ref: https://github.com/jenkinsci/configuration-as-code-plugin/blob/master/docs/features/secrets.adoc#kubernetes-secrets + # additionalSecrets: + # - name: nameOfSecret + # value: secretText + + # Generate SecretClaim resources to create Kubernetes secrets from HashiCorp Vault using kube-vault-controller. + # 'name' is the name of the secret that will be created in Kubernetes. The Jenkins fullname is prepended to this value. + # 'path' is the fully qualified path to the secret in Vault + # 'type' is an optional Kubernetes secret type. The default is 'Opaque' + # 'renew' is an optional secret renewal time in seconds + # -- List of `SecretClaim` resources to create + secretClaims: [] + # - name: secretName # required + # path: testPath # required + # type: kubernetes.io/tls # optional + # renew: 60 # optional + + # -- Name of default cloud configuration. + cloudName: "kubernetes" + + # Below is the implementation of Jenkins Configuration as Code. Add a key under configScripts for each configuration area, + # where each corresponds to a plugin or section of the UI. Each key (prior to | character) is just a label, and can be any value. + # Keys are only used to give the section a meaningful name. The only restriction is they may only contain RFC 1123 \ DNS label + # characters: lowercase letters, numbers, and hyphens. The keys become the name of a configuration yaml file on the controller in + # /var/jenkins_home/casc_configs (by default) and will be processed by the Configuration as Code Plugin. The lines after each | + # become the content of the configuration yaml file. The first line after this is a JCasC root element, e.g., jenkins, credentials, + # etc. Best reference is https:///configuration-as-code/reference. The example below creates a welcome message: + JCasC: + # -- Enables default Jenkins configuration via configuration as code plugin + defaultConfig: true + + # If true, the init container deletes all the plugin config files and Jenkins Config as Code overwrites any existing configuration + # -- Whether Jenkins Config as Code should overwrite any existing configuration + overwriteConfiguration: false + # -- Remote URLs for configuration files. + configUrls: [] + # - https://acme.org/jenkins.yaml + # -- List of Jenkins Config as Code scripts + configScripts: {} + # welcome-message: | + # jenkins: + # systemMessage: Welcome to our CI\CD server. This Jenkins is configured and managed 'as code'. + + # Allows adding to the top-level security JCasC section. For legacy purposes, by default, the chart includes apiToken configurations + # -- Jenkins Config as Code security-section + security: + apiToken: + creationOfLegacyTokenEnabled: false + tokenGenerationOnCreationEnabled: false + usageStatisticsEnabled: true + + # Ignored if securityRealm is defined in controller.JCasC.configScripts + # -- Jenkins Config as Code Security Realm-section + securityRealm: |- + local: + allowsSignup: false + enableCaptcha: false + users: + - id: "${chart-admin-username}" + name: "Jenkins Admin" + password: "${chart-admin-password}" + + # Ignored if authorizationStrategy is defined in controller.JCasC.configScripts + # -- Jenkins Config as Code Authorization Strategy-section + authorizationStrategy: |- + loggedInUsersCanDoAnything: + allowAnonymousRead: false + + # -- Annotations for the JCasC ConfigMap + configMapAnnotations: {} + + # -- Custom init-container specification in raw-yaml format + customInitContainers: [] + # - name: custom-init + # image: "alpine:3" + # imagePullPolicy: Always + # command: [ "uname", "-a" ] + + sidecars: + configAutoReload: + # If enabled: true, Jenkins Configuration as Code will be reloaded on-the-fly without a reboot. + # If false or not-specified, JCasC changes will cause a reboot and will only be applied at the subsequent start-up. + # Auto-reload uses the http:///reload-configuration-as-code endpoint to reapply config when changes to + # the configScripts are detected. + # -- Enables Jenkins Config as Code auto-reload + enabled: true + image: + # -- Registry for the image that triggers the reload + registry: docker.io + # -- Repository of the image that triggers the reload + repository: kiwigrid/k8s-sidecar + # -- Tag for the image that triggers the reload + tag: 1.27.5 + imagePullPolicy: IfNotPresent + resources: {} + # limits: + # cpu: 100m + # memory: 100Mi + # requests: + # cpu: 50m + # memory: 50Mi + # -- Enables additional volume mounts for the config auto-reload container + additionalVolumeMounts: [] + # - name: auto-reload-config + # mountPath: /var/config/logger + # - name: auto-reload-logs + # mountPath: /var/log/auto_reload + # -- Config auto-reload logging settings + logging: + # See default settings https://github.com/kiwigrid/k8s-sidecar/blob/master/src/logger.py + configuration: + # -- Enables custom log config utilizing using the settings below. + override: false + logLevel: INFO + formatter: JSON + logToConsole: true + logToFile: false + maxBytes: 1024 + backupCount: 3 + + # -- The scheme to use when connecting to the Jenkins configuration as code endpoint + scheme: http + # -- Skip TLS verification when connecting to the Jenkins configuration as code endpoint + skipTlsVerify: false + + # -- How many connection-related errors to retry on + reqRetryConnect: 10 + # -- How many seconds to wait before updating config-maps/secrets (sets METHOD=SLEEP on the sidecar) + sleepTime: + + # -- Environment variable sources for the Jenkins Config as Code auto-reload container + envFrom: [] + # -- Environment variables for the Jenkins Config as Code auto-reload container + env: {} + # - name: REQ_TIMEOUT + # value: "30" + + # SSH port value can be set to any unused TCP port. The default, 1044, is a non-standard SSH port that has been chosen at random. + # This is only used to reload JCasC config from the sidecar container running in the Jenkins controller pod. + # This TCP port will not be open in the pod (unless you specifically configure this), so Jenkins will not be + # accessible via SSH from outside the pod. Note if you use non-root pod privileges (runAsUser & fsGroup), + # this must be > 1024: + sshTcpPort: 1044 + # folder in the pod that should hold the collected dashboards: + folder: "/var/jenkins_home/casc_configs" + + # If specified, the sidecar will search for JCasC 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: + # -- Enable container security context + containerSecurityContext: + readOnlyRootFilesystem: true + allowPrivilegeEscalation: false + + # -- Configures additional sidecar container(s) for the Jenkins controller + additionalSidecarContainers: [] + ## The example below runs the client for https://smee.io as sidecar container next to Jenkins, + ## that allows triggering build behind a secure firewall. + ## https://jenkins.io/blog/2019/01/07/webhook-firewalls/#triggering-builds-with-webhooks-behind-a-secure-firewall + ## + ## Note: To use it you should go to https://smee.io/new and update the url to the generated one. + # - name: smee + # image: docker.io/twalter/smee-client:1.0.2 + # args: ["--port", "{{ .Values.controller.servicePort }}", "--path", "/github-webhook/", "--url", "https://smee.io/new"] + # resources: + # limits: + # cpu: 50m + # memory: 128Mi + # requests: + # cpu: 10m + # memory: 32Mi + + # -- Name of the Kubernetes scheduler to use + schedulerName: "" + + # ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#nodeselector + # -- Node labels for pod assignment + nodeSelector: {} + + # ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/#taints-and-tolerations-beta-feature + # -- Toleration labels for pod assignment + tolerations: [] + # -- Set TerminationGracePeriodSeconds + terminationGracePeriodSeconds: + # -- Set the termination message path + terminationMessagePath: + # -- Set the termination message policy + terminationMessagePolicy: + + # -- Affinity settings + affinity: {} + + # Leverage a priorityClass to ensure your pods survive resource shortages + # ref: https://kubernetes.io/docs/concepts/configuration/pod-priority-preemption/ + # -- The name of a `priorityClass` to apply to the controller pod + priorityClassName: + + # -- Annotations for controller pod + podAnnotations: {} + # -- Annotations for controller StatefulSet + statefulSetAnnotations: {} + + # ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#update-strategies + # -- Update strategy for StatefulSet + updateStrategy: {} + + # -- Topology spread constraints + topologySpreadConstraints: {} + + ingress: + # -- Enables ingress + enabled: false + + # Override for the default paths that map requests to the backend + # -- Override for the default Ingress paths + paths: [] + # - backend: + # serviceName: ssl-redirect + # servicePort: use-annotation + # - backend: + # serviceName: >- + # {{ template "jenkins.fullname" . }} + # # Don't use string here, use only integer value! + # servicePort: 8080 + + # For Kubernetes v1.14+, use 'networking.k8s.io/v1beta1' + # For Kubernetes v1.19+, use 'networking.k8s.io/v1' + # -- Ingress API version + apiVersion: "extensions/v1beta1" + # -- Ingress labels + labels: {} + # -- Ingress annotations + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + # 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 + + # Set this path to jenkinsUriPrefix above or use annotations to rewrite path + # -- Ingress path + path: + + # configures the hostname e.g. jenkins.example.com + # -- Ingress hostname + hostName: + # -- Hostname to serve assets from + resourceRootUrl: + # -- Ingress TLS configuration + tls: [] + # - secretName: jenkins.cluster.local + # hosts: + # - jenkins.cluster.local + + # often you want to have your controller all locked down and private, + # but you still want to get webhooks from your SCM + # A secondary ingress will let you expose different urls + # with a different configuration + secondaryingress: + enabled: false + # paths you want forwarded to the backend + # ex /github-webhook + paths: [] + # For Kubernetes v1.14+, use 'networking.k8s.io/v1beta1' + # For Kubernetes v1.19+, use 'networking.k8s.io/v1' + apiVersion: "extensions/v1beta1" + labels: {} + annotations: {} + # kubernetes.io/ingress.class: nginx + # kubernetes.io/tls-acme: "true" + # 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 + # configures the hostname e.g., jenkins-external.example.com + hostName: + tls: + # - secretName: jenkins-external.example.com + # hosts: + # - jenkins-external.example.com + + # If you're running on GKE and need to configure a backendconfig + # to finish ingress setup, use the following values. + # Docs: https://cloud.google.com/kubernetes-engine/docs/concepts/backendconfig + backendconfig: + # -- Enables backendconfig + enabled: false + # -- backendconfig API version + apiVersion: "extensions/v1beta1" + # -- backendconfig name + name: + # -- backendconfig labels + labels: {} + # -- backendconfig annotations + annotations: {} + # -- backendconfig spec + spec: {} + + # Openshift route + route: + # -- Enables openshift route + enabled: false + # -- Route labels + labels: {} + # -- Route annotations + annotations: {} + # -- Route path + path: + + # -- Allows for adding entries to Pod /etc/hosts + hostAliases: [] + # ref: https://kubernetes.io/docs/concepts/services-networking/add-entries-to-pod-etc-hosts-with-host-aliases/ + # hostAliases: + # - ip: 192.168.50.50 + # hostnames: + # - something.local + # - ip: 10.0.50.50 + # hostnames: + # - other.local + + # Expose Prometheus metrics + prometheus: + # If enabled, add the prometheus plugin to the list of plugins to install + # https://plugins.jenkins.io/prometheus + + # -- Enables prometheus service monitor + enabled: false + # -- Additional labels to add to the service monitor object + serviceMonitorAdditionalLabels: {} + # -- Set a custom namespace where to deploy ServiceMonitor resource + serviceMonitorNamespace: + # -- How often prometheus should scrape metrics + scrapeInterval: 60s + + # Defaults to the default endpoint used by the prometheus plugin + # -- The endpoint prometheus should get metrics from + scrapeEndpoint: /prometheus + + # See here: https://prometheus.io/docs/prometheus/latest/configuration/alerting_rules/ + # The `groups` root object is added by default, add the rule entries + # -- Array of prometheus alerting rules + alertingrules: [] + # -- Additional labels to add to the PrometheusRule object + alertingRulesAdditionalLabels: {} + # -- Set a custom namespace where to deploy PrometheusRule resource + prometheusRuleNamespace: "" + + # 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: [] + + googlePodMonitor: + # If enabled, It creates Google Managed Prometheus scraping config + enabled: false + # Set a custom namespace where to deploy PodMonitoring resource + # serviceMonitorNamespace: "" + scrapeInterval: 60s + # This is the default endpoint used by the prometheus plugin + scrapeEndpoint: /prometheus + + # -- Can be used to disable rendering controller test resources when using helm template + testEnabled: true + + httpsKeyStore: + # -- Enables HTTPS keystore on jenkins controller + enable: false + # -- Name of the secret that already has ssl keystore + jenkinsHttpsJksSecretName: "" + # -- Name of the key in the secret that already has ssl keystore + jenkinsHttpsJksSecretKey: "jenkins-jks-file" + # -- Name of the secret that contains the JKS password, if it is not in the same secret as the JKS file + jenkinsHttpsJksPasswordSecretName: "" + # -- Name of the key in the secret that contains the JKS password + jenkinsHttpsJksPasswordSecretKey: "https-jks-password" + disableSecretMount: false + + # When HTTPS keystore is enabled, servicePort and targetPort will be used as HTTPS port + # -- HTTP Port that Jenkins should listen to along with HTTPS, it also serves as the liveness and readiness probes port. + httpPort: 8081 + # -- Path of HTTPS keystore file + path: "/var/jenkins_keystore" + # -- Jenkins keystore filename which will appear under controller.httpsKeyStore.path + fileName: "keystore.jks" + # -- Jenkins keystore password + password: "password" + + # -- Base64 encoded Keystore content. Keystore must be converted to base64 then being pasted here + jenkinsKeyStoreBase64Encoded: + # Convert keystore.jks files content to base64 > $ cat keystore.jks | base64 +# /u3+7QAAAAIAAAABAAAAAQANamVua2luc2NpLmNvbQAAAW2r/b1ZAAAFATCCBP0wDgYKKwYBBAEq +# AhEBAQUABIIE6QbCqasvoHS0pSwYqSvdydMCB9t+VNfwhFIiiuAelJfO5sSe2SebJbtwHgLcRz1Z +# gMtWgOSFdl3bWSzA7vrW2LED52h+jXLYSWvZzuDuh8hYO85m10ikF6QR+dTi4jra0whIFDvq3pxe +# TnESxEsN+DvbZM3jA3qsjQJSeISNpDjO099dqQvHpnCn18lyk7J4TWJ8sOQQb1EM2zDAfAOSqA/x +# QuPEFl74DlY+5DIk6EBvpmWhaMSvXzWZACGA0sYqa157dq7O0AqmuLG/EI5EkHETO4CrtBW+yLcy +# 2dUCXOMA+j+NjM1BjrQkYE5vtSfNO6lFZcISyKo5pTFlcA7ut0Fx2nZ8GhHTn32CpeWwNcZBn1gR +# pZVt6DxVVkhTAkMLhR4rL2wGIi/1WRs23ZOLGKtyDNvDHnQyDiQEoJGy9nAthA8aNHa3cfdF10vB +# Drb19vtpFHmpvKEEhpk2EBRF4fTi644Fuhu2Ied6118AlaPvEea+n6G4vBz+8RWuVCmZjLU+7h8l +# Hy3/WdUPoIL5eW7Kz+hS+sRTFzfu9C48dMkQH3a6f3wSY+mufizNF9U298r98TnYy+PfDJK0bstG +# Ph6yPWx8DGXKQBwrhWJWXI6JwZDeC5Ny+l8p1SypTmAjpIaSW3ge+KgcL6Wtt1R5hUV1ajVwVSUi +# HF/FachKqPqyLJFZTGjNrxnmNYpt8P1d5JTvJfmfr55Su/P9n7kcyWp7zMcb2Q5nlXt4tWogOHLI +# OzEWKCacbFfVHE+PpdrcvCVZMDzFogIq5EqGTOZe2poPpBVE+1y9mf5+TXBegy5HToLWvmfmJNTO +# NCDuBjgLs2tdw2yMPm4YEr57PnMX5gGTC3f2ZihXCIJDCRCdQ9sVBOjIQbOCzxFXkVITo0BAZhCi +# Yz61wt3Ud8e//zhXWCkCsSV+IZCxxPzhEFd+RFVjW0Nm9hsb2FgAhkXCjsGROgoleYgaZJWvQaAg +# UyBzMmKDPKTllBHyE3Gy1ehBNGPgEBChf17/9M+j8pcm1OmlM434ctWQ4qW7RU56//yq1soFY0Te +# fu2ei03a6m68fYuW6s7XEEK58QisJWRAvEbpwu/eyqfs7PsQ+zSgJHyk2rO95IxdMtEESb2GRuoi +# Bs+AHNdYFTAi+GBWw9dvEgqQ0Mpv0//6bBE/Fb4d7b7f56uUNnnE7mFnjGmGQN+MvC62pfwfvJTT +# EkT1iZ9kjM9FprTFWXT4UmO3XTvesGeE50sV9YPm71X4DCQwc4KE8vyuwj0s6oMNAUACW2ClU9QQ +# y0tRpaF1tzs4N42Q5zl0TzWxbCCjAtC3u6xf+c8MCGrr7DzNhm42LOQiHTa4MwX4x96q7235oiAU +# iQqSI/hyF5yLpWw4etyUvsx2/0/0wkuTU1FozbLoCWJEWcPS7QadMrRRISxHf0YobIeQyz34regl +# t1qSQ3dCU9D6AHLgX6kqllx4X0fnFq7LtfN7fA2itW26v+kAT2QFZ3qZhINGfofCja/pITC1uNAZ +# gsJaTMcQ600krj/ynoxnjT+n1gmeqThac6/Mi3YlVeRtaxI2InL82ZuD+w/dfY9OpPssQjy3xiQa +# jPuaMWXRxz/sS9syOoGVH7XBwKrWpQcpchozWJt40QV5DslJkclcr8aC2AGlzuJMTdEgz1eqV0+H +# bAXG9HRHN/0eJTn1/QAAAAEABVguNTA5AAADjzCCA4swggJzAhRGqVxH4HTLYPGO4rzHcCPeGDKn +# xTANBgkqhkiG9w0BAQsFADCBgTELMAkGA1UEBhMCY2ExEDAOBgNVBAgMB29udGFyaW8xEDAOBgNV +# BAcMB3Rvcm9udG8xFDASBgNVBAoMC2plbmtpbnN0ZXN0MRkwFwYDVQQDDBBqZW5raW5zdGVzdC5p +# bmZvMR0wGwYJKoZIhvcNAQkBFg50ZXN0QHRlc3QuaW5mbzAeFw0xOTEwMDgxNTI5NTVaFw0xOTEx +# MDcxNTI5NTVaMIGBMQswCQYDVQQGEwJjYTEQMA4GA1UECAwHb250YXJpbzEQMA4GA1UEBwwHdG9y +# b250bzEUMBIGA1UECgwLamVua2luc3Rlc3QxGTAXBgNVBAMMEGplbmtpbnN0ZXN0LmluZm8xHTAb +# BgkqhkiG9w0BCQEWDnRlc3RAdGVzdC5pbmZvMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKC +# AQEA02q352JTHGvROMBhSHvSv+vnoOTDKSTz2aLQn0tYrIRqRo+8bfmMjXuhkwZPSnCpvUGNAJ+w +# Jrt/dqMoYUjCBkjylD/qHmnXN5EwS1cMg1Djh65gi5JJLFJ7eNcoSsr/0AJ+TweIal1jJSP3t3PF +# 9Uv21gm6xdm7HnNK66WpUUXLDTKaIs/jtagVY1bLOo9oEVeLN4nT2CYWztpMvdCyEDUzgEdDbmrP +# F5nKUPK5hrFqo1Dc5rUI4ZshL3Lpv398aMxv6n2adQvuL++URMEbXXBhxOrT6rCtYzbcR5fkwS9i +# d3Br45CoWOQro02JAepoU0MQKY5+xQ4Bq9Q7tB9BAwIDAQABMA0GCSqGSIb3DQEBCwUAA4IBAQAe +# 4xc+mSvKkrKBHg9/zpkWgZUiOp4ENJCi8H4tea/PCM439v6y/kfjT/okOokFvX8N5aa1OSz2Vsrl +# m8kjIc6hiA7bKzT6lb0EyjUShFFZ5jmGVP4S7/hviDvgB5yEQxOPpumkdRP513YnEGj/o9Pazi5h +# /MwpRxxazoda9r45kqQpyG+XoM4pB+Fd3JzMc4FUGxfVPxJU4jLawnJJiZ3vqiSyaB0YyUL+Er1Q +# 6NnqtR4gEBF0ZVlQmkycFvD4EC2boP943dLqNUvop+4R3SM1QMM6P5u8iTXtHd/VN4MwMyy1wtog +# hYAzODo1Jt59pcqqKJEas0C/lFJEB3frw4ImNx5fNlJYOpx+ijfQs9m39CevDq0= + +agent: + # -- Enable Kubernetes plugin jnlp-agent podTemplate + enabled: true + # -- The name of the pod template to use for providing default values + defaultsProviderTemplate: "" + + # For connecting to the Jenkins controller + # -- Overrides the Kubernetes Jenkins URL + jenkinsUrl: + + # connects to the specified host and port, instead of connecting directly to the Jenkins controller + # -- Overrides the Kubernetes Jenkins tunnel + jenkinsTunnel: + # -- The connection timeout in seconds for connections to Kubernetes API. The minimum value is 5 + kubernetesConnectTimeout: 5 + # -- The read timeout in seconds for connections to Kubernetes API. The minimum value is 15 + kubernetesReadTimeout: 15 + # -- The maximum concurrent connections to Kubernetes API + maxRequestsPerHostStr: "32" + # -- Time in minutes after which the Kubernetes cloud plugin will clean up an idle worker that has not already terminated + retentionTimeout: 5 + # -- Seconds to wait for pod to be running + waitForPodSec: 600 + # -- Namespace in which the Kubernetes agents should be launched + namespace: + # -- Custom Pod labels (an object with `label-key: label-value` pairs) + podLabels: {} + # -- Custom registry used to pull the agent jnlp image from + jnlpregistry: + image: + # -- Repository to pull the agent jnlp image from + repository: "jenkins/inbound-agent" + # -- Tag of the image to pull + tag: "3256.v88a_f6e922152-1" + # -- Configure working directory for default agent + workingDir: "/home/jenkins/agent" + nodeUsageMode: "NORMAL" + # -- Append Jenkins labels to the agent + customJenkinsLabels: [] + # -- Name of the secret to be used to pull the image + imagePullSecretName: + componentName: "jenkins-agent" + # -- Enables agent communication via websockets + websocket: false + directConnection: false + # -- Agent privileged container + privileged: false + # -- Configure container user + runAsUser: + # -- Configure container group + runAsGroup: + # -- Enables the agent to use the host network + hostNetworking: false + # -- Resources allocation (Requests and Limits) + resources: + requests: + cpu: "512m" + memory: "512Mi" + # ephemeralStorage: + limits: + cpu: "512m" + memory: "512Mi" + # ephemeralStorage: + livenessProbe: {} +# execArgs: "cat /tmp/healthy" +# failureThreshold: 3 +# initialDelaySeconds: 0 +# periodSeconds: 10 +# successThreshold: 1 +# timeoutSeconds: 1 + + # You may want to change this to true while testing a new image + # -- Always pull agent container image before build + alwaysPullImage: false + # When using Pod Security Admission in the Agents namespace with the restricted Pod Security Standard, + # the jnlp container cannot be scheduled without overriding its container definition with a securityContext. + # This option allows to automatically inject in the jnlp container a securityContext + # that is suitable for the use of the restricted Pod Security Standard. + # -- Set a restricted securityContext on jnlp containers + restrictedPssSecurityContext: false + # Controls how agent pods are retained after the Jenkins build completes + # Possible values: Always, Never, OnFailure + podRetention: "Never" + # Disable if you do not want the Yaml the agent pod template to show up + # in the job Console Output. This can be helpful for either security reasons + # or simply to clean up the output to make it easier to read. + showRawYaml: true + + # You can define the volumes that you want to mount for this container + # Allowed types are: ConfigMap, EmptyDir, EphemeralVolume, HostPath, Nfs, PVC, Secret + # Configure the attributes as they appear in the corresponding Java class for that type + # https://github.com/jenkinsci/kubernetes-plugin/tree/master/src/main/java/org/csanchez/jenkins/plugins/kubernetes/volumes + # -- Additional volumes + volumes: [] + # - type: ConfigMap + # configMapName: myconfigmap + # mountPath: /var/myapp/myconfigmap + # - type: EmptyDir + # mountPath: /var/myapp/myemptydir + # memory: false + # - type: EphemeralVolume + # mountPath: /var/myapp/myephemeralvolume + # accessModes: ReadWriteOnce + # requestsSize: 10Gi + # storageClassName: mystorageclass + # - type: HostPath + # hostPath: /var/lib/containers + # mountPath: /var/myapp/myhostpath + # - type: Nfs + # mountPath: /var/myapp/mynfs + # readOnly: false + # serverAddress: "192.0.2.0" + # serverPath: /var/lib/containers + # - type: PVC + # claimName: mypvc + # mountPath: /var/myapp/mypvc + # readOnly: false + # - type: Secret + # defaultMode: "600" + # mountPath: /var/myapp/mysecret + # secretName: mysecret + # Pod-wide environment, these vars are visible to any container in the agent pod + + # You can define the workspaceVolume that you want to mount for this container + # Allowed types are: DynamicPVC, EmptyDir, EphemeralVolume, HostPath, Nfs, PVC + # Configure the attributes as they appear in the corresponding Java class for that type + # https://github.com/jenkinsci/kubernetes-plugin/tree/master/src/main/java/org/csanchez/jenkins/plugins/kubernetes/volumes/workspace + # -- Workspace volume (defaults to EmptyDir) + workspaceVolume: {} + ## DynamicPVC example + # - type: DynamicPVC + # configMapName: myconfigmap + ## EmptyDir example + # - type: EmptyDir + # memory: false + ## EphemeralVolume example + # - type: EphemeralVolume + # accessModes: ReadWriteOnce + # requestsSize: 10Gi + # storageClassName: mystorageclass + ## HostPath example + # - type: HostPath + # hostPath: /var/lib/containers + ## NFS example + # - type: Nfs + # readOnly: false + # serverAddress: "192.0.2.0" + # serverPath: /var/lib/containers + ## PVC example + # - type: PVC + # claimName: mypvc + # readOnly: false + + # Pod-wide environment, these vars are visible to any container in the agent pod + # -- Environment variables for the agent Pod + envVars: [] + # - name: PATH + # value: /usr/local/bin + # -- Mount a secret as environment variable + secretEnvVars: [] + # - key: PATH + # optional: false # default: false + # secretKey: MY-K8S-PATH + # secretName: my-k8s-secret + + # -- Node labels for pod assignment + nodeSelector: {} + # Key Value selectors. Ex: + # nodeSelector + # jenkins-agent: v1 + + # -- Command to execute when side container starts + command: + # -- Arguments passed to command to execute + args: "${computer.jnlpmac} ${computer.name}" + # -- Side container name + sideContainerName: "jnlp" + + # Doesn't allocate pseudo TTY by default + # -- Allocate pseudo tty to the side container + TTYEnabled: false + # -- Max number of agents to launch + containerCap: 10 + # -- Agent Pod base name + podName: "default" + + # -- Allows the Pod to remain active for reuse until the configured number of minutes has passed since the last step was executed on it + idleMinutes: 0 + + + # The raw yaml of a Pod API Object, for example, this allows usage of toleration for agent pods. + # https://github.com/jenkinsci/kubernetes-plugin#using-yaml-to-define-pod-templates + # https://kubernetes.io/docs/concepts/configuration/taint-and-toleration/ + # -- The raw yaml of a Pod API Object to merge into the agent spec + yamlTemplate: "" + # yamlTemplate: |- + # apiVersion: v1 + # kind: Pod + # spec: + # tolerations: + # - key: "key" + # operator: "Equal" + # value: "value" + + # -- Defines how the raw yaml field gets merged with yaml definitions from inherited pod templates. Possible values: "merge" or "override" + yamlMergeStrategy: "override" + # -- Controls whether the defined yaml merge strategy will be inherited if another defined pod template is configured to inherit from the current one + inheritYamlMergeStrategy: false + # -- Timeout in seconds for an agent to be online + connectTimeout: 100 + # -- Annotations to apply to the pod + annotations: {} + + # Containers specified here are added to all agents. Set key empty to remove container from additional agents. + # -- Add additional containers to the agents + additionalContainers: [] + # - sideContainerName: dind + # image: + # repository: docker + # tag: dind + # command: dockerd-entrypoint.sh + # args: "" + # privileged: true + # resources: + # requests: + # cpu: 500m + # memory: 1Gi + # limits: + # cpu: 1 + # memory: 2Gi + + # Useful when configuring agents only with the podTemplates value, since the default podTemplate populated by values mentioned above will be excluded in the rendered template. + # -- Disable the default Jenkins Agent configuration + disableDefaultAgent: false + + # Below is the implementation of custom pod templates for the default configured kubernetes cloud. + # Add a key under podTemplates for each pod template. Each key (prior to | character) is just a label, and can be any value. + # Keys are only used to give the pod template a meaningful name. The only restriction is they may only contain RFC 1123 \ DNS label + # characters: lowercase letters, numbers, and hyphens. Each pod template can contain multiple containers. + # For this pod templates configuration to be loaded, the following values must be set: + # controller.JCasC.defaultConfig: true + # Best reference is https:///configuration-as-code/reference#Cloud-kubernetes. The example below creates a python pod template. + # -- Configures extra pod templates for the default kubernetes cloud + podTemplates: {} + # python: | + # - name: python + # label: jenkins-python + # serviceAccount: jenkins + # containers: + # - name: python + # image: python:3 + # command: "/bin/sh -c" + # args: "cat" + # ttyEnabled: true + # privileged: true + # resourceRequestCpu: "400m" + # resourceRequestMemory: "512Mi" + # resourceLimitCpu: "1" + # resourceLimitMemory: "1024Mi" + +# Inherits all values from `agent` so you only need to specify values which differ +# -- Configure additional +additionalAgents: {} +# maven: +# podName: maven +# customJenkinsLabels: maven +# # An example of overriding the jnlp container +# # sideContainerName: jnlp +# image: +# repository: jenkins/jnlp-agent-maven +# tag: latest +# python: +# podName: python +# customJenkinsLabels: python +# sideContainerName: python +# image: +# repository: python +# tag: "3" +# command: "/bin/sh -c" +# args: "cat" +# TTYEnabled: true + +# Here you can add additional clouds +# They inherit all values from the default cloud (including the main agent), so +# you only need to specify values which differ. If you want to override +# default additionalAgents with the additionalClouds.additionalAgents set +# additionalAgentsOverride to `true`. +additionalClouds: {} +# remote-cloud-1: +# kubernetesURL: https://api.remote-cloud.com +# additionalAgentsOverride: true +# additionalAgents: +# maven-2: +# podName: maven-2 +# customJenkinsLabels: maven +# # An example of overriding the jnlp container +# # sideContainerName: jnlp +# image: +# repository: jenkins/jnlp-agent-maven +# tag: latest +# namespace: my-other-maven-namespace +# remote-cloud-2: +# kubernetesURL: https://api.remote-cloud.com + +persistence: + # -- Enable the use of a Jenkins PVC + enabled: true + + # A manually managed Persistent Volume and Claim + # Requires persistence.enabled: true + # If defined, PVC must be created manually before volume will be bound + # -- Provide the name of a PVC + existingClaim: + + # jenkins 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) + # -- Storage class for the PVC + storageClass: + # -- Annotations for the PVC + annotations: {} + # -- Labels for the PVC + labels: {} + # -- The PVC access mode + accessMode: "ReadWriteOnce" + # -- The size of the PVC + size: "8Gi" + + # ref: https://kubernetes.io/docs/concepts/storage/volume-pvc-datasource/ + # -- Existing data source to clone PVC from + dataSource: {} + # name: PVC-NAME + # kind: PersistentVolumeClaim + + # -- SubPath for jenkins-home mount + subPath: + # -- Additional volumes + volumes: [] + # - name: nothing + # emptyDir: {} + + # -- Additional mounts + mounts: [] + # - mountPath: /var/nothing + # name: nothing + # readOnly: true + +networkPolicy: + # -- Enable the creation of NetworkPolicy resources + enabled: false + + # For Kubernetes v1.4, v1.5 and v1.6, use 'extensions/v1beta1' + # For Kubernetes v1.7, use 'networking.k8s.io/v1' + # -- NetworkPolicy ApiVersion + apiVersion: networking.k8s.io/v1 + # You can allow agents to connect from both within the cluster (from within specific/all namespaces) AND/OR from a given external IP range + internalAgents: + # -- Allow internal agents (from the same cluster) to connect to controller. Agent pods will be filtered based on PodLabels + allowed: true + # -- A map of labels (keys/values) that agent pods must have to be able to connect to controller + podLabels: {} + # -- A map of labels (keys/values) that agents namespaces must have to be able to connect to controller + namespaceLabels: {} + # project: myproject + externalAgents: + # -- The IP range from which external agents are allowed to connect to controller, i.e., 172.17.0.0/16 + ipCIDR: + # -- A list of IP sub-ranges to be excluded from the allowlisted IP range + except: [] + # - 172.17.1.0/24 + +## Install Default RBAC roles and bindings +rbac: + # -- Whether RBAC resources are created + create: true + # -- Whether the Jenkins service account should be able to read Kubernetes secrets + readSecrets: false + +serviceAccount: + # -- Configures if a ServiceAccount with this name should be created + create: true + + # The name of the ServiceAccount is autogenerated by default + # -- The name of the ServiceAccount to be used by access-controlled resources + name: + # -- Configures annotations for the ServiceAccount + annotations: {} + # -- Configures extra labels for the ServiceAccount + extraLabels: {} + # -- Controller ServiceAccount image pull secret + imagePullSecretName: + + +serviceAccountAgent: + # -- Configures if an agent ServiceAccount should be created + create: false + + # If not set and create is true, a name is generated using the fullname template + # -- The name of the agent ServiceAccount to be used by access-controlled resources + name: + # -- Configures annotations for the agent ServiceAccount + annotations: {} + # -- Configures extra labels for the agent ServiceAccount + extraLabels: {} + # -- Agent ServiceAccount image pull secret + imagePullSecretName: + +# -- Checks if any deprecated values are used +checkDeprecation: true + +awsSecurityGroupPolicies: + enabled: false + policies: + - name: "" + securityGroupIds: [] + podSelector: {} + +# Here you can configure unit tests values when executing the helm unittest in the CONTRIBUTING.md +helmtest: + # A testing framework for bash + bats: + # Bash Automated Testing System (BATS) + image: + # -- Registry of the image used to test the framework + registry: "docker.io" + # -- Repository of the image used to test the framework + repository: "bats/bats" + # -- Tag of the image to test the framework + tag: "1.11.0" diff --git a/index.yaml b/index.yaml index f98e40710..a78dfc205 100644 --- a/index.yaml +++ b/index.yaml @@ -3635,6 +3635,38 @@ entries: urls: - assets/asserts/asserts-1.40.0.tgz version: 1.40.0 + cerbos: + - annotations: + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Cerbos + catalog.cattle.io/kube-version: '>=1.23.0-0' + catalog.cattle.io/release-name: cerbos + apiVersion: v2 + appVersion: 0.37.0 + created: "2024-07-12T16:35:38.907483922Z" + description: A Helm chart to deploy Cerbos. Cerbos is an open core, language agnostic, + scalable solution that makes user permissions and authorization simple to implement + and manage by writing context-aware access control policies for your application + resources. + digest: 1e4631c850479ddbee5cea93d2bdea69feb1f859f6adce620af64abbdaef73a9 + home: https://cerbos.dev + icon: file://assets/icons/cerbos.png + keywords: + - abac + - authorization + - developer tools + - policies + - rbac + - security + kubeVersion: '>=1.23.0-0' + maintainers: + - email: help+helm@cerbos.dev + name: Cerbos authors + name: cerbos + type: application + urls: + - assets/cerbos/cerbos-0.37.0.tgz + version: 0.37.0 cf-runtime: - annotations: catalog.cattle.io/certified: partner @@ -13610,6 +13642,64 @@ entries: - assets/intel/intel-device-plugins-sgx-0.26.1.tgz version: 0.26.1 jenkins: + - annotations: + artifacthub.io/category: integration-delivery + artifacthub.io/changes: | + - Update `docker.io/kiwigrid/k8s-sidecar` to version `1.27.5` + artifacthub.io/images: | + - name: jenkins + image: docker.io/jenkins/jenkins:2.452.3-jdk17 + - name: k8s-sidecar + image: docker.io/kiwigrid/k8s-sidecar:1.27.5 + - name: inbound-agent + image: jenkins/inbound-agent:3256.v88a_f6e922152-1 + artifacthub.io/license: Apache-2.0 + artifacthub.io/links: | + - name: Chart Source + url: https://github.com/jenkinsci/helm-charts/tree/main/charts/jenkins + - name: Jenkins + url: https://www.jenkins.io/ + - name: support + url: https://github.com/jenkinsci/helm-charts/issues + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Jenkins + catalog.cattle.io/kube-version: '>=1.14-0' + catalog.cattle.io/release-name: jenkins + apiVersion: v2 + appVersion: 2.452.3 + created: "2024-07-12T16:35:39.79439551Z" + description: 'Jenkins - Build great things at any scale! As the leading open source + automation server, Jenkins provides over 1800 plugins to support building, deploying + and automating any project. ' + digest: 46be97fd7ad65923a10ba0b26b5b8a99e22b8ae0accd029811e00a6591cb0991 + home: https://www.jenkins.io/ + icon: file://assets/icons/jenkins.svg + keywords: + - jenkins + - ci + - devops + kubeVersion: '>=1.14-0' + maintainers: + - email: maor.friedman@redhat.com + name: maorfr + - email: mail@torstenwalter.de + name: torstenwalter + - email: garridomota@gmail.com + name: mogaal + - email: wmcdona89@gmail.com + name: wmcdona89 + - email: timjacomb1@gmail.com + name: timja + name: jenkins + sources: + - https://github.com/jenkinsci/jenkins + - https://github.com/jenkinsci/docker-inbound-agent + - https://github.com/maorfr/kube-tasks + - https://github.com/jenkinsci/configuration-as-code-plugin + type: application + urls: + - assets/jenkins/jenkins-5.4.2.tgz + version: 5.4.2 - annotations: artifacthub.io/category: integration-delivery artifacthub.io/changes: | @@ -21628,6 +21718,54 @@ entries: - assets/loft/loft-3.2.0.tgz version: 3.2.0 microgateway: + - annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.3/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: microgateway + charts.openshift.io/name: Airlock Microgateway + apiVersion: v2 + appVersion: 4.3.0 + created: "2024-07-12T16:35:38.658140339Z" + description: A Helm chart for deploying the Airlock Microgateway + digest: 454dfebf2f42533f83f9c2ff3328743722ec32da5e0904595d49a35ace2171b7 + home: https://www.airlock.com/en/microgateway + icon: file://assets/icons/microgateway.svg + keywords: + - WAF + - Web Application Firewall + - WAAP + - Web Application and API protection + - OWASP + - Airlock + - Microgateway + - Security + - Filtering + - DevSecOps + - shift left + - control plane + - Operator + kubeVersion: '>=1.25.0-0' + maintainers: + - email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ + name: microgateway + sources: + - https://github.com/airlock/microgateway + type: application + urls: + - assets/airlock/microgateway-4.3.0.tgz + version: 4.3.0 - annotations: artifacthub.io/category: security artifacthub.io/license: MIT @@ -21677,6 +21815,53 @@ entries: - assets/airlock/microgateway-4.2.3.tgz version: 4.2.3 microgateway-cni: + - annotations: + artifacthub.io/category: security + artifacthub.io/license: MIT + artifacthub.io/links: | + - name: Airlock Microgateway Documentation + url: https://docs.airlock.com/microgateway/4.3/ + - name: Airlock Microgateway Labs + url: https://play.instruqt.com/airlock/invite/hyi9fy4b4jzc?icp_referrer=artifacthub.io + - name: Airlock Microgateway Forum + url: https://forum.airlock.com/ + catalog.cattle.io/certified: partner + catalog.cattle.io/display-name: Airlock Microgateway CNI + catalog.cattle.io/kube-version: '>=1.25.0-0' + catalog.cattle.io/release-name: microgateway-cni + charts.openshift.io/name: Airlock Microgateway CNI + apiVersion: v2 + appVersion: 4.3.0 + created: "2024-07-12T16:35:38.65933805Z" + description: A Helm chart for deploying the Airlock Microgateway CNI plugin + digest: 25eb9eca9c81d6bf0e8803dc9438c1715733f122dbf2883074dc04a9ef8e9211 + home: https://www.airlock.com/en/microgateway + icon: file://assets/icons/microgateway-cni.svg + keywords: + - WAF + - Web Application Firewall + - WAAP + - Web Application and API protection + - OWASP + - Airlock + - Microgateway + - Security + - Filtering + - DevSecOps + - shift left + - CNI + kubeVersion: '>=1.25.0-0' + maintainers: + - email: support@airlock.com + name: Airlock + url: https://www.airlock.com/ + name: microgateway-cni + sources: + - https://github.com/airlock/microgateway + type: application + urls: + - assets/airlock/microgateway-cni-4.3.0.tgz + version: 4.3.0 - annotations: artifacthub.io/category: security artifacthub.io/license: MIT @@ -36090,4 +36275,4 @@ entries: urls: - assets/netfoundry/ziti-host-1.5.1.tgz version: 1.5.1 -generated: "2024-07-12T00:39:14.693074443Z" +generated: "2024-07-12T16:35:38.654385766Z"