policy-library/policies/templates/gcp_gke_cluster_version_v1.yaml (89 lines of code) (raw):

# Copyright 2021 Google LLC # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # https://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. apiVersion: templates.gatekeeper.sh/v1alpha1 kind: ConstraintTemplate metadata: name: gcp-gke-cluster-version-v1 spec: crd: spec: names: kind: GKEClusterVersionConstraintV1 validation: openAPIV3Schema: properties: mode: type: string enum: [denylist, allowlist] description: "String identifying the operational mode, allowlist or denylist. In allowlist mode, GKE clusters are only allowed to have a master/node version specified in the 'version_type' parameter. In denylist mode, resources are allowed any master/node versions except those listed in the 'version_type' parameter. Default is allowlist." version_type: type: string enum: [master, node] description: "String identifying the GKE cluster version type to look for, master or node. master refers to the Kubernetes version for GKE cluster masters. node refers to the Kubernetes version for GKE cluster nodes. Default is master." exemptions: type: array items: type: string description: "Array of GKE clusters in the form of glob expressions to exempt from version restriction. Delimiter is '/'. E.g. `//container.googleapis.com/projects/project-abc/zones/*/clusters/*` will exempt any clusters whose asset name has the format '//container.googleapis.com/projects/project-abc/zones/{ANY ZONE}/clusters/{ANY NAME}'. such as: '//container.googleapis.com/projects/project-abc/zones/us-central1-c/clusters/my-cluster' but NOT '//container.googleapis.com/projects/wrong-project/zones/us-central1-c/clusters/wrong-cluster'" versions: type: array items: type: string description: "Array of versions that GKE clusters are allowed to have, e.g. 1.12.10-gke.17 for master version_type or 1.4.4 for node version_type." targets: validation.gcp.forsetisecurity.org: rego: | # # Copyright 2021 Google LLC # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # https://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # package templates.gcp.GKEClusterVersionConstraintV1 import data.validator.gcp.lib as lib deny[{ "msg": message, "details": metadata, }] { constraint := input.constraint lib.get_constraint_params(constraint, params) asset := input.asset asset.asset_type == "container.googleapis.com/Cluster" mode := lib.get_default(params, "mode", "allowlist") # Check if resource is in exempt list exempt_list := lib.get_default(params, "exemptions", []) not is_exempt(exempt_list, asset.name) # Get the version value in the asset params_version_type := lib.get_default(params, "version_type", "master") target_version_type := get_version_type(params_version_type) target_version := get_target_value(asset.resource.data, target_version_type) # Check that version is in allowlist/denylist asset_version := params.versions version_matches := {target_version} & cast_set(asset_version) desired_count := target_version_match_count(params.mode) count(version_matches) == desired_count message := sprintf("Cluster %v has a disallowed %v field", [asset.name, target_version_type]) metadata := {"resource": asset.name} } ########################### # Rule Utilities ########################### get_version_type(version_type) = target_version { version_type == "master" target_version := "currentMasterVersion" } get_version_type(version_type) = target_version { version_type == "node" target_version := "currentNodeVersion" } is_exempt(exempt_list, asset_name) { exempt_list != [] glob.match(exempt_list[_], ["/"], asset_name) } get_target_value(data_resource, field_name) = output { output := lib.get_default(data_resource, field_name, "") } # Determine the overlap between versions under test and constraint # By default (allowlist), we violate if there isn't overlap. target_version_match_count(mode) = 0 { mode != "denylist" } target_version_match_count(mode) = 1 { mode == "denylist" } #ENDINLINE