The Expel Amazon Elastic Kubernetes Service (EKS) consumes audit logs from the AWS platform through Kinesis. This visibility allows Workbench to identify activity of interest in EKS, investigate, and notify organizations if action is recommended.
Quick Start
Setup includes the following steps (select any step for detailed instructions):
-
-
Enable the Expel Security Benchmark Report (No longer supported as of June 4, 2024)
Step 1: Configure MDR
1. Enable control plane logging for each EKS cluster
At a minimum, Workbench requires logging of the audit events for each cluster. This provides visibility into activity affecting resources in the cluster: the “who, what, and when” we need to detect and take action.
Update logging configuration for each cluster in each region with the AWS CLI as in the following example:
CLUSTER_REGION=<your-cluster-region> ; \ CLUSTER_NAME=<your-cluster-name> ; \ aws eks update-cluster-config \ --region ${CLUSTER_REGION} \ --name ${CLUSTER_NAME} \ --logging '{"clusterLogging":[{"types":["audit"],"enabled":true}]}'
For more information on configuring control plane logging, see the Amazon reference guide.
After control plane logging is enabled, EKS begins sending logs to CloudWatch.
2. Create a Kinesis Stream
Logs must be routed from CloudWatch to a Kinesis data stream so Workbench can consume the logs in real time.
Create a stream with the AWS CLI tool and note the following:
- The value of K_STREAM_NAME must be different for each separate region.
- We recommend creating an “ON_DEMAND” stream to allow the stream to adjust capacity based on demand.
CLUSTER_REGION=<your-cluster-region> ; \
K_STREAM_NAME=expel-aws-eks-kinesis-data-stream ; \
aws kinesis create-stream \
--region ${CLUSTER_REGION} \
--stream-name ${K_STREAM_NAME} \
--stream-mode-details '{"StreamMode": "ON_DEMAND"}'
- Note the Kinesis Stream ARN that the code CLI generates for use in later steps. The ARN is different for reach region.
CLUSTER_REGION=<your-cluster-region> ; \ K_STREAM_NAME=expel-aws-eks-kinesis-data-stream ; \ aws kinesis describe-stream-summary --stream-name=${K_STREAM_NAME} --region=${CLUSTER_REGION}
3. Create an IAM role for CloudWatch log delivery
An IAM role is required to allow CloudWatch to deliver logs to your Kinesis stream.
-
Create a trust policy document that allows the CloudWatch service to assume the role.
The following sample code writes a policy JSON definition to
expel_cw_assume_role_trust
in the current directory. For convenience, the system creates a JSON document named
_policy_doc.jsonexpel_cw_assume_role_trust_policy_doc.json
on the local machine.AWS_REGION=<your-aws-region> ; \ AWS_ACCOUNT_ID=<your-aws-account-id> ; \ cat > expel_cw_assume_role_trust_policy_doc.json << EOF { "Statement": { "Effect": "Allow", "Principal": { "Service": "logs.${AWS_REGION}.amazonaws.com" }, "Action": "sts:AssumeRole", "Condition": { "StringLike": { "aws:SourceArn": "arn:aws:logs:${AWS_REGION}:${AWS_ACCOUNT_ID}:*" } } } } EOF
-
Create the IAM role with the
expel_cw_assume_role_trust_policy_doc.json
document:EXPEL_CW_ASSUME_ROLE_NAME=expel-aws-eks-cloudwatch-assume-role ; \ aws iam create-role \ --role-name ${EXPEL_CW_ASSUME_ROLE_NAME} \ --assume-role-policy-document file://expel_cw_assume_role_trust_policy_doc.json
4. Create an IAM policy document for the producer role
Create an IAM policy document that grants the following role permissions to put records into your Kinesis stream:
K_STREAM_ARN=<kinesis stream ARN saved in earlier step> ; \
cat > expel_iam_policy_doc.json << EOF
{
"Statement": [
{
"Effect": "Allow",
"Action": "kinesis:PutRecord",
"Resource": "${K_STREAM_ARN}"
}
]
}
EOF
5. Apply policy to producer role
Apply the IAM policy to the CloudWatch role:
EXPEL_CLOUDWATCH_ROLE_NAME=expel-aws-eks-cloudwatch-assume-role; \ EXPEL_IAM_POLICY_NAME=expel-aws-eks-producer-policy; \ aws iam put-role-policy \ --role-name "${EXPEL_CLOUDWATCH_ROLE_NAME}" \ --policy-name "${EXPEL_IAM_POLICY_NAME}" \ --policy-document file://expel_iam_policy_doc.json
6. Route logs from CloudWatch to Kinesis
Logs must be routed to Kinesis with a CloudWatch log group subscription filter.
Create the filter with the AWS CLI:
K_STREAM_ARN=<kinesis stream ARN saved in earlier step> ; \ EKS_LOG_GROUP_NAME=<your choice of log group name> ; \ CLUSTER_AWS_REGION=<your-aws-region> ; \ EXPEL_CLOUDWATCH_ROLE_NAME=expel-aws-eks-cloudwatch-assume-role ; \ aws logs put-subscription-filter \ --region ${CLUSTER_AWS_REGION} \ --log-group-name ${EKS_LOG_GROUP_NAME} \ --filter-name "AllEKSLogs" \ --filter-pattern "" \ --destination-arn ${K_STREAM_ARN} \ --role-arn ${EXPEL_CLOUDWATCH_ROLE_NAME}
7. Create an IAM Role for Workbench
To authenticate to your AWS account and retrieve logs, Workbench requires an IAM role. If you already have a Workbench IAM role for another Workbench integration (like CloudWatch or Amazon GuardDuty), this existing role can be reused and you can go to Grant Workbench IAM role permissions.
- Create a trust policy document that allows Workbench to assume the role. For convenience, the system creates a JSON document named
expel_wb_iam_role_policy_doc.json
on the local machine.
Note: You can find your GUID by navigating to Organization Settings > My Organization(s) in Workbench.
ORG_GUID=<Your workbench org_guid(formerly customer id)> ; \ cat > expel_wb_iam_role_policy_doc.json << EOF { "Statement": [ { "Effect": "Allow", "Principal": { "AWS": "arn:aws:iam::012205512454:user/ExpelCloudService" }, "Action": "sts:AssumeRole", "Condition": { "StringEquals": { "sts:ExternalId": "${ORG_GUID}" } } } ] } EOF
-
Create the IAM role with the
expel_wb_iam_role_policy_doc.json
document:EXPEL_ASSUME_ROLE_NAME=ExpelServiceAssumeRole ; \ aws iam create-role \ --role-name ${EXPEL_ASSUME_ROLE_NAME} \ --assume-role-policy-document file://expel_wb_iam_role_policy_doc.json
8. Grant Workbench IAM role permissions
Workbench requires IAM permissions to retrieve logs from Kinesis and to investigate the activity of interest affecting your EKS clusters.
-
Create an IAM policy document that grants Workbench the required permissions for Kinesis and EKS:
K_STREAM_ARN=< kinesis stream ARN saved in earlier step > ; \ cat > expel_k8s_ro_policy_doc.json << EOF { "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Action": [ "kinesis:DescribeLimits", "kinesis:DescribeStream", "kinesis:DescribeStreamSummary", "kinesis:GetRecords", "kinesis:GetShardIterator", "kinesis:ListShards" ], "Resource": "${K_STREAM_ARN}" }, { "Effect": "Allow", "Action": [ "eks:AccessKubernetesApi", "eks:DescribeCluster", "eks:DescribeNodegroup", "eks:ListClusters", "eks:ListNodegroups", "eks:ListUpdates", "sts:GetCallerIdentity", "ec2:DescribeRegions", "autoscaling:DescribeAutoScalingGroups" ], "Resource": "*" } ] } EOF
-
Apply the policy to the Expel IAM role:
EXPEL_ASSUME_ROLE_NAME=ExpelServiceAssumeRole ; \ EXPEL_EKS_CONSUMER_POLICY_NAME=expel-aws-eks-eks-consumer-policy ; \ aws iam put-role-policy \ --role-name ${EXPEL_ASSUME_ROLE_NAME} \ --policy-name ${EXPEL_EKS_CONSUMER_POLICY_NAME} \ --policy-document file://expel_k8s_ro_policy_doc.json
Step 2: Configure Workbench
In this step we will add Amazon EKS as a security device and then enable network access by allowing specific IP addresses in AWS.
- Log in to Workbench.
- Navigate to Organization Settings > Security Devices.
- Select Add Security Device.
-
Scroll down or search "Kubernetes (EKS)" and select Kubernetes Service (EKS).
-
Complete the Add Security Device form:
-
Name - enter a name that might help you more easily identify this integration. This name will display in Workbench under the Name column, and is a text string that you can filter on.
-
Location - enter the location of your integration, for example “cloud” or “AWS cloud” or “on prem;” this is also a text string that you can filter on, so we recommend being consistent with location naming across your Expel integrations.
-
Role ARN - enter the ARN for the cluster role you created in the previous step, Create an IAM role for Workbench.
-
Role Session Name - enter a Role Session Name; we recommend
ExpelSession
. -
Region - enter the AWS region containing the Kubernetes cluster.
-
Kinesis Stream Name - enter the log stream name you created in the previous step, Create a Kinesis Stream.
-
- Select Save.
- Next, enable network access by adding the IP addresses listed in this article to your allow list in AWS.
Step 3: Enable the Expel Security Benchmark Report
Note: As of June 4, 2024, the Expel Security Benchmark Report is no longer supported.
In this step you will grant Workbench read-only cluster permissions and map the Expel Role ARN to a cluster, providing the permissions needed to generate the report. See your Engagement Manager for assistance with accessing the report.
Expel recommends providing limited read-only access to your EKS clusters to enable deep investigation of interesting activity. These permissions can be fine-tuned based on your needs. Workbench can deliver service without this access, but it significantly limits our ability to thoroughly investigate activity in your clusters.
-
Create an Expel
ClusterRole
manifest by saving the following code asexpel-cluster-role.yml
:apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRole metadata: name: expel-reader-clusterrole rules: - apiGroups: - "" - admissionregistration.k8s.io - apps - networking.k8s.io - rbac.authorization.k8s.io resources: - apiservices - clusterrolebindings - clusterroles - cronjobs - daemonsets - deployments - events - flowschemas - horizontalpodautoscalers - ingressclasses - ingresses - jobs - localsubjectaccessreviews - mutatingwebhookconfigurations - namespaces - networkpolicies - nodes - persistentvolumes - poddisruptionbudgets - pods - podsecuritypolicies - podtemplates - replicasets - rolebindings - roles - selfsubjectaccessreviews - selfsubjectrulesreviews - serviceaccounts - services - statefulsets - subjectaccessreviews - tokenreviews - validatingwebhookconfigurations - volumeattachments verbs: - get - list
-
Use
kubectl
to apply the role:kubectl apply -f expel-cluster-role.yml
-
Create a cluster role binding by saving the following code as
expel-cluster-role-binding.yml
:kind: ClusterRoleBinding apiVersion: rbac.authorization.k8s.io/v1 metadata: name: expel-reader-clusterrolebinding subjects: - kind: User name: expel-user roleRef: kind: ClusterRole name: expel-reader-clusterrole apiGroup: rbac.authorization.k8s.io
-
Use
kubectl
to apply the role binding:kubectl apply -f expel-cluster-role-binding.yml
-
Map the Expel Role ARN to a cluster user using
eksctl
:CLUSTER_REGION=<your-cluster-region> ; \ CLUSTER_NAME=<your-cluster-name> ; \ EXPEL_SERVICE_ASSUME_ROLE_ARN=<expel-service-assume-role-arn> ; \ eksctl create iamidentitymapping \ --cluster ${CLUSTER_NAME} \ --region ${CLUSTER_REGION} \ --arn ${EXPEL_SERVICE_ASSUME_ROLE_ARN} \ --username expel-user
-
Confirm the mapping with the following code:
CLUSTER_REGION=<your-cluster-region> ; \ CLUSTER_NAME=<your-cluster-name> ; \ eksctl get iamidentitymapping \ --cluster ${CLUSTER_NAME} \ --region ${CLUSTER_REGION}
Reference
This integration requires configuration of certain resources in order to function. Refer to the following list for details:
Resource |
Purpose |
---|---|
Kinesis Stream |
MDR |
Kinesis Stream Filter |
MDR |
IAM Role |
MDR |
IAM Policy “Puts” (binds) |
Inventory/metrics |
EKS ClusterRole |
Security benchmarking |
EKS ClusterRoleBinding |
Security benchmarking |
EKS IAM Identity Mapping |
Security benchmarking |
Network Allow Rule |
All Workbench investigation access |