Version information
This version is compatible with:
- Puppet Enterprise 2023.8.x, 2023.7.x, 2023.6.x, 2023.5.x, 2023.4.x, 2023.3.x, 2023.2.x, 2023.1.x, 2021.7.x
- Puppet >= 7.24 < 9.0.0
- , , , , , , ,
Plans:
- deploy
Start using this module
Add this module to your Puppetfile:
mod 'jtl-kubecm', '0.1.1'
Learn more about managing modules with a PuppetfileDocumentation
KubeCM: Kubernetes Configuration Management
- Overview
- Features
- Motivation
- Getting Started
- Usage
- Custom Resources, Values, and Patches
- YAML Plans
- Troubleshooting
- Contributing
Overview
KubeCM (Kubernetes Configuration Management) empowers you to control and deploy Kubernetes resources your way, combining the strengths of Puppet, Helm, and Kustomize. With KubeCM, you can manage, customize, and configure Kubernetes deployments using a hierarchical data model and YAML plans that provide clarity and reusability.
Features
- Unified Management: Deploy Helm charts, custom resources, and Kustomize patches under one plan.
- Hierarchical Configuration: Use Hiera to store, merge, and override configurations at multiple levels.
- Secure Secrets: Encrypt sensitive data using Hiera-EYAML.
- Custom Resource Deployment: Deploy any Kubernetes resource directly without requiring a Helm chart.
- Integrated Orchestration: Bolt-powered orchestration for on-demand or scheduled deployments.
- Cloud Transition: Leverage existing Puppet data and automation to ease migrations to Kubernetes.
Motivation
Kubernetes excels at managing clusters, but it struggles with managing entire deployments as cohesive units. Helm addresses this gap with templated "charts," but real-world applications often require configuration changes that Helm templates do not expose. KubeCM steps in to bridge this gap by leveraging Puppet's powerful configuration management features (like Hiera and Bolt) to manage and customize Helm, Kustomize, and raw Kubernetes resources as a unified whole.
Getting Started
Prerequisites
Ensure the following tools are installed and configured on a Linux system:
- kubectl: Kubernetes CLI for managing clusters.
- helm: Kubernetes package manager.
- bolt: Puppet orchestration tool.
- hiera-eyaml: Optional, for encrypted secrets.
Installation
-
Create a Bolt project:
mkdir acme_inc cd acme_inc bolt project init
-
Add KubeCM to
bolt-project.yaml
:--- name: acme_inc modules: - jtl-kubecm
-
Install the module:
bolt module install
Usage
Deploying Applications
The following parameters in the kubecm::deploy
plan specify key aspects of a
deployment:
- release: The unique name of the deployment within the Kubernetes namespace. Each deployment should have a distinct release name.
- chart: The name of the Helm chart to be used for the deployment.
- chart_source: The location of the Helm chart from a repo, a local directory, or an OCI registry reference. This tells Helm where to retrieve the chart. It can also be undefined to deploy without a chart, using only resources from Hiera.
- repo_url: If using a repo reference in
chart_source
(e.g.repo_name/chart_name
), the HTTP location of the Helm repository.
Use the kubecm::deploy
plan to install applications via Helm charts.
Example:
bolt plan run kubecm::deploy release=shop chart=wordpress chart_source=oci://registry-1.docker.io/bitnamicharts/wordpress
bolt plan run kubecm::deploy release=blog chart=wordpress chart_source=oci://registry-1.docker.io/bitnamicharts/wordpress
bolt plan run kubecm::deploy release=vaultwarden chart_source=guerzon/vaultwarden repo_url=https://guerzon.github.io/vaultwarden
This deploys two WordPress instances ("shop" and "blog") and a Vaultwarden instance.
Configuring Deployments
Configuration is managed using Hiera's hierarchical structure. Create a
hiera.yaml
in the project directory:
---
version: 5
defaults:
datadir: data
data_hash: yaml_data
hierarchy:
- name: 'Releases'
paths:
- "release/%{kubecm::deploy::release}/%{kubecm::deploy::namespace}.yaml"
- "release/%{kubecm::deploy::release}.yaml"
- name: 'Charts'
paths:
- "chart/%{kubecm::deploy::chart}/%{kubecm::deploy::namespace}.yaml"
- "chart/%{kubecm::deploy::chart}.yaml"
- name: 'Common'
paths:
- 'common.yaml'
To provide additional customization, you can override KubeCM's default key names for various deployment settings. These keys include:
- include_key: Used to define additional Puppet classes to be included.
- resources_key: Used to define Kubernetes resources to be created as part of the deployment.
- values_key: Used to provide Helm chart values for the deployment.
- patches_key: Used to define Kustomize patches to apply to resources.
These key names can be set in data/common.yaml
to override the defaults as
shown below:
---
kubecm::deploy::include_key: 'include'
kubecm::deploy::resources_key: 'resources'
kubecm::deploy::values_key: 'values'
kubecm::deploy::patches_key: 'patches'
This configuration allows for a more customized, DSL-like experience when managing Kubernetes deployments with KubeCM. You can modify these key names to avoid conflicts with other existing keys in your control repository.
Custom Resources, Values, and Patches
Values Examples:
Memory Resource Parameterization:
This example demonstrates how to set default memory requests for Helm deployments and override them at a namespace-specific level.
data/chart/wordpress.yaml
---
memory: 200Mi
values:
resources:
requests:
memory: "%{lookup('memory')}"
data/chart/wordpress/test.yaml
---
memory: 100Mi
Ingress Hostname Parameterization:
This example shows how to set a default ingress hostname pattern for all releases and override it for specific releases.
data/chart/wordpress.yaml
---
ingress_hostname: "%{kubecm::deploy::release}.example.com"
values:
ingress:
enabled: true
hostname: "%{lookup('ingress_hostname')}"
data/release/shop.yaml
---
ingress_hostname: 'store.example.com'
Vaultwarden Admin Token Hash Calculation:
This example demonstrates how to hash the Vaultwarden admin token dynamically and make it available as a Helm chart value.
data/chart/vaultwarden.yaml
---
include:
- 'acme_inc::vaultwarden'
admin_token: supersecuretoken # encrypt this with hiera-eyaml!
values:
adminToken:
value: "%{acme_inc::vaultwarden::admin_token_hash}"
manifests/vaultwarden.pp
class acme_inc::vaultwarden {
$admin_token = lookup('admin_token')
$admin_token_hash = generate(
'/bin/sh',
'-c',
"echo -n ${admin_token.shellquote} | argon2 `openssl rand -base64 32` -e -id -k 65540 -t 3 -p 4",
).chomp
}
Registry Auths Example:
This example demonstrates how to manage Docker registry authentication secrets to be used by Kubernetes deployments.
data/common.yaml
---
include:
- 'acme_inc'
registry_tokens:
registry.example.com: fake-token # encrypt this with hiera-eyaml!
resources:
registry_auths:
- apiVersion: v1
kind: Secret
metadata:
name: "%{kubecm::deploy::service}-registry-auths"
namespace: "%{kubecm::deploy::namespace}"
data:
.dockerconfigjson: "%{acme_inc::registry_auths_base64}"
type: kubernetes.io/dockerconfigjson
manifests/init.pp
class acme_inc {
$registry_tokens = lookup('registry_tokens')
$registry_auths_base64 = base64('encode', stdlib::to_json({
'auths' => $registry_tokens.reduce({}) |$result, $token| {
$result + { $token[0] => { 'auth' => base64('encode', $token[1]).chomp } }
},
}))
}
Patch Example
This example demonstrates how to use a Kustomize patch to modify Kubernetes resources in place. Patches like this are useful when you need to modify resource properties that aren't exposed as chart values, such as node affinity, tolerations, or annotations. Patching can be especially helpful when dealing with third-party charts where direct configuration is limited:
data/chart/vaultwarden.yaml
---
node_role: special
patches:
10-placement:
patch:
kind: not-important
metadata:
name: not-important
spec:
template:
spec:
affinity:
nodeAffinity:
requiredDuringSchedulingIgnoredDuringExecution:
nodeSelectorTerms:
- matchExpressions:
- key: "node-role.kubernetes.io/%{lookup('node_role')}"
operator: Exists
tolerations:
- key: "node-role.kubernetes.io/%{lookup('node_role')}"
operator: Exists
effect: NoSchedule
targets:
kind: (Deployment|StatefulSet|Job)
YAML Plans
YAML plans provide a simple way to define deployment workflows using YAML syntax. They enable you to automate the execution of multiple tasks, making it easier to manage and update Kubernetes resources in a consistent manner.
YAML plans are especially useful for repeatable deployments, where you want to ensure that all necessary steps are executed in the correct order. Below are two example YAML plans for deploying Vaultwarden and WordPress applications.
Example 1: Deploying Vaultwarden
This YAML plan defines a simple process to deploy the Vaultwarden application
using KubeCM's kubecm::deploy
plan.
plans/deploy_vaultwarden.yaml
---
description: 'Deploy Acme Inc Vaultwarden'
steps:
- description: 'Deploy Vaultwarden'
plan: kubecm::deploy
parameters:
release: 'vaultwarden'
chart_source: 'guerzon/vaultwarden'
repo_url: 'https://guerzon.github.io/vaultwarden'
This plan deploys the Vaultwarden release using the specified chart and repository URL.
Example 2: Deploying WordPress
This YAML plan defines a more flexible deployment workflow for WordPress. It allows the user to specify which instance (either 'shop' or 'blog') should be deployed.
plans/deploy_wordpress.yaml
---
description: 'Deploy Acme Inc WordPress'
parameters:
site:
description: The release to deploy
type: Enum['shop', 'blog']
steps:
- description: 'Deploy WordPress'
plan: kubecm::deploy
parameters:
release: $site
chart: wordpress
chart_source: oci://registry-1.docker.io/bitnamicharts/wordpress
This plan introduces the parameters
block, which requires the user to specify
the site (either shop
or blog
) to be deployed. This is useful when you want
to reuse the same plan logic for multiple deployments.
To run these plans, you would use the following Bolt commands:
bolt plan run acme_inc::deploy_vaultwarden
bolt plan run acme_inc::deploy_wordpress site=shop
bolt plan run acme_inc::deploy_wordpress site=blog
These YAML plans streamline deployments, enforce consistency, and ensure repeatable, parameterized workflows that are easy to maintain and execute.
Troubleshooting
If you encounter issues with your deployments, KubeCM provides several tools to help diagnose and debug problems.
Using the render_to
Parameter
The render_to
parameter allows you to render the final output of the Helm or
Kustomize configuration to a local file. This is useful for verifying the
rendered Kubernetes manifests before deploying them.
Example usage:
bolt plan run kubecm::deploy release=shop chart=wordpress chart_source=oci://registry-1.docker.io/bitnamicharts/wordpress render_to=output.yaml
After the command runs, the rendered configuration will be saved in
output.yaml
for review.
Exploring the Build Directory
KubeCM generates a build directory during deployment. This directory contains intermediate files and configurations produced during the rendering and patching process. The build directory provides insight into the final state of configurations before they are applied to the Kubernetes cluster.
By default, the build directory is named build
and is created in the current
working directory.
To explore the build directory, you can navigate into it and review the files:
cd build
ls -al
Inside, you will find rendered manifests and debug information, which can be invaluable when troubleshooting failed deployments or understanding how your configurations were combined.
Contributing
Contributions are welcome! Follow these steps to contribute:
-
Use your fork for development:
Modify the
bolt-project.yaml
to use your forked repository:modules: - name: kubecm git: <repo-url> ref: <branch-name>
Then run:
bolt module install
-
Project is managed with PDK:
Before contributing changes back to the project, ensure that your changes pass PDK validation and tests:
pdk validate pdk test unit
-
Submit a pull request.
Happy Deploying!
Reference
Table of Contents
Classes
Private Classes
kubecm::deploy
: Puppet entrypoint for the kubecm::deploy plan
Functions
Private Functions
kubecm::print_report
: Print apply results like a Puppet report
Plans
kubecm::deploy
: Install or upgrade a release
Plans
kubecm::deploy
Install or upgrade a release
Parameters
The following parameters are available in the kubecm::deploy
plan:
release
chart
chart_source
deploy
hooks
namespace
remove_resources
remove_patches
render_to
repo_url
version
wait
sleep
subcharts
parent
build_dir
release
Data type: String
Installation name (what is this deployment called)
chart
Data type: String
Chart name, or your name for this deployment and set chart_source
Default value: $release
chart_source
Data type: Optional[String]
Typically repo_name/chart_name
or an oci://
URI, but could be a local path,
a valid Puppet file source, or undef
for no chart (just Hiera resources).
Default value: undef
deploy
Data type: Boolean
Run or skip the deployment
Default value: true
hooks
Data type: Boolean
Enable or disable install hooks
Default value: true
namespace
Data type: Optional[String]
Kubernetes namespace to manage
Default value: undef
remove_resources
Data type: Array[String]
A list of keys to remove from the resources hash (i.e. don't deploy these!)
Default value: []
remove_patches
Data type: Array[String]
A list of keys to remove from the patches hash (i.e. don't apply these!)
Default value: []
render_to
Data type: Optional[String]
Just save the fully-rendered chart to this yaml file
Default value: undef
repo_url
Data type: Optional[String]
Optional URL of the Helm repo to add
Default value: undef
version
Data type: Optional[String]
Optional Helm chart version
Default value: undef
wait
Data type: Boolean
Wait for resources to become available
Default value: false
sleep
Data type: Optional[Numeric]
Wait additional time, in seconds (for example, for a VIP to propagate)
Default value: undef
subcharts
Data type: Array[Hash]
Additional charts to deploy as part of this one
Default value: []
parent
Data type: Optional[String]
Private. The parent release this one is being rendered for.
Default value: undef
build_dir
Data type: String
Scratch directory for compiling manifests
Default value: lookup('kubecm::deploy::build_dir')
What are plans?
Modules can contain plans that take action outside of a desired state managed by Puppet. It’s perfect for troubleshooting or deploying one-off changes, distributing scripts to run across your infrastructure, or automating changes that need to happen in a particular order as part of an application deployment.
Changelog
All notable changes to this project will be documented in this file.
Release 0.1.1
Remove file accidentally included in previous release.
Release 0.1.0
Initial release including fully functional deploy plan originally from Nest. Parameter names have been changed to be a little more generic, and the code has been significantly refactored for public consumption. Added tests and documentation.
Features
- Deploy charts with custom resources, values, and patches.
- Integrate into your own project with parameters to customize lookups.
- Automatically generate Helm chart to manage bare resources.
- Manage multiple charts as one release with subcharts.
Bugfixes
- Build intermediate files in separate directories to avoid conflicts.
Known Issues
- This project does not currently run on Windows.
Dependencies
- puppetlabs-stdlib (>= 9.0.0 < 10.0.0)
Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "{}" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright {yyyy} {name of copyright owner} Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.