Kubernetes Overview
Pants is currently building support for Kubernetes. Simple use cases might be supported, but many options are missing.
Please share feedback for what you need to use Pants with your Kubernetes queries by either opening a GitHub issue or joining our Slack!
Initial setupâ
First, activate the relevant backend in pants.toml
:
[GLOBAL]
backend_packages = [
...
"pants.backend.experimental.k8s",
]
The Kubernetes backend adds k8s_source
and
k8s_sources
target types for Kubernetes object
files.
For example, create a file src/k8s/webpages.yaml
:
---
apiVersion: v1
kind: ConfigMap
metadata:
name: webpages
data:
index.html: |
<html>
<head>Hello pants!</head>
<body>Hello pants!</body>
</html>
Now add a k8s_sources
target in src/k8s/BUILD
:
k8s_sources()
Deploying objects to a clusterâ
We'll be using a local kind cluster throughout the tutorial. First, spin up a cluster:
kind create cluster
Creating cluster "kind" ...
â Ensuring node image (kindest/node:v1.25.3) đŧ
â Preparing nodes đĻ
â Writing configuration đ
â Starting control-plane đšī¸
â Installing CNI đ
â Installing StorageClass đž
Set kubectl context to "kind-kind"
Second, configure the list of available contexts in pants.toml
:
...
[k8s]
available_contexts = [
"kind-kind",
]
Third, create a deployable target k8s_bundle
in src/k8s/BUILD
:
k8s_sources()
k8s_bundle(
name="webpages",
sources=("src/k8s/webpages.yaml",),
context="kind-kind",
)
Now you can deploy the target:
pants experimental-deploy src/k8s:webpages
â src/k8s:webpages deployed to context kind-kind
To prevent accidentally deploying kubernetes manifests to the wrong cluster,
the context field is required on k8s_bundle
for deployment. For deploying the
same k8s_bundle
to multiple contexts, consider using parametrize
builtin
like k8s_bundle(..., context=parametrize("stage", "prod"))
. For CI agents
which will only have access to a single context, set the
[kubectl].pass_context
to false in pants.toml
to have them use their
default context.
Simple templatesâ
At some point, you may need to inject variables from a BUILD file into
Kubernetes resources â also known as templating. The simplest way to achieve
this is by using the python_format_string
target, which generates
k8s_sources
by substituting the values you specify in the BUILD file.
First, add the codegen backend:
backend_packages = [
...
"pants.backend.experimental.k8s",
"pants.backend.experimental.codegen.python_format_string",
"pants.backend.experimental.codegen.python_format_string.k8s",
]
Then parametrize the yaml using python format string syntax, e.g. the namespace:
---
apiVersion: v1
kind: ConfigMap
metadata:
name: webpages
namespace: {namespace}
data:
index.html: |
<html>
<head>Hello pants!</head>
<body>Hello pants!</body>
</html>
Now replace k8s_sources
with a python_format_string
target and pass the
namespace value:
python_format_string(
name="webpages-template",
source="webpages.yaml",
values={"namespace": "web"},
)
k8s_bundle(
name="webpages",
sources=("src/k8s:webpages-template",),
context="kind-kind",
)
Now you can deploy the bundle:
pants experimental-deploy src/k8s/:webpages
This setup can now be used to deploy the same resource to multiple namespaces
with parametrize
builtin:
python_format_string(
name="webpages-template",
source="webpages.yaml",
**parametrize("default", values={"namespace": "default"}),
**parametrize("web", values={"namespace": "web"}),
)
k8s_bundle(
name="webpages",
context="kind-kind",
**parametrize("default", sources=("src/k8s:webpages-template@parametrize=default",)),
**parametrize("web", sources=("src/k8s:webpages-template@parametrize=web",)),
)
Docker imagesâ
Before we continue, add the docker backend:
[GLOBAL]
backend_packages = [
...
"pants.backend.docker",
]
[dockerfile-parser]
use_rust_parser = true
To use docker images you most likely will need some templating. This is because your docker image tags will be probably versioned.
You might want to use git to generate the version for your images, but you
can't directly run git commands in a BUILD file. You can use a
.pants.bootstrap
file as a workaround:
#!/bin/sh
VERSION="${VERSION:-$(git describe --tags --dirty --match "[0-9\.]*" || echo 0.0.1)}"
export VERSION
This script will run before every pants command, so you can now use the
VERSION
env var in the BUILD file:
...
docker_image(
name="custom-nginx",
instructions=["FROM nginx"],
image_tags=[env("VERSION")],
)
python_format_string(
name="webserver-template",
source="deployment.yaml",
values={"VERSION": env("VERSION")},
)
k8s_bundle(
name="webserver",
context="kind-kind",
sources=("src/k8s:webserver-template",),
dependencies=(":custom-nginx",),
)
Create the deployment:
---
apiVersion: apps/v1
kind: Deployment
metadata:
name: webserver
namespace: web
spec:
replicas: 1
selector:
matchLabels:
app: webserver
template:
metadata:
labels:
app: webserver
spec:
containers:
- name: nginx
image: custom-nginx:{VERSION}
Now deploy the bundle:
pants experimental-deploy src/k8s:webserver
Notice, that pants will automatically publish the image. This happens because
we've configured dependencies=(":custom-nginx",)
field on k8s_bundle
target. You can disable this behaviour and publish the image manually:
pants publish src/k8s:custom-nginx
pants experimental-deploy --no-publish-dependencies src/k8s:webserver