Scenario: you have a Java-based server storefront in production that various internal development teams (signups, checkout, search, etc.) contribute to.
The server runs in different environments: development, testing, staging and production, accepting configuration parameters from java property files.
Using one big properties file for each environment is difficult to manage. The files change frequently, and have to be changed by devops exclusively because
- the files must at least partially agree on certain values that devops cares about and that developers ignore and
- because the production properties contain sensitive data like production database credentials.
With some study, we notice that the properties are separable into categories.
E.g. internationalization data, static data like physical constants, location of external services, etc.
Things that are the same regardless of environment.
Only one set of values is needed.
Place them in a file called
common.properties
(relative location defined below).
E.g. serving location of static content (HTML, CSS, javascript), location of product and customer database tables, ports expected by load balancers, log sinks, etc.
The different values for these properties are precisely what sets the environments apart.
Devops or SRE will want full control over the values used in production. Testing will have fixed databases supporting testing. Developers will want to do whatever they want to try scenarios under development.
Places these values in
development/plumbing.properties
staging/plumbing.properties
production/plumbing.properties
E.g. location of actual user tables, database credentials, decryption keys, etc.
Things that are a subset of devops controls, that nobody else has (or should want) access to.
Places these values in
development/secret.properties
staging/secret.properties
production/secret.properties
and control access to them with (for example) unix file
owner and mode bits, or better yet, put them in
a server dedicated to storing password protected
secrets, and use a field called secretGenerator
in your kustomization to create a kubernetes
secret holding them (not covering that here).
The way to create n cluster environments that share some common information is to create n overlays of a common base.
For the rest of this example, we'll do n==2, just development and production, since adding more environments follows the same pattern.
A cluster environment is created by
running kustomize build
on a target that happens to
be an overlay.
The following example will do that, but will focus on configMap construction, and not worry about how to connect the configMaps to deployments (that is covered in the helloworld example).
All files - including the shared property files discussed above - will be created in a directory tree that is consistent with the base vs overlay file layout defined in the helloworld demo.
It will all live in this work directory:
DEMO_HOME=$(mktemp -d)
Make a place to put the base configuration:
mkdir -p $DEMO_HOME/base
Make the data for the base. This direction by
definition should hold resources common to all
environments. Here we're only defining a java
properties file, and a kustomization
file that
references it.
cat <<EOF >$DEMO_HOME/base/common.properties
color=blue
height=10m
EOF
cat <<EOF >$DEMO_HOME/base/kustomization.yaml
configMapGenerator:
- name: my-configmap
files:
- common.properties
EOF
Make an abbreviation for the parent of the overlay directories:
OVERLAYS=$DEMO_HOME/overlays
Create the files that define the development overlay:
mkdir -p $OVERLAYS/development
cat <<EOF >$OVERLAYS/development/plumbing.properties
port=30000
EOF
cat <<EOF >$OVERLAYS/development/secret.properties
dbpassword=mothersMaidenName
EOF
cat <<EOF >$OVERLAYS/development/kustomization.yaml
bases:
- ../../base
namePrefix: dev-
nameSuffix: -v1
configMapGenerator:
- name: my-configmap
behavior: merge
files:
- plumbing.properties
- secret.properties
EOF
One can now generate the configMaps for development:
kustomize build $OVERLAYS/development
The name of the generated ConfigMap
is visible in this
output.
The name should be something like dev-my-configmap-v1-2gccmccgd5
:
"dev-"
comes from thenamePrefix
field,"my-configmap"
comes from theconfigMapGenerator/name
field,"-v1"
comes from thenameSuffix
field,"-2gccmccgd5"
comes from a deterministic hash thatkustomize
computes from the contents of the configMap.
The hash suffix is critical. If the configMap content
changes, so does the configMap name, along with all
references to that name that appear in the YAML output
from kustomize
.
The name change means deployments will do a rolling restart to get new data if this YAML is applied to the cluster using a command like
kustomize build $OVERLAYS/development | kubectl apply -f -
A deployment has no means to automatically know when or if a configMap in use by the deployment changes.
If one changes a configMap without changing its name and all references to that name, one must imperatively restart the cluster to pick up the change.
The best practice is to treat configMaps as immutable.
Instead of editing configMaps, modify your declarative
specification of the cluster's desired state to
point deployments to new configMaps with new names.
kustomize
makes this easy with its
configMapGenerator
directive and associated naming
controls. A GC process in the k8s master eventually
deletes unused configMaps.
Next, create the files for the production overlay:
mkdir -p $OVERLAYS/production
cat <<EOF >$OVERLAYS/production/plumbing.properties
port=8080
EOF
cat <<EOF >$OVERLAYS/production/secret.properties
dbpassword=thisShouldProbablyBeInASecretInstead
EOF
cat <<EOF >$OVERLAYS/production/kustomization.yaml
bases:
- ../../base
namePrefix: prod-
configMapGenerator:
- name: my-configmap
behavior: merge
files:
- plumbing.properties
- secret.properties
EOF
One can now generate the configMaps for production:
kustomize build $OVERLAYS/production
A CICD process could apply this directly to the cluster using:
kustomize build $OVERLAYS/production | kubectl apply -f -