Skip to content

Latest commit

 

History

History
122 lines (83 loc) · 5.33 KB

constraint_creation.md

File metadata and controls

122 lines (83 loc) · 5.33 KB

How Constraints are Created

Policy File Parsing

When using konstraint create, Konstraint will only generate templates and constraints for policy files with at least one violation[] rule. The violation rule is required by Gatekeeper when evaluating policies on a Kubernetes cluster.

When using konstraint doc, Konstraint will create documentation for each policy file and assign a severity based on the rule names found in the policy.

The following rule names are organized into their own sections as they have special meaning within the context of Gatekeeper and Conftest:

  • violation (supported by Gatekeeper and Conftest)
  • warn (supported by Conftest)

If a policy file does not contain any of the above rules, the policy is added to the Other section.

Importing Libraries

The Rego for the libraries will be added to the generated ConstraintTemplate if and only if the policy imports the library. This helps prevent importing Rego code that will go unused.

Resource Naming

The name of the templates and constraints are derived from the name of the folder that the policy was found in.

For example, a policy found in: policies/pod-volume-size-limits/src.rego generates the following in the policies/pod-volume-size-limits directory:

  • template.yaml (defining a ConstraintTemplate)

    • kind: ConstraintTemplate
    • name: podvolumesizelimits
    • CRD kind (to add to Kubernetes API): PodVolumeSizeLimits
  • constraint.yaml (implementing the above ConstraintTemplate)

    • kind: PodVolumeSizeLimits
    • name: podvolumesizelimits

When using the --output parameter, all templates and constraints will be generated in the path specified in the parameter with the format:

  • constraint_PodVolumeSizeLimits.yaml
  • template_PodVolumeSizeLimits.yaml

NOTE: While not technically required, the tool works best with a folder structure similar to how Gatekeeper itself structures policies and templates.

Annotating Rules

To further promote that the .rego file is the source of truth for policy, a block comment can be added to each policy file.

This comment block should:

  • Include a human readable description of what the policy does.
  • Set the matchers used when generating the Constraints.

It may also specify the enforcement action (either deny or dryrun) that Gatekeeper should take when a resource violates the constraint. If no enforcement action is specified, Konstraint defaults to using deny to align with Gatekeeper's default action. If the enforcement is set to dryrun, the policy will be skipped in the documentation generation.

# @title Pods must not run with access to the host IPC
#
# Pods that are allowed to access the host IPC can read memory of
# the other containers, breaking that security boundary.
#
# @enforcement deny
# @kinds apps/DaemonSet apps/Deployment apps/StatefulSet core/Pod
package pod_deny_host_ipc

import data.lib.core
import data.lib.pods

violation[msg] {
    pod_has_hostipc

    msg := core.format(sprintf("%s/%s: Pod allows for accessing the host IPC", [core.kind, core.name]))
}

pod_has_hostipc {
    pods.pod.spec.hostIPC
}

The comment block is also what is used when generating documentation via the doc command.

Annotating rules for matchers

  • @kinds generates constraint.spec.match.kinds. The allowed format is space-separated <group>/<kind> list.
    # @kinds apps/DaemonSet apps/Deployment apps/StatefulSet core/Pod
    
  • @matchlabels generates constraint.spec.match.labelSelector.matchLabels. The allowed format is space-separated <label-key>=<label-val> list.
    # @matchlabels app.kubernetes.io/name=mysql app.kubernetes.io/version=5.8
    

Skipping generation of the Constraint resource

In some scenarios, you may wish for Konstraint to skip the generation of the Constraint resource for a policy and manage that externally. To do so, add the @skip-constraint tag in the header comment block.

Using Input Parameters

Gatekeeper has the ability for a single ConstraintTemplate resource to be used by multiple Constraints. One of the reasons for this is that it allows for passing input parameters to the policy so a single policy to avoid duplication. Konstraint supports these input parameters via @parameter tags in the header comment block. NOTE: When input parameters are specified, Konstraint skips the generation of the Constraint resource.

To use parameters, add one or more @parameter <name> <type> statements where <name> is the name of the parameter and <type> is the OpenAPI v3 type of the parameter (string, integer, etc.). Arrays are supported via @parameter <name> array <type>. Each parameter tag must be on its own line. Each parameter in the rule body must have a @parameter tag in the comment block header.

# @title Required Labels
#
# This policy allows you to require certain labels are set on a resource.
#
# @parameter labels array string
package required_labels

import data.lib.core

violation[msg] {
    missing := missing_labels
    count(missing) > 0

    msg := core.format(sprintf("%s/%s: Missing required labels: %v", [core.kind, core.name, missing]))
}

missing_labels = missing {
    provided := {label | core.labels[label]}
    required := {label | label := input.parameters.labels[_]}
    missing := required - provided
}