mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-05-17 18:25:26 +00:00
Extend example with components
Describe in more detail the solution with stock kustomize variants to better showcase the benefits of kustomize components. Plus, revamp certain parts of the example with minor fixes.
This commit is contained in:
committed by
Alex Pyrgiotis
parent
6063a6bde8
commit
0152dbb0dc
@@ -27,7 +27,6 @@ want to deploy it with some features enabled and others not.
|
||||
Here's a matrix with the deployments of this application and the features
|
||||
enabled for each one:
|
||||
|
||||
|
||||
| | External DB | LDAP | reCAPTCHA |
|
||||
|------------|:------------------:|:------------------:|:------------------:|
|
||||
| Community | :heavy_check_mark: | | :heavy_check_mark: |
|
||||
@@ -37,21 +36,388 @@ enabled for each one:
|
||||
So, you want to make it easy to deploy your application in any of the above
|
||||
three environments. This seems like a work for [variants], so you try to create
|
||||
three overlays; a `community/`, an `enterprise/` and a `dev/` overlay, that each
|
||||
has the appropriate features. However, there are two issues:
|
||||
provides the appropriate features for their audience, i.e., public, customers and
|
||||
developers, respectfully.
|
||||
|
||||
1. The external DB feature is repeated in the `community/` and `enterprise/`
|
||||
overlays. The rest of the features are optionally repeated on the `dev/`
|
||||
overlay as well.
|
||||
2. The `dev/` overlay is dynamic, and uncommenting many lines of YAML to enable
|
||||
a single feature is cumbersome.
|
||||
## Variants example
|
||||
|
||||
Ideally, you want to move each feature under a separate overlay, and enable
|
||||
them per deployment. Enter components.
|
||||
Here's the common and most simplistic approach to solve this problem. As we will
|
||||
soon see, this approach does not scale well in more complex scenarios. However,
|
||||
it will help you get a better grasp of the problem we are about to tackle and
|
||||
demonstrate where there is room for improvement.
|
||||
|
||||
First, define a place to work:
|
||||
|
||||
```shell
|
||||
DEMO_HOME=$(mktemp -d)
|
||||
```
|
||||
|
||||
Define a common **base** that has a `Deployment` and a simple `ConfigMap`, that
|
||||
is mounted on the application's container.
|
||||
|
||||
```shell
|
||||
BASE=$DEMO_HOME/base
|
||||
mkdir $BASE
|
||||
|
||||
cat <<EOF >$BASE/kustomization.yaml
|
||||
resources:
|
||||
- deployment.yaml
|
||||
|
||||
configMapGenerator:
|
||||
- name: conf
|
||||
literals:
|
||||
- main.conf=|
|
||||
color=cornflower_blue
|
||||
log_level=info
|
||||
EOF
|
||||
|
||||
cat <<EOF >$BASE/deployment.yaml
|
||||
apiVersion: apps/v1
|
||||
kind: Deployment
|
||||
metadata:
|
||||
name: example
|
||||
spec:
|
||||
template:
|
||||
spec:
|
||||
containers:
|
||||
- name: example
|
||||
image: example:1.0
|
||||
volumeMounts:
|
||||
- name: conf
|
||||
mountPath: /etc/config
|
||||
volumes:
|
||||
- name: conf
|
||||
configMap:
|
||||
name: conf
|
||||
EOF
|
||||
```
|
||||
|
||||
Define a **community** overlay that:
|
||||
|
||||
- generates `Secrets` for external DB's password and reCAPTCHA's keys
|
||||
- patches the `ConfigMap` of the common base with configurations for external DB
|
||||
and reCAPTCHA
|
||||
- patches the `Deployment` of the common base to mount the generated `Secrets`
|
||||
for external DB and reCAPTCHA
|
||||
|
||||
```shell
|
||||
COMMUNITY=$DEMO_HOME/overlays/community
|
||||
mkdir -p $COMMUNITY
|
||||
|
||||
cat <<EOF >$COMMUNITY/kustomization.yaml
|
||||
kind: Kustomization
|
||||
|
||||
resources:
|
||||
- ../../base
|
||||
|
||||
secretGenerator:
|
||||
- name: dbpass
|
||||
files:
|
||||
- dbpass.txt
|
||||
- name: recaptcha
|
||||
files:
|
||||
- site_key.txt
|
||||
- secret_key.txt
|
||||
|
||||
patches:
|
||||
- configmap.yaml
|
||||
|
||||
patches:
|
||||
- target:
|
||||
group: apps
|
||||
version: v1
|
||||
kind: Deployment
|
||||
name: example
|
||||
path: deployment.yaml
|
||||
EOF
|
||||
|
||||
cat <<EOF >$COMMUNITY/deployment.yaml
|
||||
- op: add
|
||||
path: /spec/template/spec/volumes/0
|
||||
value:
|
||||
name: dbpass
|
||||
secret:
|
||||
secretName: dbpass
|
||||
- op: add
|
||||
path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
value:
|
||||
mountPath: /var/run/secrets/db/
|
||||
name: dbpass
|
||||
- op: add
|
||||
path: /spec/template/spec/volumes/0
|
||||
value:
|
||||
name: recaptcha
|
||||
secret:
|
||||
secretName: recaptcha
|
||||
- op: add
|
||||
path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
value:
|
||||
mountPath: /var/run/secrets/recaptcha/
|
||||
name: recaptcha
|
||||
EOF
|
||||
|
||||
cat <<EOF >$COMMUNITY/configmap.yaml
|
||||
apiVersion: v1
|
||||
kind: ConfigMap
|
||||
metadata:
|
||||
name: conf
|
||||
data:
|
||||
db.conf: |
|
||||
endpoint=127.0.0.1:1234
|
||||
name=app
|
||||
user=admin
|
||||
pass=/var/run/secrets/db/dbpass.txt
|
||||
recaptcha.conf: |
|
||||
enabled=true
|
||||
site_key=/var/run/secrets/recaptcha/site_key.txt
|
||||
secret_key=/var/run/secrets/recaptcha/secret_key.txt
|
||||
EOF
|
||||
```
|
||||
|
||||
Define a **enterprise** overlay that:
|
||||
|
||||
- generates `Secrets` for LDAP's password and external DB's password
|
||||
- patches the `ConfigMap` of the common base with configurations for LDAP and
|
||||
external DB
|
||||
- patches the `Deployment` of the common base to mount the generated `Secrets`
|
||||
for LDAP and external DB
|
||||
|
||||
```shell
|
||||
ENTERPRISE=$DEMO_HOME/overlays/enterprise
|
||||
mkdir -p $ENTERPRISE
|
||||
|
||||
cat <<EOF >$ENTERPRISE/kustomization.yaml
|
||||
kind: Kustomization
|
||||
|
||||
resources:
|
||||
- ../../base
|
||||
|
||||
secretGenerator:
|
||||
- name: ldappass
|
||||
files:
|
||||
- ldappass.txt
|
||||
- name: dbpass
|
||||
files:
|
||||
- dbpass.txt
|
||||
|
||||
patches:
|
||||
- configmap.yaml
|
||||
|
||||
patches:
|
||||
- target:
|
||||
group: apps
|
||||
version: v1
|
||||
kind: Deployment
|
||||
name: example
|
||||
path: deployment.yaml
|
||||
EOF
|
||||
|
||||
cat <<EOF >$ENTERPRISE/deployment.yaml
|
||||
- op: add
|
||||
path: /spec/template/spec/volumes/0
|
||||
value:
|
||||
name: dbpass
|
||||
secret:
|
||||
secretName: dbpass
|
||||
- op: add
|
||||
path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
value:
|
||||
mountPath: /var/run/secrets/db/
|
||||
name: dbpass
|
||||
- op: add
|
||||
path: /spec/template/spec/volumes/0
|
||||
value:
|
||||
name: ldappass
|
||||
secret:
|
||||
secretName: ldappass
|
||||
- op: add
|
||||
path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
value:
|
||||
mountPath: /var/run/secrets/ldap/
|
||||
name: ldappass
|
||||
EOF
|
||||
|
||||
cat <<EOF >$ENTERPRISE/configmap.yaml
|
||||
apiVersion: v1
|
||||
kind: ConfigMap
|
||||
metadata:
|
||||
name: conf
|
||||
data:
|
||||
db.conf: |
|
||||
endpoint=127.0.0.1:1234
|
||||
name=app
|
||||
user=admin
|
||||
pass=/var/run/secrets/db/dbpass.txt
|
||||
ldap.conf: |
|
||||
endpoint=ldap://ldap.example.com
|
||||
bindDN=cn=admin,dc=example,dc=com
|
||||
pass=/var/run/secrets/ldap/ldappass.txt
|
||||
EOF
|
||||
```
|
||||
|
||||
Define a **dev** overlay that supports all three features(ExternalDB, LDAP,
|
||||
reCAPTCHA) and conditionally enables some or all of them. In this example, we
|
||||
define a dev overlay that supports all the features, but has disabled the LDAP
|
||||
support, by doing the following::
|
||||
|
||||
- generates `Secrets` for external DB's password and reCAPTCHA's keys
|
||||
- patches the `ConfigMap` of the common base with configurations for external DB
|
||||
and reCAPTCHA
|
||||
- patches the `Deployment` of the common base to mount the generated `Secrets`
|
||||
for external DB and reCAPTCHA
|
||||
|
||||
```shell
|
||||
DEV=$DEMO_HOME/overlays/dev
|
||||
mkdir -p $DEV
|
||||
|
||||
cat <<EOF >$DEV/kustomization.yaml
|
||||
kind: Kustomization
|
||||
|
||||
resources:
|
||||
- ../../base
|
||||
|
||||
secretGenerator:
|
||||
# - name: ldappass <-- Commenting to disable LDAP support
|
||||
# files:
|
||||
# - ldappass.txt
|
||||
- name: dbpass
|
||||
files:
|
||||
- dbpass.txt
|
||||
- name: recaptcha
|
||||
files:
|
||||
- site_key.txt
|
||||
- secret_key.txt
|
||||
|
||||
patches:
|
||||
- configmap.yaml
|
||||
|
||||
patches:
|
||||
- target:
|
||||
group: apps
|
||||
version: v1
|
||||
kind: Deployment
|
||||
name: example
|
||||
path: deployment.yaml
|
||||
EOF
|
||||
|
||||
cat <<EOF >$DEV/deployment.yaml
|
||||
- op: add
|
||||
path: /spec/template/spec/volumes/0
|
||||
value:
|
||||
name: dbpass
|
||||
secret:
|
||||
secretName: dbpass
|
||||
- op: add
|
||||
path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
value:
|
||||
mountPath: /var/run/secrets/db/
|
||||
name: dbpass
|
||||
# - op: add <-- Commenting to disable LDAP support
|
||||
# path: /spec/template/spec/volumes/0
|
||||
# value:
|
||||
# name: ldappass
|
||||
# secret:
|
||||
# secretName: ldappass
|
||||
# - op: add
|
||||
# path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
# value:
|
||||
# mountPath: /var/run/secrets/ldap/
|
||||
# name: ldappass
|
||||
- op: add
|
||||
path: /spec/template/spec/volumes/0
|
||||
value:
|
||||
name: recaptcha
|
||||
secret:
|
||||
secretName: recaptcha
|
||||
- op: add
|
||||
path: /spec/template/spec/containers/0/volumeMounts/0
|
||||
value:
|
||||
mountPath: /var/run/secrets/recaptcha/
|
||||
name: recaptcha
|
||||
|
||||
EOF
|
||||
|
||||
cat <<EOF >$DEV/configmap.yaml
|
||||
apiVersion: v1
|
||||
kind: ConfigMap
|
||||
metadata:
|
||||
name: conf
|
||||
data:
|
||||
db.conf: |
|
||||
endpoint=127.0.0.1:1234
|
||||
name=app
|
||||
user=admin
|
||||
pass=/var/run/secrets/db/dbpass.txt
|
||||
# ldap.conf: | <-- Commenting to disable LDAP support
|
||||
# endpoint=ldap://ldap.example.com
|
||||
# bindDN=cn=admin,dc=example,dc=com
|
||||
# pass=/var/run/secrets/ldap/ldappass.txt
|
||||
recaptcha.conf: |
|
||||
enabled=true
|
||||
site_key=/var/run/secrets/recaptcha/site_key.txt
|
||||
secret_key=/var/run/secrets/recaptcha/secret_key.txt
|
||||
EOF
|
||||
```
|
||||
|
||||
The above commands result in the following structure:
|
||||
|
||||
```shell
|
||||
├── base
|
||||
│ ├── deployment.yaml
|
||||
│ └── kustomization.yaml
|
||||
└── overlays
|
||||
├── community
|
||||
│ ├── configmap.yaml
|
||||
│ ├── dbpass.txt
|
||||
│ ├── deployment.yaml
|
||||
│ ├── kustomization.yaml
|
||||
│ ├── secret_key.txt
|
||||
│ └── site_key.txt
|
||||
├── dev
|
||||
│ ├── configmap.yaml <-- Refers to multiple features and might contain comments
|
||||
│ ├── dbpass.txt
|
||||
│ ├── deployment.yaml <-- Refers to multiple features and might contain comments
|
||||
│ ├── kustomization.yaml <-- Refers to multiple features and might contain comments
|
||||
│ ├── secret_key.txt
|
||||
│ └── site_key.txt
|
||||
└── enterprise
|
||||
├── configmap.yaml
|
||||
├── dbpass.txt
|
||||
├── deployment.yaml
|
||||
├── kustomization.yaml
|
||||
└── ldappass.txt
|
||||
```
|
||||
|
||||
The main issues observed with this solution are:
|
||||
|
||||
1. Since some features are repeated in the `community/`, `enterprise/` and
|
||||
`dev/` overlays, one needs to manually define patches with content that is
|
||||
partially identical to patches of different overlays, that also enable this
|
||||
feature.
|
||||
2. The `dev/` overlay is dynamic, i.e., supports multiple optional features. To
|
||||
enable/disable any single feature one needs to uncomment/comment many lines
|
||||
of YAML which is cumbersome and hard to maintain. Alternatively, one needs
|
||||
to maintain a multitude of overlays and track all possible combinations of
|
||||
features.
|
||||
3. Overlays that combine more than one features define patches for resources
|
||||
whose content is not dedicated to a single feature. That is, there is no
|
||||
semantic isolation per feature, everything gets mixed into a single,
|
||||
multi-feature, resource-specific patch.
|
||||
|
||||
The variants approach may solve this simple example but it won't scale in the
|
||||
long run, as the number of features and deployments grow. What if you have `N`
|
||||
opt-in features and `M` real-world deployment scenarios that ship with `0-N` of
|
||||
these features?
|
||||
|
||||
Ideally, you want to move each feature under a separate, reusable overlay and
|
||||
enable them on-demand per deployment, i.e., in kustomization files of top-level
|
||||
overlays. Enter components.
|
||||
|
||||
## Components example
|
||||
|
||||
Here's a way to solve this issue, by using a Kustomize feature called
|
||||
"components".
|
||||
Here's an alternative and more [DRY] approach that solves this issue by using a
|
||||
Kustomize feature called "components". Each opt-in feature gets packaged as a
|
||||
component, so that it can be referred to from higher-level overlays.
|
||||
|
||||
First, define a place to work:
|
||||
|
||||
@@ -293,7 +659,7 @@ resources:
|
||||
EOF
|
||||
```
|
||||
|
||||
Define a `dev` overlay, that point's to all the components and has LDAP
|
||||
Define a `dev` overlay, that points to all the components and has LDAP
|
||||
disabled:
|
||||
|
||||
```shell
|
||||
@@ -350,4 +716,12 @@ kustomize build overlays/enterprise
|
||||
kustomize build overlays/dev
|
||||
```
|
||||
|
||||
## Takeaway
|
||||
|
||||
At the end of the day, Kustomize components provide a more flexible way to
|
||||
enable/disable features and configurations for applications directly from the
|
||||
kustomization file. This results in more readable, concise and intuitive
|
||||
overlays.
|
||||
|
||||
[variants]: multibases/README.md
|
||||
[DRY principle]: https://en.wikipedia.org/wiki/Don%27t_repeat_yourself
|
||||
|
||||
Reference in New Issue
Block a user