kubeapply
Contents
Overview
kubeapply
is a lightweight tool for git-based management of Kubernetes configs.
It supports configuration in raw YAML, templated YAML,
Helm charts, and/or skycfg,
and facilitates the complete change workflow including config expansion, validation,
diff generation, and applying.
It can be run from either the command-line or in a webhooks mode that responds
interactively to Github pull request events:
Motivation
Managing Kubernetes configuration in a large organization can be painful. Existing tools like
Helm are useful for certain parts of the workflow (e.g., config generation),
but are often too heavyweight for small, internal services. Once configs are generated, it's hard
to understand what impact they will have in production and then apply them in a consistent way.
We built kubeapply
to make the end-to-end config management process easier and more
consistent. The design choices made were motivated by the following goals:
- Support git-based workflows. i.e., can look at a repo to understand the state of a cluster
- Make it easy to share configuration between different environments (e.g., staging vs. production)
and cluster types
- Wrap existing tooling (
kubectl
, helm
, etc.) whenever possible as opposed to
reimplementing their functionality
- Allow running on either the command-line or in Github
- Support Helm charts, simple templates, and skycfg
See this blog post for more details.
Disclaimer
The tool is designed for our Kubernetes-related workflows at Segment.
While we hope it can work for others, not all features might be directly applicable to other
environments. We welcome feedback and collaboration to make kubeapply
useful to more people!
Getting started
Prerequisites
kubeapply
depends on the following tools:
kubectl
: v1.16 or newer
helm
: v3.5.0 or newer (only needed if using helm charts)
Make sure that they're installed locally and available in your path.
Installing
Install the kubeapply
binary by running:
GO111MODULE="on" go get github.com/segmentio/kubeapply/cmd/kubeapply
You can also build and install the binary by running make install
in the root of this
repo.
Quick tour
See the README in the
examples/kubeapply-test-cluster
directory.
Configuration
Repo layout
Each cluster type to be managed by kubeapply
lives in a directory in a source-controlled
repo. This directory has a set of cluster configs and a profile that is shared by the former.
A cluster config plus the profile is expanded by kubeapply
to create a set of expanded,
kubectl-compatible YAML configs for the cluster. Although these expanded configs can be derived in
a fully reproducible way from the cluster configs and profile, they're typically checked-in to the
repo for easier code review.
The following diagram shows the recommended directory layout:
.
└── clusters
└── [cluster type]
├── expanded
| ├── ...
├── profile
│ ├── [namespace 1]
│ │ ├── [config 1]
│ │ ├── [config 2]
│ ├── [namespace 2]
│ ├── ...
├── [cluster config1]
└── [cluster config2]
Each of the subcomponents is described in more detail in the sections below. See also
examples/kubeapply-test-cluster
for a full example.
Cluster config
Each cluster instance is configured in a single YAML file. Typically, these instances
will vary by the environment (e.g., staging vs. production) and/or the region/account
in which they're deployed, but will share the same profile. At Segment, we name the files
by the environment and region, e.g., stage-us-west-2.yaml
, production-us-west-2.yaml
,
but you can use any naming convention that feels comfortable.
Each cluster config has the following format:
# Basic information about the cluster. The combination of these should uniquely identify
# a single cluster instance running in a single location.
cluster: my-cluster # Name of the cluster
region: us-west-2 # Region in which the cluster is running
env: staging # Environment/account in which the cluster is running
# Where charts can be found by default. Only required if using Helm chart sources.
# See the section below for the supported URL formats.
charts: "file://../../charts"
# Arbitrary parameters that can be used in templates, Helm charts, and skycfg modules.
#
# These are typically used for things that will vary by cluster instance and/or will
# frequently change, e.g. the number of replicas for deployments, container image URIs, etc.
parameters:
service1:
imageTag: abc123
replicas: 2
service2:
imageTag: def678
replicas: 5
...
Profile
The profile
directory contains source files that are used to generate Kubernetes
configs for a specific cluster. By convention, these files are organized into subdirectories
by namespace, and can be further subdivided below that.
The tool currently supports four kinds of input source configs, described in more detail
below.
(1) Raw YAML
Files of the form [name].yaml
will be treated as normal YAML files and copied to the
expanded
directory as-is.
(2) Templated YAML
Files with names ending in .gotpl.yaml
will be templated using the
golang text/template
package with the cluster config
as the input data. You can also use the functions in the
sprig library.
See this file
for an example.
Note that template expansion happens before Helm chart evaluation, so you can template Helm
value files as well.
(3) Helm chart values
Files named [chart name].helm.yaml
will be treated as values files for the associated chart.
The chart will be expanded using helm template ...
and the outputs copied into the expanded
directory. See this file
for an example (which references the envoy chart).
By default, charts are sourced from the URL set in the cluster config charts
parameter.
Currently, the tool supports URLs of the form file://
, http://
, https://
, git://
,
git-https://
, and s3://
.
You can override the source for a specific chart by including a # charts: [url]
comment at the top of the values file. This is helpful for testing out a new version
for just one chart in the profile.
(4) Skycfg/starlark modules
Files ending in .star
will be evaluated using the
skycfg framework to generate one or more Kubernetes protobufs.
The latter are then converted to kubectl-compatible YAML and copied into the expanded
directory.
Skycfg uses the Starlark language along with typed
Kubernetes structs (from Protocol Buffers),
so it can provide more structure and less repetition than YAML-based sources. See
this file for an example.
The skycfg support in kubeapply
should be considered experimental. We're using it for
a few test services in Segment, but it has not been widely exercised internally.
Expanded configs
The expanded
directory contains the results of expanding out the profile
for
a cluster instance. These configs are pure YAML that can be applied directly via kubectl apply
or, preferably, using the kubeapply apply
command (described below).
Usage (CLI)
Expand
kubeapply expand [path to cluster config]
This will expand out all of the configs for the cluster instance, and put them into
a subdirectory of the expanded
directory. Helm charts are expanded via helm template
;
other source types use custom code in the kubeapply
binary.
Validate
kubeapply validate [path to cluster config] --policy=[path to OPA policy in rego format]
This validates all of the expanded configs for the cluster using the
kubeconform
library. It also, optionally, supports
validating configs using one or more OPA policies in
rego format; see the "Experimental features" section below for more details.
Diff
kubeapply diff [path to cluster config] --kubeconfig=[path to kubeconfig]
This wraps kubectl diff
to show a diff between the expanded configs on disk and the
associated resources in the cluster.
Apply
kubeapply apply [path to cluster config] --kubeconfig=[path to kubeconfig]
This wraps kubectl apply
, with some extra logic to apply in a "safe" order
(e.g., configmaps before deployments, etc.).
Usage (Github webhooks)
In addition to interactions through the command-line, kubeapply
also supports an
Atlantis-inspired, Github-based workflow for the diff
and
apply
steps above. This allows the diffs to be more closely reviewed before being
applied, and also ensures that the configuration in the repo stays in-sync with the cluster.
Workflow
The end-to-end user flow is fairly similar to the one used with Atlantis for
Terraform changes:
- Team member changes a cluster config or profile file in the repo, runs
kubeapply expand
locally
- A pull request is opened in Github with the changes
- Kubeapply server gets webhook from Github, posts a friendly "help" message and then
runs an initial diff
- PR owner iterates on change, gets it reviewed
- When ready, PR owner posts a
kubeapply apply
comment
- Kubeapply server gets webhook, checks that change has green status and is approved,
then applies it
- If all changes have been successfully applied, change is automatically merged
Backend
Using the Github webhooks flow requires that you run an HTTP service somewhere that is accessible
to Github. Since the requests are sporadic and can be handled without any local state,
processing them is a nice use case for a serverless framework like
AWS Lambda, and this is how we run it at Segment. Alternatively,
you can run a long-running server that responds to the webhooks.
The sections below contain some implementation details for each option.
Option 1: Run via AWS Lambda
The exact setup steps will vary based on your environment and chosen tooling. At a high-level,
however, the setup process is:
- Build a lambda bundle by running
make lambda-zip
- Upload the bundle zip to a location in S3
- Generate a Github webhook token and a shared secret that will be used for webhook
authentication; store these in SSM
- Using Terraform, the AWS console, or other tooling of your choice, create:
- An-externally facing ALB
- An IAM role for your lambda function that has access to the zip bundle in S3, secrets in SSM,
etc.
- A security group for your lambda function that has access to your cluster control planes
- A lambda function that runs the code in the zip bundle when triggered by ALB requests
The lambda is configured via a set of environment variables that are documented in
the lambda entrypoint. We use SSM for storing secrets like
Github tokens, but it's possible to adapt the code to get these from other places.
Option 2: Run via long-running server
We've provided a basic server entrypoint here. Build a binary
via make kubeapply-server
, configure and deploy this on your infrastructure of choice,
and expose the server to the Internet.
Github configuration
Once you have an externally accessible webhook URL, go to the settings for your repo
and add a new webhook:
In the "Event triggers" section, select "Issue comments" and "Pull requests" only. Then, test it
out by opening up a new pull request that modifies an expanded kubeapply config.
Experimental features
kubestar
This repo now contains an experimental tool, kubestar
, for converting YAML to
skycfg-compatible starlark. See this README for details.
Multi-profile support
The cluster config now supports using multiple profiles. Among other use cases, this is useful if
you want to share profile-style YAML templates across multiple clusters without dealing with Helm.
To use this, add a profiles
section to the cluster config:
cluster: my-cluster
...
profiles:
- name: [name of first profile]
url: [url for first profile]
- name: [name of second profile]
url: [url for second profile]
...
where the url
s are in the same format as those for Helm chart locations,
e.g. file://path/to/my/file
. The outputs of each profile will be expanded into
[expanded dir]/[profile name]/...
.
OPA policy checks
The kubeapply validate
subcommand now supports checking expanded configs against policies in
Open Policy Agent (OPA) format. This can be helpful for
enforcing organization-specific standards, e.g. that images need to be pulled from a particular
private registry, that all labels are in a consistent format, etc.
To use this, write up your policies as .rego
files as described in the OPA documentation and run
the former subcommand with one or more --policy=[path to policy]
arguments. By default, policies
should be in the com.segment.kubeapply
package. Denial reasons, if any, are returned by
setting a deny
variable with a set of denial reason strings. If this set is empty,
kubeapply
will assume that the config has passed all checks in the policy file.
If a denial reason begins with the string warn:
, then that denial will be treated as a
non-blocking warning as opposed to an error that causes validation to fail.
See this unit test for some examples.
Testing
Unit tests
Run make test
in the repo root.
On Github changes
You can simulate Github webhook responses by running kubeapply
with the pull-request
subcommand:
kubeapply pull-request \
--github-token=[personal token] \
--repo=[repo in format owner/name] \
--pull-request=[pull request num] \
--comment-body="kubeapply help"
This will respond locally using the codepath that would be executed
in response to a Github webhook for the associated repo and pull request.