Skip to content
This repository has been archived by the owner on Apr 1, 2021. It is now read-only.

Latest commit

 

History

History
226 lines (168 loc) · 6.28 KB

README.md

File metadata and controls

226 lines (168 loc) · 6.28 KB
Continuous integration License
Continuous integration License: Apache 2.0

This project defines a Chimera Policy written in Rust.

The goal

Given the following scenario:

As an operator of a Kubernetes cluster used by multiple tenants, I have reserved a set of nodes to a specific tenant, hence I want a policy that prevents untrusted users from running their workloads on these reserved nodes.

This scenario can be implemented using the concepts of taints and tolerations built into Kubernetes:

  1. Reserved nodes are tainted by the cluster operator. That prevents generic workloads from being scheduled on them.
  2. Trusted users should put add a "special" toleration to the workloads that must be scheduled on their reserved nodes.
  3. Untrusted users should not be able to fool the Kubernetes scheduler by adding the "special" toleration used by the trusted users.

Unfortunately Kubernetes doesn't have any built-in mechanism that can solve the last point. This is a task for a specially crafted dynamic admissions controller, like Chimera.

This Chimera Policy ensures only trusted users can schedule workloads that have the toleration required to be running on the reserved nodes.

The policy does that by inspecting CREATE and UPDATE requests of Pod resources.

Usage

Let's assume some nodes of the cluster have been tainted in this way by the cluster operator:

$ kubectl taint nodes node1 dedicated=tenantA:NoSchedule
$ kubectl taint nodes node2 dedicated=tenantA:NoSchedule
$ # goes on...

That means regular workloads, regardless of the tenant, will never be scheduled to these nodes.

Only workloads defined with the right toleration can be scheduled on them.

For example, this workload could be scheduled on one of these nodes:

apiVersion: v1
kind: Pod
metadata:
  name: nginx
spec:
  containers:
  - name: nginx
    image: nginx
    imagePullPolicy: IfNotPresent
  tolerations:
  - key: "dedicated"
    operator: "Equal"
    value: "tenantA"
    effect: "NoSchedule"

Also this workload could be scheduled on the reserved nodes:

apiVersion: v1
kind: Pod
metadata:
  name: nginx
spec:
  containers:
  - name: nginx
    image: nginx
    imagePullPolicy: IfNotPresent
  tolerations:
  - key: "dedicated"
    operator: "Exists"
    effect: "NoSchedule"

Note well: a toleration with the Exists operator could be abused by an evil user to allow his workloads to be scheduled on a dedicated node.

How the policy works

This policy allows the cluster operator to define a taint to be "protected". By protecting a taint nobody in the cluster will be able to tolerate it via the Exists operator.

Only the trusted users will be allowed to create Pods tolerating the protected taint with a toleration that uses the Equal operator.

Configuration

The policy has no hard-coded value neither for the taint nor for the usernames or groups that are entitled to tolerate the taint.

The policy can be configured with the following data structure:

taint:
  key: dedicated # replace with the name of the taint
  value: tenantA # replace with the value of the taint
allowed_groups: # list of groups
- administrators
- system:masters
allowed_users: # list of users
- joe

Let's go through each field:

  • taint: dictionary with key and value keys. This defines the taint to monitor. Required.
  • allowed_users: list of users who are allowed to use this toleration. Optional.
  • allowed_groups: list of groups who are allowed to use this toleration. Optional.

Obtain policy

The policy is automatically published as an OCI artifact inside of this container registry:

Example

Given a cluster with:

  • Two groups of users: tenantA-users and tenantB-users
  • Some nodes tainted with the taint dedicated:tenantA

And a policy instantiated with this configuration:

taint:
  key: dedicated
  value: tenantA
allowed_groups:
- tenantA-users

A Pod using the following toleration:

  tolerations:
  - key: "dedicated"
    operator: "Exists"
    effect: "NoSchedule"

Will always be rejected by the policy, regardless of the group to which the user belongs to.

On the other hand, a Pod defined with this toleration:

  tolerations:
  - key: "dedicated"
    operator: "Equal"
    value: "tenantA"
    effect: "NoSchedule"

Will be accepted only when created by a user who belongs to the group tenantA-users.

Finally, a Pod with the following toleration:

  tolerations:
  - key: "dedicated"
    operator: "Equal"
    value: "experiments"
    effect: "NoSchedule"

Would never be rejected by the policy.

Building

Handle the rust installation using rustup. Then add the wasm32-wasi target:

$ rustup target add wasm32-unknown-unknown

Use this command to build the Wasm code:

$ make build

This will produce a .wasm file under target/wasm32-unknown-unknown/release/pod_toleration_policy.wasm.

Using the policy

The easiest way to use this policy is through the chimera-controller.

Trying the policy

The policy can be ran outside of Chimera's policy-server by using the chimera-policy-testdrive CLI tool:

$ chimera-policy-testdrive \
    --policy target/wasm32-unknown-unknown/release/pod_toleration_policy.wasm \
    -settings '{"taint": {"key": "dedicated", "value": "tenantA"}, "allowed_groups": ["administrators", "system:masters"]}'
    --request-file test_data/req_pod_with_equal_toleration.json

This will produce the following output:

{"accepted":false,"message":"User not allowed to create Pods that tolerate the taint key: dedicated, value : tenantA"}

You can find more example files under the test_data directory.