From a38a3c5bbcd83a766e923fccf0ed3605f91885be Mon Sep 17 00:00:00 2001
From: Peter Balogh
Date: Fri, 10 Jan 2025 10:18:50 +0100
Subject: [PATCH] feat: CIS EKS 1.5.0 (#1653)
* feat(cfg): add EKS 1.5.0
* fix(cfg): target map
* fix: update eks job
* fix: target mapping
* feat: use CIS EKS 1.5.0 by default
* fix: scored in node.yaml
Signed-off-by: Peter Balogh
* doc: add CIS EKS 1.5.0
Signed-off-by: Peter Balogh
---------
Signed-off-by: Peter Balogh
---
cfg/config.yaml | 7 +
cfg/eks-1.5.0/config.yaml | 9 +
cfg/eks-1.5.0/controlplane.yaml | 32 ++
cfg/eks-1.5.0/managedservices.yaml | 227 +++++++++++++++
cfg/eks-1.5.0/master.yaml | 6 +
cfg/eks-1.5.0/node.yaml | 453 +++++++++++++++++++++++++++++
cfg/eks-1.5.0/policies.yaml | 250 ++++++++++++++++
cmd/common_test.go | 6 +
cmd/util.go | 2 +-
cmd/util_test.go | 2 +-
docs/architecture.md | 7 +-
docs/platforms.md | 3 +-
job-eks.yaml | 4 +-
13 files changed, 1000 insertions(+), 8 deletions(-)
create mode 100644 cfg/eks-1.5.0/config.yaml
create mode 100644 cfg/eks-1.5.0/controlplane.yaml
create mode 100644 cfg/eks-1.5.0/managedservices.yaml
create mode 100644 cfg/eks-1.5.0/master.yaml
create mode 100644 cfg/eks-1.5.0/node.yaml
create mode 100644 cfg/eks-1.5.0/policies.yaml
diff --git a/cfg/config.yaml b/cfg/config.yaml
index d5d170bbd..72afbed3c 100644
--- a/cfg/config.yaml
+++ b/cfg/config.yaml
@@ -286,6 +286,7 @@ version_mapping:
"eks-1.0.1": "eks-1.0.1"
"eks-1.1.0": "eks-1.1.0"
"eks-1.2.0": "eks-1.2.0"
+ "eks-1.5.0": "eks-1.5.0"
"gke-1.0": "gke-1.0"
"gke-1.2.0": "gke-1.2.0"
"gke-1.6.0": "gke-1.6.0"
@@ -405,6 +406,12 @@ target_mapping:
- "controlplane"
- "policies"
- "managedservices"
+ "eks-1.5.0":
+ - "master"
+ - "node"
+ - "controlplane"
+ - "policies"
+ - "managedservices"
"rh-0.7":
- "master"
- "node"
diff --git a/cfg/eks-1.5.0/config.yaml b/cfg/eks-1.5.0/config.yaml
new file mode 100644
index 000000000..17301a751
--- /dev/null
+++ b/cfg/eks-1.5.0/config.yaml
@@ -0,0 +1,9 @@
+---
+## Version-specific settings that override the values in cfg/config.yaml
+## These settings are required if you are using the --asff option to report findings to AWS Security Hub
+## AWS account number is required.
+AWS_ACCOUNT: ""
+## AWS region is required.
+AWS_REGION: ""
+## EKS Cluster ARN is required.
+CLUSTER_ARN: ""
diff --git a/cfg/eks-1.5.0/controlplane.yaml b/cfg/eks-1.5.0/controlplane.yaml
new file mode 100644
index 000000000..6323f03c6
--- /dev/null
+++ b/cfg/eks-1.5.0/controlplane.yaml
@@ -0,0 +1,32 @@
+---
+controls:
+version: "eks-1.5.0"
+id: 2
+text: "Control Plane Configuration"
+type: "controlplane"
+groups:
+ - id: 2.1
+ text: "Logging"
+ checks:
+ - id: 2.1.1
+ text: "Enable audit Logs (Automated)"
+ remediation: |
+ From Console:
+ 1. For each EKS Cluster in each region;
+ 2. Go to 'Amazon EKS' > 'Clusters' > '' > 'Configuration' > 'Logging'.
+ 3. Click 'Manage logging'.
+ 4. Ensure that all options are toggled to 'Enabled'.
+ API server: Enabled
+ Audit: Enabled
+ Authenticator: Enabled
+ Controller manager: Enabled
+ Scheduler: Enabled
+ 5. Click 'Save Changes'.
+
+ From CLI:
+ # For each EKS Cluster in each region;
+ aws eks update-cluster-config \
+ --region '${REGION_CODE}' \
+ --name '${CLUSTER_NAME}' \
+ --logging '{"clusterLogging":[{"types":["api","audit","authenticator","controllerManager","scheduler"],"enabled":true}]}'
+ scored: false
diff --git a/cfg/eks-1.5.0/managedservices.yaml b/cfg/eks-1.5.0/managedservices.yaml
new file mode 100644
index 000000000..4ad870acd
--- /dev/null
+++ b/cfg/eks-1.5.0/managedservices.yaml
@@ -0,0 +1,227 @@
+---
+controls:
+version: "eks-1.5.0"
+id: 5
+text: "Managed Services"
+type: "managedservices"
+groups:
+ - id: 5.1
+ text: "Image Registry and Image Scanning"
+ checks:
+ - id: 5.1.1
+ text: "Ensure Image Vulnerability Scanning using Amazon ECR image scanning or a third party provider (Automated)"
+ type: "manual"
+ remediation: |
+ To utilize AWS ECR for Image scanning please follow the steps below:
+
+ To create a repository configured for scan on push (AWS CLI):
+ aws ecr create-repository --repository-name $REPO_NAME --image-scanning-configuration scanOnPush=true --region $REGION_CODE
+
+ To edit the settings of an existing repository (AWS CLI):
+ aws ecr put-image-scanning-configuration --repository-name $REPO_NAME --image-scanning-configuration scanOnPush=true --region $REGION_CODE
+
+ Use the following steps to start a manual image scan using the AWS Management Console.
+
+ 1. Open the Amazon ECR console at https://console.aws.amazon.com/ecr/repositories.
+ 2. From the navigation bar, choose the Region to create your repository in.
+ 3. In the navigation pane, choose Repositories.
+ 4. On the Repositories page, choose the repository that contains the image to scan.
+ 5. On the Images page, select the image to scan and then choose Scan.
+ scored: false
+
+ - id: 5.1.2
+ text: "Minimize user access to Amazon ECR (Manual)"
+ type: "manual"
+ remediation: |
+ Before you use IAM to manage access to Amazon ECR, you should understand what IAM features
+ are available to use with Amazon ECR. To get a high-level view of how Amazon ECR and other
+ AWS services work with IAM, see AWS Services That Work with IAM in the IAM User Guide.
+ scored: false
+
+ - id: 5.1.3
+ text: "Minimize cluster access to read-only for Amazon ECR (Manual)"
+ type: "manual"
+ remediation: |
+ You can use your Amazon ECR images with Amazon EKS, but you need to satisfy the following prerequisites.
+
+ The Amazon EKS worker node IAM role (NodeInstanceRole) that you use with your worker nodes must possess
+ the following IAM policy permissions for Amazon ECR.
+
+ {
+ "Version": "2012-10-17",
+ "Statement": [
+ {
+ "Effect": "Allow",
+ "Action": [
+ "ecr:BatchCheckLayerAvailability",
+ "ecr:BatchGetImage",
+ "ecr:GetDownloadUrlForLayer",
+ "ecr:GetAuthorizationToken"
+ ],
+ "Resource": "*"
+ }
+ ]
+ }
+ scored: false
+
+ - id: 5.1.4
+ text: "Minimize Container Registries to only those approved (Manual)"
+ type: "manual"
+ remediation: |
+ To minimize AWS ECR container registries to only those approved, you can follow these steps:
+
+ 1. Define your approval criteria: Determine the criteria that containers must meet to
+ be considered approved. This can include factors such as security, compliance,
+ compatibility, and other requirements.
+ 2. Identify all existing ECR registries: Identify all ECR registries that are currently
+ being used in your organization.
+ 3. Evaluate ECR registries against approval criteria: Evaluate each ECR registry
+ against your approval criteria to determine whether it should be approved or not.
+ This can be done by reviewing the registry settings and configuration, as well as
+ conducting security assessments and vulnerability scans.
+ 4. Establish policies and procedures: Establish policies and procedures that outline
+ how ECR registries will be approved, maintained, and monitored. This should
+ include guidelines for developers to follow when selecting a registry for their
+ container images.
+ 5. Implement access controls: Implement access controls to ensure that only
+ approved ECR registries are used to store and distribute container images. This
+ can be done by setting up IAM policies and roles that restrict access to
+ unapproved registries or create a whitelist of approved registries.
+ 6. Monitor and review: Continuously monitor and review the use of ECR registries
+ to ensure that they continue to meet your approval criteria. This can include
+ scored: false
+
+ - id: 5.2
+ text: "Identity and Access Management (IAM)"
+ checks:
+ - id: 5.2.1
+ text: "Prefer using dedicated Amazon EKS Service Accounts (Automated)"
+ type: "manual"
+ remediation: |
+ With IAM roles for service accounts on Amazon EKS clusters, you can associate an
+ IAM role with a Kubernetes service account. This service account can then provide
+ AWS permissions to the containers in any pod that uses that service account. With this
+ feature, you no longer need to provide extended permissions to the worker node IAM
+ role so that pods on that node can call AWS APIs.
+ Applications must sign their AWS API requests with AWS credentials. This feature
+ provides a strategy for managing credentials for your applications, similar to the way
+ that Amazon EC2 instance profiles provide credentials to Amazon EC2 instances.
+ Instead of creating and distributing your AWS credentials to the containers or using the
+ Amazon EC2 instance’s role, you can associate an IAM role with a Kubernetes service
+ account. The applications in the pod’s containers can then use an AWS SDK or the
+ AWS CLI to make API requests to authorized AWS services.
+
+ The IAM roles for service accounts feature provides the following benefits:
+
+ - Least privilege - By using the IAM roles for service accounts feature, you no
+ longer need to provide extended permissions to the worker node IAM role so that
+ pods on that node can call AWS APIs. You can scope IAM permissions to a
+ service account, and only pods that use that service account have access to
+ those permissions. This feature also eliminates the need for third-party solutions
+ such as kiam or kube2iam.
+ - Credential isolation - A container can only retrieve credentials for the IAM role
+ that is associated with the service account to which it belongs. A container never
+ has access to credentials that are intended for another container that belongs to
+ another pod.
+ - Audit-ability - Access and event logging is available through CloudTrail to help
+ ensure retrospective auditing.
+ scored: false
+
+ - id: 5.3
+ text: "AWS EKS Key Management Service"
+ checks:
+ - id: 5.3.1
+ text: "Ensure Kubernetes Secrets are encrypted using Customer Master Keys (CMKs) managed in AWS KMS (Manual)"
+ type: "manual"
+ remediation: |
+ This process can only be performed during Cluster Creation.
+
+ Enable 'Secrets Encryption' during Amazon EKS cluster creation as described
+ in the links within the 'References' section.
+ scored: false
+
+ - id: 5.4
+ text: "Cluster Networking"
+ checks:
+ - id: 5.4.1
+ text: "Restrict Access to the Control Plane Endpoint (Automated)"
+ type: "manual"
+ remediation: |
+ By enabling private endpoint access to the Kubernetes API server, all communication
+ between your nodes and the API server stays within your VPC. You can also limit the IP
+ addresses that can access your API server from the internet, or completely disable
+ internet access to the API server.
+ With this in mind, you can update your cluster accordingly using the AWS CLI to ensure
+ that Private Endpoint Access is enabled.
+ If you choose to also enable Public Endpoint Access then you should also configure a
+ list of allowable CIDR blocks, resulting in restricted access from the internet. If you
+ specify no CIDR blocks, then the public API server endpoint is able to receive and
+ process requests from all IP addresses by defaulting to ['0.0.0.0/0'].
+ For example, the following command would enable private access to the Kubernetes
+ API as well as limited public access over the internet from a single IP address (noting
+ the /32 CIDR suffix):
+ aws eks update-cluster-config --region $AWS_REGION --name $CLUSTER_NAME --resources-vpc-config endpointPrivateAccess=true,endpointPrivateAccess=true,publicAccessCidrs="203.0.113.5/32"
+
+ Note: The CIDR blocks specified cannot include reserved addresses.
+ There is a maximum number of CIDR blocks that you can specify. For more information,
+ see the EKS Service Quotas link in the references section.
+ For more detailed information, see the EKS Cluster Endpoint documentation link in the
+ references section.
+ scored: false
+
+ - id: 5.4.2
+ text: "Ensure clusters are created with Private Endpoint Enabled and Public Access Disabled (Automated)"
+ type: "manual"
+ remediation: |
+ By enabling private endpoint access to the Kubernetes API server, all communication
+ between your nodes and the API server stays within your VPC.
+ With this in mind, you can update your cluster accordingly using the AWS CLI to ensure
+ that Private Endpoint Access is enabled.
+ For example, the following command would enable private access to the Kubernetes
+ API and ensure that no public access is permitted:
+ aws eks update-cluster-config --region $AWS_REGION --name $CLUSTER_NAME --resources-vpc-config endpointPrivateAccess=true,endpointPublicAccess=false
+
+ Note: For more detailed information, see the EKS Cluster Endpoint documentation link
+ in the references section.
+ scored: false
+
+ - id: 5.4.3
+ text: "Ensure clusters are created with Private Nodes (Automated)"
+ type: "manual"
+ remediation: |
+ aws eks update-cluster-config \
+ --region region-code \
+ --name my-cluster \
+ --resources-vpc-config endpointPublicAccess=true,publicAccessCidrs="203.0.113.5/32",endpointPrivateAccess=true
+ scored: false
+
+ - id: 5.4.4
+ text: "Ensure Network Policy is Enabled and set as appropriate (Automated)"
+ type: "manual"
+ remediation: |
+ Utilize Calico or other network policy engine to segment and isolate your traffic.
+ scored: false
+
+ - id: 5.4.5
+ text: "Encrypt traffic to HTTPS load balancers with TLS certificates (Manual)"
+ type: "manual"
+ remediation: |
+ Your load balancer vendor can provide details on configuring HTTPS with TLS.
+ scored: false
+
+
+ - id: 5.5
+ text: "Authentication and Authorization"
+ checks:
+ - id: 5.5.1
+ text: "Manage Kubernetes RBAC users with AWS IAM Authenticator for Kubernetes or Upgrade to AWS CLI v1.16.156 or greater (Manual)"
+ type: "manual"
+ remediation: |
+ Refer to the 'Managing users or IAM roles for your cluster' in Amazon EKS documentation.
+
+ Note: If using AWS CLI version 1.16.156 or later there is no need to install the AWS
+ IAM Authenticator anymore.
+ The relevant AWS CLI commands, depending on the use case, are:
+ aws eks update-kubeconfig
+ aws eks get-token
+ scored: false
diff --git a/cfg/eks-1.5.0/master.yaml b/cfg/eks-1.5.0/master.yaml
new file mode 100644
index 000000000..8aba89c7e
--- /dev/null
+++ b/cfg/eks-1.5.0/master.yaml
@@ -0,0 +1,6 @@
+---
+controls:
+version: "eks-1.5.0"
+id: 1
+text: "Control Plane Components"
+type: "master"
diff --git a/cfg/eks-1.5.0/node.yaml b/cfg/eks-1.5.0/node.yaml
new file mode 100644
index 000000000..c9ff2ca6d
--- /dev/null
+++ b/cfg/eks-1.5.0/node.yaml
@@ -0,0 +1,453 @@
+---
+controls:
+version: "eks-1.5.0"
+id: 3
+text: "Worker Node Security Configuration"
+type: "node"
+groups:
+ - id: 3.1
+ text: "Worker Node Configuration Files"
+ checks:
+ - id: 3.1.1
+ text: "Ensure that the kubeconfig file permissions are set to 644 or more restrictive (Automated)"
+ audit: '/bin/sh -c ''if test -e $kubeletkubeconfig; then stat -c permissions=%a $kubeletkubeconfig; fi'' '
+ tests:
+ test_items:
+ - flag: "permissions"
+ compare:
+ op: bitmask
+ value: "644"
+ remediation: |
+ Run the below command (based on the file location on your system) on the each worker node.
+ For example,
+ chmod 644 $kubeletkubeconfig
+ scored: true
+
+ - id: 3.1.2
+ text: "Ensure that the kubelet kubeconfig file ownership is set to root:root (Automated)"
+ audit: '/bin/sh -c ''if test -e $kubeletkubeconfig; then stat -c %U:%G $kubeletkubeconfig; fi'' '
+ tests:
+ test_items:
+ - flag: root:root
+ remediation: |
+ Run the below command (based on the file location on your system) on the each worker node.
+ For example,
+ chown root:root $kubeletkubeconfig
+ scored: true
+
+ - id: 3.1.3
+ text: "Ensure that the kubelet configuration file has permissions set to 644 or more restrictive (Automated)"
+ audit: '/bin/sh -c ''if test -e $kubeletconf; then stat -c permissions=%a $kubeletconf; fi'' '
+ tests:
+ test_items:
+ - flag: "permissions"
+ compare:
+ op: bitmask
+ value: "644"
+ remediation: |
+ Run the following command (using the config file location identified in the Audit step)
+ chmod 644 $kubeletconf
+ scored: true
+
+ - id: 3.1.4
+ text: "Ensure that the kubelet configuration file ownership is set to root:root (Automated)"
+ audit: '/bin/sh -c ''if test -e $kubeletconf; then stat -c %U:%G $kubeletconf; fi'' '
+ tests:
+ test_items:
+ - flag: root:root
+ remediation: |
+ Run the following command (using the config file location identified in the Audit step)
+ chown root:root $kubeletconf
+ scored: true
+
+ - id: 3.2
+ text: "Kubelet"
+ checks:
+ - id: 3.2.1
+ text: "Ensure that the Anonymous Auth is Not Enabled (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: "--anonymous-auth"
+ path: '{.authentication.anonymous.enabled}'
+ set: true
+ compare:
+ op: eq
+ value: false
+ remediation: |
+ Remediation Method 1:
+ If configuring via the Kubelet config file, you first need to locate the file.
+ To do this, SSH to each node and execute the following command to find the kubelet
+ process:
+ ps -ef | grep kubelet
+ The output of the above command provides details of the active kubelet process, from
+ which we can see the location of the configuration file provided to the kubelet service
+ with the --config argument. The file can be viewed with a command such as more or
+ less, like so:
+ sudo less /path/to/kubelet-config.json
+ Disable Anonymous Authentication by setting the following parameter:
+ "authentication": { "anonymous": { "enabled": false } }
+
+ Remediation Method 2.
+ If using executable arguments, edit the kubelet service file on each worker node and
+ ensure the below parameters are part of the KUBELET_ARGS variable string.
+ For systems using systemd, such as the Amazon EKS Optimised Amazon Linux or
+ Bottlerocket AMIs, then this file can be found at
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf. Otherwise,
+ you may need to look up documentation for your chosen operating system to determine
+ which service manager is configured:
+ --anonymous-auth=false
+
+ For Both Remediation Steps:
+ Based on your system, restart the kubelet service and check the service status.
+ The following example is for operating systems using systemd, such as the Amazon
+ EKS Optimised Amazon Linux or Bottlerocket AMIs, and invokes the systemctl
+ command. If systemctl is not available then you will need to look up documentation for
+ your chosen operating system to determine which service manager is configured:
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
+
+ - id: 3.2.2
+ text: "Ensure that the --authorization-mode argument is not set to AlwaysAllow (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: --authorization-mode
+ path: '{.authorization.mode}'
+ set: true
+ compare:
+ op: nothave
+ value: AlwaysAllow
+ remediation: |
+ Remediation Method 1:
+ If configuring via the Kubelet config file, you first need to locate the file.
+ To do this, SSH to each node and execute the following command to find the kubelet
+ process:
+ ps -ef | grep kubelet
+ The output of the above command provides details of the active kubelet process, from
+ which we can see the location of the configuration file provided to the kubelet service
+ with the --config argument. The file can be viewed with a command such as more or
+ less, like so:
+ sudo less /path/to/kubelet-config.json
+ Enable Webhook Authentication by setting the following parameter:
+ "authentication": { "webhook": { "enabled": true } }
+ Next, set the Authorization Mode to Webhook by setting the following parameter:
+ "authorization": { "mode": "Webhook }
+ Finer detail of the authentication and authorization fields can be found in the
+ Kubelet Configuration documentation.
+
+ Remediation Method 2:
+ If using executable arguments, edit the kubelet service file on each worker node and
+ ensure the below parameters are part of the KUBELET_ARGS variable string.
+ For systems using systemd, such as the Amazon EKS Optimised Amazon Linux or
+ Bottlerocket AMIs, then this file can be found at
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf. Otherwise,
+ you may need to look up documentation for your chosen operating system to determine
+ which service manager is configured:
+ --authentication-token-webhook
+ --authorization-mode=Webhook
+
+ For Both Remediation Steps:
+ Based on your system, restart the kubelet service and check the service status.
+ The following example is for operating systems using systemd, such as the Amazon
+ EKS Optimised Amazon Linux or Bottlerocket AMIs, and invokes the systemctl
+ command. If systemctl is not available then you will need to look up documentation for
+ your chosen operating system to determine which service manager is configured:
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
+
+ - id: 3.2.3
+ text: "Ensure that a Client CA File is Configured (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: --client-ca-file
+ path: '{.authentication.x509.clientCAFile}'
+ set: true
+ remediation: |
+ Remediation Method 1:
+ If configuring via the Kubelet config file, you first need to locate the file.
+ To do this, SSH to each node and execute the following command to find the kubelet
+ process:
+ ps -ef | grep kubelet
+ The output of the above command provides details of the active kubelet process, from
+ which we can see the location of the configuration file provided to the kubelet service
+ with the --config argument. The file can be viewed with a command such as more or
+ less, like so:
+ sudo less /path/to/kubelet-config.json
+ Configure the client certificate authority file by setting the following parameter
+ appropriately:
+ "authentication": { "x509": {"clientCAFile": } }"
+
+ Remediation Method 2:
+ If using executable arguments, edit the kubelet service file on each worker node and
+ ensure the below parameters are part of the KUBELET_ARGS variable string.
+ For systems using systemd, such as the Amazon EKS Optimised Amazon Linux or
+ Bottlerocket AMIs, then this file can be found at
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf. Otherwise,
+ you may need to look up documentation for your chosen operating system to determine
+ which service manager is configured:
+ --client-ca-file=
+
+ For Both Remediation Steps:
+ Based on your system, restart the kubelet service and check the service status.
+ The following example is for operating systems using systemd, such as the Amazon
+ EKS Optimised Amazon Linux or Bottlerocket AMIs, and invokes the systemctl
+ command. If systemctl is not available then you will need to look up documentation for
+ your chosen operating system to determine which service manager is configured:
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
+
+ - id: 3.2.4
+ text: "Ensure that the --read-only-port is disabled (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: "--read-only-port"
+ path: '{.readOnlyPort}'
+ set: true
+ compare:
+ op: eq
+ value: 0
+ remediation: |
+ If modifying the Kubelet config file, edit the kubelet-config.json file
+ /etc/kubernetes/kubelet/kubelet-config.json and set the below parameter to 0
+ "readOnlyPort": 0
+ If using executable arguments, edit the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf on each
+ worker node and add the below parameter at the end of the KUBELET_ARGS variable
+ string.
+ --read-only-port=0
+
+ Based on your system, restart the kubelet service and check status
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
+
+ - id: 3.2.5
+ text: "Ensure that the --streaming-connection-idle-timeout argument is not set to 0 (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: --streaming-connection-idle-timeout
+ path: '{.streamingConnectionIdleTimeout}'
+ set: true
+ compare:
+ op: noteq
+ value: 0
+ - flag: --streaming-connection-idle-timeout
+ path: '{.streamingConnectionIdleTimeout}'
+ set: false
+ bin_op: or
+ remediation: |
+ Remediation Method 1:
+ If modifying the Kubelet config file, edit the kubelet-config.json file
+ /etc/kubernetes/kubelet/kubelet-config.json and set the below parameter to a
+ non-zero value in the format of #h#m#s
+ "streamingConnectionIdleTimeout": "4h0m0s"
+ You should ensure that the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf does not
+ specify a --streaming-connection-idle-timeout argument because it would
+ override the Kubelet config file.
+
+ Remediation Method 2:
+ If using executable arguments, edit the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf on each
+ worker node and add the below parameter at the end of the KUBELET_ARGS variable
+ string.
+ --streaming-connection-idle-timeout=4h0m0s
+
+ Remediation Method 3:
+ If using the api configz endpoint consider searching for the status of
+ "streamingConnectionIdleTimeout": by extracting the live configuration from the
+ nodes running kubelet.
+ **See detailed step-by-step configmap procedures in Reconfigure a Node's Kubelet in a
+ Live Cluster, and then rerun the curl statement from audit process to check for kubelet
+ configuration changes
+ kubectl proxy --port=8001 &
+ export HOSTNAME_PORT=localhost:8001 (example host and port number)
+ export NODE_NAME=ip-192.168.31.226.ec2.internal (example node name from "kubectl get nodes")
+ curl -sSL "http://${HOSTNAME_PORT}/api/v1/nodes/${NODE_NAME}/proxy/configz"
+
+ For all three remediations:
+ Based on your system, restart the kubelet service and check status
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
+
+ - id: 3.2.6
+ text: "Ensure that the --make-iptables-util-chains argument is set to true (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: --make-iptables-util-chains
+ path: '{.makeIPTablesUtilChains}'
+ set: true
+ compare:
+ op: eq
+ value: true
+ - flag: --make-iptables-util-chains
+ path: '{.makeIPTablesUtilChains}'
+ set: false
+ bin_op: or
+ remediation: |
+ Remediation Method 1:
+ If modifying the Kubelet config file, edit the kubelet-config.json file
+ /etc/kubernetes/kubelet/kubelet-config.json and set the below parameter to
+ true
+ "makeIPTablesUtilChains": true
+ Ensure that /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf
+ does not set the --make-iptables-util-chains argument because that would
+ override your Kubelet config file.
+
+ Remediation Method 2:
+ If using executable arguments, edit the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf on each
+ worker node and add the below parameter at the end of the KUBELET_ARGS variable
+ string.
+ --make-iptables-util-chains:true
+
+ Remediation Method 3:
+ If using the api configz endpoint consider searching for the status of
+ "makeIPTablesUtilChains.: true by extracting the live configuration from the nodes
+ running kubelet.
+ **See detailed step-by-step configmap procedures in Reconfigure a Node's Kubelet in a
+ Live Cluster, and then rerun the curl statement from audit process to check for kubelet
+ configuration changes
+ kubectl proxy --port=8001 &
+ export HOSTNAME_PORT=localhost:8001 (example host and port number)
+ export NODE_NAME=ip-192.168.31.226.ec2.internal (example node name from "kubectl get nodes")
+ curl -sSL "http://${HOSTNAME_PORT}/api/v1/nodes/${NODE_NAME}/proxy/configz"
+
+ For all three remediations:
+ Based on your system, restart the kubelet service and check status
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
+
+ - id: 3.2.7
+ text: "Ensure that the --eventRecordQPS argument is set to 0 or a level which ensures appropriate event capture (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: --event-qps
+ path: '{.eventRecordQPS}'
+ set: true
+ compare:
+ op: gte
+ value: 0
+ remediation: |
+ If using a Kubelet config file, edit the file to set eventRecordQPS: to an appropriate
+ level.
+ If using command line arguments, edit the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubeadm.conf on each worker node
+ and set the below parameter in KUBELET_SYSTEM_PODS_ARGS variable.
+ Based on your system, restart the kubelet service. For example:
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ scored: true
+
+ - id: 3.2.8
+ text: "Ensure that the --rotate-certificates argument is not present or is set to true (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: --rotate-certificates
+ path: '{.rotateCertificates}'
+ set: true
+ compare:
+ op: eq
+ value: true
+ - flag: --rotate-certificates
+ path: '{.rotateCertificates}'
+ set: false
+ bin_op: or
+ remediation: |
+ Remediation Method 1:
+ If modifying the Kubelet config file, edit the kubelet-config.json file
+ /etc/kubernetes/kubelet/kubelet-config.json and set the below parameter to
+ true
+ "RotateCertificate":true
+ Additionally, ensure that the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf does not set the --RotateCertificate
+ executable argument to false because this would override the Kubelet
+ config file.
+
+ Remediation Method 2:
+ If using executable arguments, edit the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf on each
+ worker node and add the below parameter at the end of the KUBELET_ARGS variable
+ string.
+ --RotateCertificate=true
+ scored: true
+
+ - id: 3.2.9
+ text: "Ensure that the RotateKubeletServerCertificate argument is set to true (Automated)"
+ audit: "/bin/ps -fC $kubeletbin"
+ audit_config: "/bin/cat $kubeletconf"
+ tests:
+ test_items:
+ - flag: RotateKubeletServerCertificate
+ path: '{.featureGates.RotateKubeletServerCertificate}'
+ set: true
+ compare:
+ op: eq
+ value: true
+ remediation: |
+ Remediation Method 1:
+ If modifying the Kubelet config file, edit the kubelet-config.json file
+ /etc/kubernetes/kubelet/kubelet-config.json and set the below parameter to
+ true
+
+ "featureGates": {
+ "RotateKubeletServerCertificate":true
+ },
+
+ Additionally, ensure that the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf does not set
+ the --rotate-kubelet-server-certificate executable argument to false because
+ this would override the Kubelet config file.
+
+ Remediation Method 2:
+ If using executable arguments, edit the kubelet service file
+ /etc/systemd/system/kubelet.service.d/10-kubelet-args.conf on each
+ worker node and add the below parameter at the end of the KUBELET_ARGS variable
+ string.
+ --rotate-kubelet-server-certificate=true
+
+ Remediation Method 3:
+ If using the api configz endpoint consider searching for the status of
+ "RotateKubeletServerCertificate": by extracting the live configuration from the
+ nodes running kubelet.
+ **See detailed step-by-step configmap procedures in Reconfigure a Node's Kubelet in a
+ Live Cluster, and then rerun the curl statement from audit process to check for kubelet
+ configuration changes
+ kubectl proxy --port=8001 &
+ export HOSTNAME_PORT=localhost:8001 (example host and port number)
+ export NODE_NAME=ip-192.168.31.226.ec2.internal (example node name from "kubectl get nodes")
+ curl -sSL "http://${HOSTNAME_PORT}/api/v1/nodes/${NODE_NAME}/proxy/configz"
+
+ For all three remediation methods:
+ Restart the kubelet service and check status. The example below is for when using
+ systemctl to manage services:
+ systemctl daemon-reload
+ systemctl restart kubelet.service
+ systemctl status kubelet -l
+ scored: true
diff --git a/cfg/eks-1.5.0/policies.yaml b/cfg/eks-1.5.0/policies.yaml
new file mode 100644
index 000000000..69c273b72
--- /dev/null
+++ b/cfg/eks-1.5.0/policies.yaml
@@ -0,0 +1,250 @@
+---
+controls:
+version: "eks-1.5.0"
+id: 4
+text: "Policies"
+type: "policies"
+groups:
+ - id: 4.1
+ text: "RBAC and Service Accounts"
+ checks:
+ - id: 4.1.1
+ text: "Ensure that the cluster-admin role is only used where required (Automated)"
+ type: "manual"
+ remediation: |
+ Identify all clusterrolebindings to the cluster-admin role. Check if they are used and if
+ they need this role or if they could use a role with fewer privileges.
+ Where possible, first bind users to a lower privileged role and then remove the
+ clusterrolebinding to the cluster-admin role :
+ kubectl delete clusterrolebinding [name]
+ scored: false
+
+ - id: 4.1.2
+ text: "Minimize access to secrets (Automated)"
+ type: "manual"
+ remediation: |
+ Where possible, remove get, list and watch access to secret objects in the cluster.
+ scored: false
+
+ - id: 4.1.3
+ text: "Minimize wildcard use in Roles and ClusterRoles (Automated)"
+ type: "manual"
+ remediation: |
+ Where possible replace any use of wildcards in clusterroles and roles with specific
+ objects or actions.
+ scored: false
+
+ - id: 4.1.4
+ text: "Minimize access to create pods (Automated)"
+ type: "manual"
+ remediation: |
+ Where possible, remove create access to pod objects in the cluster.
+ scored: false
+
+ - id: 4.1.5
+ text: "Ensure that default service accounts are not actively used. ((Automated)"
+ type: "manual"
+ remediation: |
+ Create explicit service accounts wherever a Kubernetes workload requires specific
+ access to the Kubernetes API server.
+ Modify the configuration of each default service account to include this value
+ automountServiceAccountToken: false
+
+ Automatic remediation for the default account:
+ kubectl patch serviceaccount default -p
+ $'automountServiceAccountToken: false'
+ scored: false
+
+ - id: 4.1.6
+ text: "Ensure that Service Account Tokens are only mounted where necessary (Automated)"
+ type: "manual"
+ remediation: |
+ Modify the definition of pods and service accounts which do not need to mount service
+ account tokens to disable it.
+ scored: false
+
+ - id: 4.1.7
+ text: "Avoid use of system:masters group (Automated)"
+ type: "manual"
+ remediation: |
+ Remove the system:masters group from all users in the cluster.
+ scored: false
+
+ - id: 4.1.8
+ text: "Limit use of the Bind, Impersonate and Escalate permissions in the Kubernetes cluster (Manual)"
+ type: "manual"
+ remediation: |
+ Where possible, remove the impersonate, bind and escalate rights from subjects.
+ scored: false
+
+ - id: 4.2
+ text: "Pod Security Standards"
+ checks:
+ - id: 4.2.1
+ text: "Minimize the admission of privileged containers (Automated)"
+ type: "manual"
+ remediation: |
+ Add policies to each namespace in the cluster which has user workloads to restrict the
+ admission of privileged containers.
+ To enable PSA for a namespace in your cluster, set the pod-security.kubernetes.io/enforce
+ label with the policy value you want to enforce.
+ kubectl label --overwrite ns NAMESPACE pod-security.kubernetes.io/enforce=restricted
+ The above command enforces the restricted policy for the NAMESPACE namespace.
+ You can also enable Pod Security Admission for all your namespaces. For example:
+ kubectl label --overwrite ns --all pod-security.kubernetes.io/warn=baseline
+ scored: false
+
+ - id: 4.2.2
+ text: "Minimize the admission of containers wishing to share the host process ID namespace (Automated)"
+ type: "manual"
+ remediation: |
+ Add policies to each namespace in the cluster which has user workloads to restrict the
+ admission of hostPID containers.
+ scored: false
+
+ - id: 4.2.3
+ text: "Minimize the admission of containers wishing to share the host IPC namespace (Automated)"
+ type: "manual"
+ remediation: |
+ Add policies to each namespace in the cluster which has user workloads to restrict the
+ admission of hostIPC containers.
+ scored: false
+
+ - id: 4.2.4
+ text: "Minimize the admission of containers wishing to share the host network namespace (Automated)"
+ type: "manual"
+ remediation: |
+ Add policies to each namespace in the cluster which has user workloads to restrict the
+ admission of hostNetwork containers.
+ scored: false
+
+ - id: 4.2.5
+ text: "Minimize the admission of containers with allowPrivilegeEscalation (Automated)"
+ type: "manual"
+ remediation: |
+ Add policies to each namespace in the cluster which has user workloads to restrict the
+ admission of containers with .spec.allowPrivilegeEscalation set to true.
+ scored: false
+
+ - id: 4.3
+ text: "CNI Plugin"
+ checks:
+ - id: 4.3.1
+ text: "Ensure CNI plugin supports network policies (Manual)"
+ type: "manual"
+ remediation: |
+ As with RBAC policies, network policies should adhere to the policy of least privileged
+ access. Start by creating a deny all policy that restricts all inbound and outbound traffic
+ from a namespace or create a global policy using Calico.
+ scored: false
+
+ - id: 4.3.2
+ text: "Ensure that all Namespaces have Network Policies defined (Automated)"
+ type: "manual"
+ remediation: |
+ Follow the documentation and create NetworkPolicy objects as you need them.
+ scored: false
+
+ - id: 4.4
+ text: "Secrets Management"
+ checks:
+ - id: 4.4.1
+ text: "Prefer using secrets as files over secrets as environment variables (Automated)"
+ type: "manual"
+ remediation: |
+ If possible, rewrite application code to read secrets from mounted secret files, rather than
+ from environment variables.
+ scored: false
+
+ - id: 4.4.2
+ text: "Consider external secret storage (Manual)"
+ type: "manual"
+ remediation: |
+ Refer to the secrets management options offered by your cloud provider or a third-party
+ secrets management solution.
+ scored: false
+
+ - id: 4.5
+ text: "General Policies"
+ checks:
+ - id: 4.5.1
+ text: "Create administrative boundaries between resources using namespaces (Manual)"
+ type: "manual"
+ remediation: |
+ Follow the documentation and create namespaces for objects in your deployment as you need
+ them.
+ scored: false
+
+ - id: 4.5.2
+ text: "Apply Security Context to Your Pods and Containers (Manual)"
+ type: "manual"
+ remediation: |
+ As a best practice we recommend that you scope the binding for privileged pods to
+ service accounts within a particular namespace, e.g. kube-system, and limiting access
+ to that namespace. For all other serviceaccounts/namespaces, we recommend
+ implementing a more restrictive policy such as this:
+
+ apiVersion: policy/v1beta1
+ kind: PodSecurityPolicy
+ metadata:
+ name: restricted
+ annotations:
+ seccomp.security.alpha.kubernetes.io/allowedProfileNames: 'docker/default,runtime/default'
+ apparmor.security.beta.kubernetes.io/allowedProfileNames: 'runtime/default'
+ seccomp.security.alpha.kubernetes.io/defaultProfileName: 'runtime/default'
+ apparmor.security.beta.kubernetes.io/defaultProfileName: 'runtime/default'
+ spec:
+ privileged: false
+ # Required to prevent escalations to root.
+ allowPrivilegeEscalation: false
+ # This is redundant with non-root + disallow privilege escalation,
+ # but we can provide it for defense in depth.
+ requiredDropCapabilities:
+ - ALL
+ # Allow core volume types.
+ volumes:
+ - 'configMap'
+ - 'emptyDir'
+ - 'projected'
+ - 'secret'
+ - 'downwardAPI'
+ # Assume that persistentVolumes set up by the cluster admin are safe to use.
+ - 'persistentVolumeClaim'
+ hostNetwork: false
+ hostIPC: false
+ hostPID: false
+ runAsUser:
+ # Require the container to run without root privileges.
+ rule: 'MustRunAsNonRoot'
+ seLinux:
+ # This policy assumes the nodes are using AppArmor rather than SELinux.
+ rule: 'RunAsAny'
+ supplementalGroups:
+ rule: 'MustRunAs'
+ ranges:
+ # Forbid adding the root group.
+ - min: 1
+ max: 65535
+ fsGroup:
+ rule: 'MustRunAs'
+ ranges:
+ # Forbid adding the root group.
+ - min: 1
+ max: 65535
+ readOnlyRootFilesystem: false
+
+ This policy prevents pods from running as privileged or escalating privileges. It also
+ restricts the types of volumes that can be mounted and the root supplemental groups
+ that can be added.
+ Another, albeit similar, approach is to start with policy that locks everything down and
+ incrementally add exceptions for applications that need looser restrictions such as
+ logging agents which need the ability to mount a host path.
+ scored: false
+
+ - id: 4.5.3
+ text: "The default namespace should not be used (Automated)"
+ type: "manual"
+ remediation: |
+ Ensure that namespaces are created to allow for appropriate segregation of Kubernetes
+ resources and that all new resources are created in a specific namespace.
+ scored: false
diff --git a/cmd/common_test.go b/cmd/common_test.go
index 53793a000..009462bcd 100644
--- a/cmd/common_test.go
+++ b/cmd/common_test.go
@@ -460,6 +460,12 @@ func TestValidTargets(t *testing.T) {
targets: []string{"node", "policies", "controlplane", "managedservices"},
expected: true,
},
+ {
+ name: "eks-1.5.0 valid",
+ benchmark: "eks-1.5.0",
+ targets: []string{"node", "policies", "controlplane", "managedservices"},
+ expected: true,
+ },
}
for _, c := range cases {
diff --git a/cmd/util.go b/cmd/util.go
index 275de2326..7e8e0a45a 100644
--- a/cmd/util.go
+++ b/cmd/util.go
@@ -489,7 +489,7 @@ func getPlatformBenchmarkVersion(platform Platform) string {
glog.V(3).Infof("getPlatformBenchmarkVersion platform: %s", platform)
switch platform.Name {
case "eks":
- return "eks-1.2.0"
+ return "eks-1.5.0"
case "gke":
switch platform.Version {
case "1.15", "1.16", "1.17", "1.18", "1.19":
diff --git a/cmd/util_test.go b/cmd/util_test.go
index 2c24a7a95..aea9d6462 100644
--- a/cmd/util_test.go
+++ b/cmd/util_test.go
@@ -650,7 +650,7 @@ func Test_getPlatformBenchmarkVersion(t *testing.T) {
args: args{
platform: Platform{Name: "eks"},
},
- want: "eks-1.2.0",
+ want: "eks-1.5.0",
},
{
name: "gke 1.19",
diff --git a/docs/architecture.md b/docs/architecture.md
index c65978f0a..70cdbf02d 100644
--- a/docs/architecture.md
+++ b/docs/architecture.md
@@ -5,11 +5,11 @@ The tests (or "controls") are maintained in YAML documents. There are different
## Kube-bench benchmarks
The test files for the various versions of Benchmarks can be found in directories
-with same name as the Benchmark versions under the `cfg` directory next to the kube-bench executable,
+with same name as the Benchmark versions under the `cfg` directory next to the kube-bench executable,
for example `./cfg/cis-1.5` will contain all test files for [CIS Kubernetes Benchmark v1.5.1](https://workbench.cisecurity.org/benchmarks/4892) which are:
-master.yaml, controlplane.yaml, node.yaml, etcd.yaml, policies.yaml and config.yaml
+master.yaml, controlplane.yaml, node.yaml, etcd.yaml, policies.yaml and config.yaml
-Check the contents of the benchmark directory under `cfg` to see which targets are available for that benchmark. Each file except `config.yaml` represents a target (also known as a `control` in other parts of this documentation).
+Check the contents of the benchmark directory under `cfg` to see which targets are available for that benchmark. Each file except `config.yaml` represents a target (also known as a `control` in other parts of this documentation).
The following table shows the valid targets based on the CIS Benchmark version.
@@ -29,6 +29,7 @@ The following table shows the valid targets based on the CIS Benchmark version.
| eks-1.0.1 | controlplane, node, policies, managedservices |
| eks-1.1.0 | controlplane, node, policies, managedservices |
| eks-1.2.0 | controlplane, node, policies, managedservices |
+| eks-1.5.0 | controlplane, node, policies, managedservices |
| ack-1.0 | master, controlplane, node, etcd, policies, managedservices |
| aks-1.0 | controlplane, node, policies, managedservices |
| rh-0.7 | master,node|
diff --git a/docs/platforms.md b/docs/platforms.md
index d6fbcf712..3f8cdba70 100644
--- a/docs/platforms.md
+++ b/docs/platforms.md
@@ -5,7 +5,7 @@ kube-bench supports running tests for Kubernetes.
Most of our supported benchmarks are defined in one of the following:
[CIS Kubernetes Benchmarks](https://www.cisecurity.org/benchmark/kubernetes/)
[STIG Document Library](https://public.cyber.mil/stigs/downloads)
-
+
Some defined by other hardenening guides.
| Source | Kubernetes Benchmark | kube-bench config | Kubernetes versions |
@@ -24,6 +24,7 @@ Some defined by other hardenening guides.
| CIS | [EKS 1.0.1](https://workbench.cisecurity.org/benchmarks/6041) | eks-1.0.1 | EKS |
| CIS | [EKS 1.1.0](https://workbench.cisecurity.org/benchmarks/6248) | eks-1.1.0 | EKS |
| CIS | [EKS 1.2.0](https://workbench.cisecurity.org/benchmarks/9681) | eks-1.2.0 | EKS |
+| CIS | [EKS 1.5.0](https://workbench.cisecurity.org/benchmarks/17733) | eks-1.5.0 | EKS |
| CIS | [ACK 1.0.0](https://workbench.cisecurity.org/benchmarks/6467) | ack-1.0 | ACK |
| CIS | [AKS 1.0.0](https://workbench.cisecurity.org/benchmarks/6347) | aks-1.0 | AKS |
| RHEL | RedHat OpenShift hardening guide | rh-0.7 | OCP 3.10-3.11 |
diff --git a/job-eks.yaml b/job-eks.yaml
index beaf39194..068be2914 100644
--- a/job-eks.yaml
+++ b/job-eks.yaml
@@ -18,9 +18,9 @@ spec:
"kube-bench",
"run",
"--targets",
- "node",
+ "node,policies,managedservices,controlplane",
"--benchmark",
- "eks-1.2.0",
+ "eks-1.5.0",
]
volumeMounts:
- name: var-lib-kubelet