Careful!
You are browsing documentation for a version of Kuma that is not the latest release.
Data plane on Kubernetes
On Kubernetes the Dataplane
entity is automatically created for you, and because transparent proxying is used to communicate between the service and the sidecar proxy, no code changes are required in your applications.
The Kuma control plane injects a kuma-sidecar
container into your Pod
’s container. If
you’re not using the CNI, it also injects a kuma-init
into initContainers
to
setup transparent proxying.
You can control whether Kuma automatically injects the data plane proxy by labeling either the Namespace or the Pod with
kuma.io/sidecar-injection=enabled
, e.g.
To opt out of data-plane injection into a particular Pod
, you need to label it
with kuma.io/sidecar-injection=disabled
, e.g.
Once your pod is running you can see the data plane CRD that matches it using kubectl
:
Tag generation
When Dataplane
entities are automatically created, all labels from Pod are converted into Dataplane
tags.
Labels with keys that contains kuma.io/
are not converted because they are reserved to Kuma.
The following tags are added automatically and cannot be overridden using Pod labels.
kuma.io/service
: Identifies the service name based on a Service that selects a Pod. This will be of format<name>_<namespace>_svc_<port>
where<name>
,<namespace>
and<port>
are from the Kubernetes service that is associated with this particular pod. When a pod is spawned without being associated with any Kubernetes Service resource the data plane tag will bekuma.io/service: <name>_<namespace>_svc
, where<name>
and<namespace>
are extracted from the Pod resource metadata.kuma.io/zone
: Identifies the zone name in a multi-zone deployment.kuma.io/protocol
: Identifies the protocol that was defined by theappProtocol
field on the Service that selects the Pod.k8s.kuma.io/namespace
: Identifies the Pod’s namespace. Example:kuma-demo
.k8s.kuma.io/service-name
: Identifies the name of Kubernetes Service that selects the Pod. Example:demo-app
.k8s.kuma.io/service-port
: Identifies the port of Kubernetes Service that selects the Pod. Example:80
.
- If a Kubernetes service exposes more than 1 port, multiple inbounds will be generated all with different
kuma.io/service
. - If a pod is attached to more than one Kubernetes service, multiple inbounds will also be generated.
Example
Will generate the following inbounds in your Kuma dataplane:
Notice how kuma.io/service
is built on <serviceName>_<namespace>_svc_<port>
and kuma.io/protocol
is the appProtocol
field of your service entry.
Capabilities
The only required
capability for the sidecar is NET_BIND_SERVICE
.
Use ContainerPatch
to
control capabilities for the sidecar.
Lifecycle
Joining the mesh
On Kubernetes, Dataplane
resource is automatically created by kuma-cp. For each Pod
with sidecar-injection label, a new
Dataplane
resource will be created.
To join the mesh in a graceful way, we need to first make sure the application is ready to serve traffic before it can be considered a valid traffic destination.
Init containers
Due to the way that Kuma implements transparent proxying and sidecars in Kubernetes, network calls from init containers while running a mesh can be a challenge.
Network calls to outside of the mesh
The common pitfall is the idea that it’s possible to order init containers so that the mesh init container is run after other init containers. However, when injecting these init containers into a Pod via webhooks, such as the Vault init container, there is no assurance of the order. The ordering of init containers also doesn’t provide a solution when the Kuma CNI is used, as traffic redirection to the sidecar occurs even before any init container runs.
To solve this issue, start the init container with a specific user ID and exclude specific ports from interception.
Remember also about excluding port of DNS interception. Here is an example of annotations to enable HTTPS traffic for a container running as user id 1234
.
Network calls inside the mesh with mTLS enabled
In this scenario, using the init container is simply impossible
because kuma-dp
is responsible for encrypting the traffic and only runs after all init containers have exited.
Leaving the mesh
To leave the mesh in a graceful shutdown, we need to remove the traffic destination from all the clients before shutting it down.
When the Kuma sidecar receives a SIGTERM signal it:
- Starts draining Envoy listeners.
- Waits the entire drain time.
- Terminates.
While draining, Envoy can still accept connections, however:
- It is marked unhealthy on the Envoy Admin
/ready
endpoint. - It sends
connection: close
for HTTP/1.1 requests and theGOAWAY
frame for HTTP/2. This forces clients to close their connection and reconnect to the new instance.
You can read the Kubernetes docs to learn how Kubernetes handles the Pod
lifecycle. Here is the summary including the parts relevant for Kuma.
Whenever a user or system deletes a Pod
, Kubernetes does the following:
- It marks the
Pod
as terminated. - For every container concurrently it:
- Executes any pre stop hook if defined.
- Sends a SIGTERM signal.
- Waits until container is terminated for maximum of graceful termination time (by default 60s).
- Sends a SIGKILL to the container.
- It removes the
Pod
object from the system.
When Pod
is marked as terminated, Kuma, the CP marks the Dataplane
object unhealthy, which triggers a configuration update to all the clients in order to remove it as a destination.
This can take a couple of seconds depending on the size of the mesh, resources available to the CP, XDS configuration interval, etc.
If the application served by the Kuma sidecar quits immediately after the SIGTERM signal, there is a high chance that clients will still try to send traffic to this destination.
To mitigate this, we need to either
- Support graceful shutdown in the application. For example, the application should wait X seconds to exit after receiving the first SIGTERM signal.
- Add a pre-stop hook to postpone stopping the application container. Example:
When a Pod
is deleted, its matching Dataplane
resource is deleted as well. This is possible thanks to the
owner reference set on the Dataplane
resource.
Custom Container Configuration
If you want to modify the default container configuration you can use
the ContainerPatch
Kubernetes CRD. It allows configuration of both sidecar
and init containers. ContainerPatch
resources are namespace scoped and can
only be applied in a namespace where Kuma CP is running.
In the vast majority of cases you shouldn’t need to override the sidecar and
init container configurations. ContainerPatch
is a feature which requires good
understanding of both Kuma and Kubernetes.
A ContainerPatch
specification consists of the list of JSON patch
strings that describe the modifications. Consult the entire
resource schema.
Example
When using ContainerPath, every value
field must be a string containing valid JSON.
This will change the securityContext
section of kuma-sidecar
container from:
to:
and similarly change the securityContext section of the init container from:
to:
Resources requests cpu
will be changed from:
to:
Resources limits
will be changed from:
to:
Workload matching
A ContainerPatch
is matched to a Pod
via an kuma.io/container-patches
annotation on the workload. Each annotation may be an ordered list of
ContainerPatch
names, which will be applied in the order specified.
If a workload refers to a ContainerPatch
which does not exist, the injection
will explicitly fail and log the failure.
Example
Default patches
You can configure kuma-cp
to apply the list of default patches for workloads
which don’t specify their own patches by modifying the containerPatches
value
from the kuma-dp
configuration:
If you specify the list of default patches (i.e. ["default-patch-1", "default-patch-2]
)
but your workload will be annotated with its own list of patches (i.e.
["pod-patch-1", "pod-patch-2]
) only the latter will be applied.
To install a CP with env vars you can do:
Error modes and validation
When applying ContainerPatch
Kuma will validate that the rendered container
spec meets the Kubernetes specification. Kuma will not validate that it is
a sane configuration.
If a workload refers to a ContainerPatch
which does not exist, the injection
will explicitly fail and log the failure.
Direct access to services
By default, on Kubernetes data plane proxies communicate with each other by leveraging the ClusterIP
address of the Service
resources. Also by default, any request made to another service is automatically load balanced client-side by the data plane proxy that originates the request (they are load balanced by the local Envoy proxy sidecar proxy).
There are situations where we may want to bypass the client-side load balancing and directly access services by using their IP address (ie: in the case of Prometheus wanting to scrape metrics from services by their individual IP address).
When an originating service wants to directly consume other services by their IP address, the originating service’s Deployment
resource must include the following annotation:
Where the value is a comma separated list of Kuma services that will be consumed directly. For example:
Note: When using direct access with headless service, destination service will be accessible at: Kuma-service.pod-name.mesh
We can also use *
to indicate direct access to every service in the Mesh:
Using *
to directly access every service is a resource intensive operation, so we must use it carefully.
Schema
ContainerPatch stores a list of patches to apply to init and sidecar containers.
Type: object
Properties
- apiVersion
- APIVersion defines the versioned schema of this representation of an object. Servers should convert recognized schemas to the latest internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources
- Type:
string
- kind
- Kind is a string value representing the REST resource this object represents. Servers may infer this from the endpoint the client submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds
- Type:
string
- mesh
- Type:
string
- Type:
- metadata
- Type:
object
- Properties
- Type:
- spec
- ContainerPatchSpec specifies the options available for a ContainerPatch
- Type:
object
- Properties
- initPatch
- InitPatch specifies jsonpatch to apply to an init container.
- Type:
array
- Items
- JsonPatchBlock is one json patch operation block.
- Type:
object
- Properties
- from
- From is a jsonpatch from string, used by move and copy operations.
- Type:
string
- op
required
- Op is a jsonpatch operation string.
- Type:
string
- The value is restricted to the following:
- "add"
- "remove"
- "replace"
- "move"
- "copy"
- path
required
- Path is a jsonpatch path string.
- Type:
string
- value
- Value must be a string representing a valid json object used by replace and add operations. String has to be escaped with " to be valid a json object.
- Type:
string
- from
- sidecarPatch
- SidecarPatch specifies jsonpatch to apply to a sidecar container.
- Type:
array
- Items
- JsonPatchBlock is one json patch operation block.
- Type:
object
- Properties
- from
- From is a jsonpatch from string, used by move and copy operations.
- Type:
string
- op
required
- Op is a jsonpatch operation string.
- Type:
string
- The value is restricted to the following:
- "add"
- "remove"
- "replace"
- "move"
- "copy"
- path
required
- Path is a jsonpatch path string.
- Type:
string
- value
- Value must be a string representing a valid json object used by replace and add operations. String has to be escaped with " to be valid a json object.
- Type:
string
- from
- initPatch
Generated with json-schema-md-doc