There are two ways to store configuration settings in ConfigMaps - either as key-value pairs, which you’ll surface as environment variables, or as text data which you’ll surface as files in the container filesystem.
YAML overview</summary>
Key-value pairs are defined in YAML like this:
apiVersion: v1
kind: ConfigMap
metadata:
  name: configurable-env
data:
  Configurable__Environment: uat
The metadata is standard - you’ll reference the name of the ConfigMap in the Pod spec to load settings.
data - list of settings as key-value pairs, separated with colonsIn the Pod spec you add a reference:
spec:
  containers:
    - name: app
      image: sixeyed/configurable:21.04
      envFrom:
        - configMapRef:
            name: configurable-env
envFrom - load all the values in the source as environment variablesText files are defined in the same YAML structure, with an entry for each file:
apiVersion: v1
kind: ConfigMap
metadata:
  name: configurable-override
data:
  override.json: |-
    {
      "Configurable": {
        "Release": "21.04.01"
      }
    }
Careful with the whitespace - the file data needs to be indented one stop more than the filename
The API spec is the same, but in this format:
data - list of files, with the filename set and the contents following the separator |-In the Pod spec you can load all the values into the container filesystem as volume mounts:
spec:
  containers:
    - name: app
      image: sixeyed/configurable:21.04
      volumeMounts:
        - name: config-override
          mountPath: "/app/config"
          readOnly: true
  volumes:
    - name: config-override
      configMap:
        name: configurable-override
Volumes are defined at the Pod level - they are storage units which are part of the Pod environment. You load the storage unit into the container filesystem using a mount.
volumes - list of storage units to load, can be ConfigMaps, Secrets or other typesvolumeMounts - list of volumes to mount into the container filesystemvolumeMounts.name - matches the name of the volumevolumeMounts.mountPath - the directory path where the volume is surfacedvolumeMounts.readOnly - flag whether the volume is read-only or editable</details>
The demo app for this lab has the logic to merge config from multiple sources.
Defaults are built into the appsettings.json file inside the Docker image - run a Pod with no config applied to see the defaults:
kubectl run configurable --image=sixeyed/configurable:21.04 --labels='kubernetes.courselabs.co=configmaps'
kubectl wait --for=condition=Ready pod configurable
kubectl port-forward pod/configurable 8080:80
These are useful commands for quick testing or debugging, but in real life it’s all YAML
Check the app at http://localhost:8080 (or your node’s IP address if you have a remote cluster).
You see the default configuration settings from the JSON file in the container image. The environment variables come from Dockerfile, plus the container OS and some set by Kubernetes.
📋 Exit the port-forward and remove the Pod.
Not sure how?</summary>
# Ctrl-C to exit the command
kubectl delete pod configurable
</details>
The Pod spec is where you apply configuration:
📋 Deploy the app from the folder labs/kubernetes/configmaps/specs/configurable/
Not sure how?</summary>
kubectl apply -f labs/kubernetes/configmaps/specs/configurable/
</details>
You can check the environment variable is set by running printenv inside the Pod container:
kubectl exec deploy/configurable -- printenv | grep __
You should see
Configurable__Release=24.01.1
📋 Confirm that by browsing to the app from your Service.
Not sure how?</summary>
# print the Service details:
kubectl get svc -l app=configurable
</details>
Environment variables in Pod specs are fine for single settings like feature flags. Typically you’ll have lots of settings and you’ll use a ConfigMap:
kubectl apply -f labs/kubernetes/configmaps/specs/configurable/config-env/
This creates a new ConfigMap and updates the Deployment. Remember which object the Deployment uses to manage Pods?
📋 Print the environment variables set in the updated Pod.
Not sure how?</summary>
kubectl exec deploy/configurable -- printenv | grep __
</details>
You should see the release is now
24.01.2and there’s a new settingConfigurable__Environment=uat
Environment variables are limited too. They’re visible to all processes so there’s a potential to leak sensitive information. There can also be collisions if the same keys are used by different processes.
The filesystem is a more reliable store for configuration; permissions can be set for files, and it allows for more complex config with nested settings.
The demo app can use JSON configuration as well as environment variables, and it supports loading additional settings from an override file:
kubectl apply -f labs/kubernetes/configmaps/specs/configurable/config-json/
Refresh the web app and you’ll see new settings coming from the
config/override.jsonfile
📋 Check the filesystem inside the container to see the file loaded from the ConfigMap into the /app/config path.
Not sure how?</summary>
Explore the container filesystem with exec commands:
kubectl exec deploy/configurable -- ls /app/
kubectl exec deploy/configurable -- ls /app/config/
kubectl exec deploy/configurable -- cat /app/config/override.json
The first JSON file is from the container image, the second is from the ConfigMap volume mount.
</details>
Something’s not quite right though - the release setting is still coming from the environment variable:
kubectl exec deploy/configurable -- cat /app/config/override.json
kubectl exec deploy/configurable -- printenv | grep __
The config hierarchy in this app puts environment variables ahead of settings in files, so they get overidden. You’ll need to understand the hierarchy for your apps to model config correctly.
Mapping configuration in ConfigMap YAML works well and it means you can deploy your whole app with kubectl apply. But it won’t suit every organization, and Kubernetes also supports creating ConfigMaps directly from values and config files - without any YAML.
Create two new ConfigMaps to support the Deployment in deployment-lab.yaml and set these values:
Configuration__Release=21.04-labFeatures.DarkMode=trueCleanup by removing objects with this lab’s label:
kubectl delete configmap,deploy,svc,pod -l kubernetes.courselabs.co=configmaps