Customize the Kubernetes operator and resources it manages
Cluster resource customization using ProxyClass
Tailscale operator v1.60 and later provides the ability to customize the configuration of cluster resources created by the operator using ProxyClass
Custom Resource Definition.
You can specify cluster resource configuration for custom labels and resource requests using a ProxyClass
Custom Resource.
You can then:
-
Apply configuration from a particular
ProxyClass
to cluster resources created for a tailscaleIngress
orService
using atailscale.com/proxy-class=<proxy-class-name>
label on theIngress
orService
. -
Apply configuration from a particular
ProxyClass
to cluster resources created for aConnector
usingconnector.spec.proxyClass
field.
The following example shows how to use a ProxyClass
that specifies custom labels and node selectors. These are applied to Pod
s for a tailscale Ingress
, a cluster egress proxy, a Connector
, and a ProxyGroup
:
-
Create a
ProxyClass
resource:apiVersion: tailscale.com/v1alpha1 kind: ProxyClass metadata: name: prod spec: statefulSet: pod: labels: team: eng environment: prod nodeSelector: beta.kubernetes.io/os: "linux"
-
Create a tailscale
Ingress
withtailscale.com/proxy-class=prod
label:apiVersion: networking.k8s.io/v1 kind: Ingress metadata: name: my-app labels: tailscale.com/proxy-class: "prod" spec: rules: ... ingressClassName: tailscale
-
Create a cluster egress
Service
with atailscale.com/proxy-class=prod
label:apiVersion: v1 kind: Service metadata: annotations: tailscale.com/tailnet-ip: <tailnet-ip> labels: tailscale.com/proxy-class: "prod" name: my-tailnet-service spec:
-
Create a
Connector
that refers to the 'prod'ProxyClass
:apiVersion: tailscale.com/v1alpha1 kind: Connector metadata: name: prod spec: proxyClass: prod ...
-
Create a
ProxyGroup
that refers to the 'prod'ProxyClass
:apiVersion: tailscale.com/v1alpha1 kind: ProxyGroup metadata: name: egress spec: proxyClass: prod ...
You can find all available ProxyClass
configuration options on GitHub →
Default ProxyClass
Tailscale v1.74 and later allows specifying a default ProxyClass
. Configuration from a default ProxyClass
is applied
to cluster ingress, cluster egress and
ProxyGroup
proxies that don't have a ProxyClass
explicitly set.
You can set a default ProxyClass
for the cluster via the proxyConfig.defaultProxyClass
Helm value if installing
using Helm or via the PROXY_DEFAULT_CLASS
environment variable if installing using static
manifests.
Proxy configuration
You can specify a ProxyClass
for Connector
resources, and egress and ingress proxies.
The API server proxy currently runs as part of the same process as the Kubernetes operator. You can use the available operator configuration options to configure the API server proxy parameters.
Customizing tags
All the proxies that the operator creates are Tailscale devices tagged by one or more tags.
The Tailscale operator must be a tag owner of all the proxy tags: if you want to tag a proxy device with tag:prod,tag:emea
, the tagOwners
section of the tailnet policy file must list tag:k8s-operator
as one of the owners of both tag:prod
and tag:emea
.
Currently, tags can not be modified after a proxy has been created.
Default tags
By default, a proxy device joins your tailnet tagged with the tag tag:k8s
. You can modify the default tag or tags when installing the operator.
If you install the operator with Helm, you can use .proxyConfig.defaultTags
in the Helm values file.
If you install the operator with static manifests, you can set the PROXY_TAGS
environment variable in the deployment manifest.
Multiple tags must be passed as a comma separated string, that is, tag:prod,tag:emea
.
Tags for individual proxies
To override the default tags for an individual proxy created for a Tailscale Service
or Ingress
, you can set the tailscale.com/tags
annotation on the Service
or Ingress
resource to a comma separated list of the desired tags.
For example, setting tailscale.com/tags: "tag:prod,tag:emea"
annotation will result in the proxy device having the tags tag:prod
and tag:emea
.
To override the default tags for the proxy created for a Connector
custom resource, you can set tags via the spec.tags
field.
See also Common patterns for tag names for best practices around tag names.
Exposing metrics
This functionality is available in Tailscale 1.78 and later.
When specified for a resource, the following ProxyClass
definition will enable client metrics at the path /metrics
on a container port named "metrics":
apiVersion: tailscale.com/v1alpha1
kind: ProxyClass
metadata:
name: prod
spec:
metrics:
enable: true
Additionally the operator will also create a metrics Service
for the proxy in the operator's namespace that will also expose metrics on the path /metrics
on a port named "metrics".
Prometheus ServiceMonitor
The Kubernetes Operator can also create a Prometheus ServiceMonitor
for proxy resources.
To enable it:
-
Ensure that Prometheus operator including its Custom Resource Definitions is installed
-
Apply
ProxyClass
that enables metrics andServiceMonitor
creation:apiVersion: tailscale.com/v1alpha1 kind: ProxyClass metadata: name: prod spec: metrics: enable: true serviceMonitor: enable: true
The ingested metrics will have labels that identify the proxy to which the ProxyClass
was applied:
-
ts_proxy_type
: type of the proxy. Values can beingress_service
,ingress_resource
,connector
orproxygroup
. -
ts_proxy_parent_name
: name of the proxy's parent resource. That is, name of theIngress
resource, TailscaleService
,Connector
orProxyGroup
. -
ts_proxy_parent_namespace
: namespace of the proxy's parent resource. This only applies to namespaced resources.
Currently it is not possible to customize the created ServiceMonitor
. We welcome your feedback as to what customization options would be useful for you.
Debug endpoints
Debug endpoints are unstable, may change without notice, and are not recommended for production use.
If you rely on the debug metrics (at /debug/metrics
), you must explicitly enable the following debug option before upgrading to 1.82 which will always default debug to disabled.
If enabled, the debug endpoints will be available on a container port named "debug". The endpoints include /debug/metrics
and /debug/pprof/
paths from Go's net/http/pprof
library.
To maintain backward compatibility, debug endpoints default to enabled if .spec.metrics.enable
is set to true
. If .spec.metrics.enable
is set to false
, the debug endpoints default to disabled.
In Tailscale v1.82 and later, the debug endpoints always default to disabled. You can override the default for debug endpoints using ProxyClass
:
apiVersion: tailscale.com/v1alpha1
kind: ProxyClass
metadata:
name: prod
spec:
metrics:
enable: true
statefulSet:
pod:
tailscaleContainer:
debug:
enable: false
Using custom machine names
Cluster ingress and egress proxies support overriding the hostname they announce while registering with Tailscale. For Service
s, you can set a custom hostname using a tailscale.com/hostname
annotation. For Ingress
es, you can set a custom hostname using the .spec.tls.hosts
field (only the first value will be used).
Note that this only sets a custom operating system (OS) hostname reported by the device. The actual machine name will differ if a device is on the network with the same name.
Machine names are subject to the constraints of DNS: they can be up to 63 characters long, must start and end with a letter, and consist of only letters, numbers, and -
.
Customize permissions
Pods for proxies created for cluster ingress using Service
, cluster egress, Connector
and ProxyGroup
contain a main tailscale
container and an init
container.
Since Tailscale v1.78, both containers run as privileged containers.
The main tailscale
container requires privileged permissions to create a /dev/net/tun
device.
As an alternative, you can restrict the main container's permissions and delegate the device creation to a Kubernetes device plugin:
-
Install the generic device plugin to your cluster. Pass a
--device
flag that configures the plugin to create/dev/net/tun
devices:--device='{"name": "tun", "groups": [{"paths": [{"path": "/dev/net/tun"}]}, "count": 1000]}'
-
Apply a
ProxyClass
that restricts thetailscale
container's permissions toNET_ADMIN
andNET_RAW
capabilities, and tells the device plugin to create a/dev/net/tun
device:apiVersion: tailscale.com/v1alpha1 kind: ProxyClass metadata: name: tailscale-tun spec: statefulSet: pod: tailscaleContainer: resources: limits: squat.ai/tun: "1" securityContext: allowPrivilegeEscalation: false capabilities: drop: - ALL add: - NET_ADMIN - NET_RAW
-
Ensure that the
ProxyClass
is applied to all proxies created for Tailscale ingressService
s, Tailscale egressService
s,Connector
s andProxyGroup
s.
Refer to the detailed instructions and discussion on GitHub →
Limitations
- You cannot enable metrics for egress proxies that do not use a
ProxyGroup
.