Following features are supported by IAM Manager
IAM Roles Management
IAM Role for Service Accounts (IRSA)
AWS Service-Linked Roles
Default Trust Policy for All Roles
Maximum Number of Roles per Namespace
Attaching Managed IAM Policies for All Roles
Multiple Trust policies
Note: Please Note that Permission Boundary will be automatically added for each role and you can configure permission boundary name using config map variable.
iam.managed.permission.boundary.policy: "permission-boundary-policy-name"
Simplest way to create an IAM Role is to provide the PolicyDocument and AssumeRolePolicyDocument. You might have already guessed by now, to keep it simple, iam-manager tried to keep AWS CFN(Cloud Formation) format for IamRole Spec.
Example:
apiVersion: iammanager.keikoproj.io/v1alpha1
kind: Iamrole
metadata:
name: iam-manager-iamrole
spec:
# Add fields here
PolicyDocument:
Statement:
-
Effect: "Allow"
Action:
- "s3:Get*"
Resource:
- "arn:aws:s3:::intu-oim*"
Sid: "AllowS3Access"
AssumeRolePolicyDocument:
Version: "2012-10-17"
Statement:
-
Effect: "Allow"
Action: "sts:AssumeRole"
Principal:
AWS:
- "arn:aws:iam::XXXXXXXXXXX:role/20190504-k8s-kiam-role"
IAM Manager supports IRSA(IAM Role for service accounts) starting from IAM Manager 0.0.4 version.
Pre-requisites
For EKS Clusters, You can provide following params in config map and IAM Manager will automatically sets up OIDC IDP in AWS IAM if it doesn't exist
iam.irsa.enabled : "true"
k8s.cluster.name : "EKS Cluster Name"
OR
You can provide OIDC URL (OIDC IDP must be created in AWS IAM)
k8s.cluster.oidc.issuer.url: "<OIDC URL from K8s cluster>"
Note: To setup OIDC IDP in AWS IAM, you must provide required permissions to iam-manager role. Here is the minimum permissions needed for this setup
- Effect: "Allow"
Action:
- "iam:CreateOpenIDConnectProvider"
- "eks:DescribeCluster"
Resource: "*"
Sid: "IRSANeededPermissions"
Request:
Once pre-requisites are completed, Attach following annotation to the IAM Role CR and IAM Manager will automatically attaches required trust policy to IAM Role.
iam.amazonaws.com/irsa-service-account: "service_account_name"
IAM Manager will create Service Account if its doesn't exist or update the service account with required annotations.
Note: For kops clusters, you must install AWS amazon-eks-pod-identity-webhook
For more info: AWS Blog
Example:
apiVersion: iammanager.keikoproj.io/v1alpha1
kind: Iamrole
metadata:
name: iam-manager-iamrole-irsa
annotations:
iam.amazonaws.com/irsa-service-account: aws-sa
spec:
# Add fields here
PolicyDocument:
Statement:
-
Effect: "Allow"
Action:
- "s3:Get*"
Resource:
- "arn:aws:s3:::intu-oim*"
Sid: "AllowS3Access"
This should automatically create service account if it doesn't exist
mtvl15367e28a:iam-manager nmogulla$ k get sa aws-sa -o yaml
apiVersion: v1
kind: ServiceAccount
metadata:
annotations:
eks.amazonaws.com/role-arn: arn:aws:iam::XXXXXXXXX:role/k8s-iam-manager-iamrole-irsa
creationTimestamp: "2020-06-01T06:19:41Z"
name: aws-sa
namespace: a-test-usw2-test-123
resourceVersion: "10670180"
selfLink: /api/v1/namespaces/a-test-usw2-test-123/serviceaccounts/aws-sa
uid: e1307b2a-a3cf-11ea-8b0e-0a6a4f8e42d4
secrets:
- name: aws-sa-token-bcflm
IAM Manager can also be used to create service linked roles for example "eks.amazonaws.com" to allow EKS to perform the required activities to run cluster.
apiVersion: iammanager.keikoproj.io/v1alpha1
kind: Iamrole
metadata:
name: iam-manager-iamrole-eks-role
spec:
# Add fields here
PolicyDocument:
Statement:
-
Effect: "Allow"
Action:
- "s3:Get*"
Resource:
- "arn:aws:s3:::intu-oim*"
Sid: "AllowS3Access"
AssumeRolePolicyDocument:
Version: "2012-10-17"
Statement:
-
Effect: "Allow"
Action: "sts:AssumeRole"
Principal:
Service: "eks.amazonaws.com"
There might be a situations where as an administrator you might want to control the trust policy. For example, in KIAM use case every role must be trusted by master server role where kiam server is deployed. That can be configured in IAM Manager using config map variable
iam.default.trust.policy : "Assume Role Policy Json as a string"
The above config map variable does also accept Go Template to replace following values during runtime
- AccountID
- ClusterName
- NamespaceName
- Region
Here is a sample value from config map
iam.default.trust.policy: '{"Version": "2012-10-17", "Statement": [{"Effect": "Allow","Principal": {"Federated": "arn:aws:iam::{{.AccountID}}:oidc-provider/OIDC_PROVIDER"},"Action": "sts:AssumeRoleWithWebIdentity","Condition": {"StringEquals": {"OIDC_PROVIDER:sub": "system:serviceaccount:{{.NamespaceName}}:SERVICE_ACCOUNT_NAME"}}}, {"Effect": "Allow","Principal": {"AWS": ["arn:aws:iam::{{.AccountID}}:role/trust_role"]},"Action": "sts:AssumeRole"}]}'
AccountID and NamespaceName will be replaced at run time.
Example:
apiVersion: iammanager.keikoproj.io/v1alpha1
kind: Iamrole
metadata:
name: iam-manager-iamrole-default-trust
spec:
# Add fields here
PolicyDocument:
Statement:
-
Effect: "Allow"
Action:
- "s3:Get*"
Resource:
- "arn:aws:s3:::intu-oim*"
Sid: "AllowS3Access"
This should automatically add the default trust policy from config map.
By default, maximum number of roles per namespace is 1. You can configure the max roles per namespace using config map variable
iam.role.max.limit.per.namespace : "10"
You can attach any managed iam policies to all the roles created by IAM Manager by configuring config map variable
iam.managed.policies: "shared.security-policy-20200504-k8s"
This might be useful in use cases where security team wants to attach a managed policies for all the roles.
You can always use the combination of trust policies. For example: An IAM Role might need to access it from application as well as AwS service.
Example:
apiVersion: iammanager.keikoproj.io/v1alpha1
kind: Iamrole
metadata:
name: iam-manager-iamrole-multiple-trust
annotations:
iam.amazonaws.com/irsa-service-account: aws-sa-multiple
spec:
# Add fields here
PolicyDocument:
Statement:
-
Effect: "Allow"
Action:
- "s3:Get*"
Resource:
- "arn:aws:s3:::intu-oim*"
Sid: "AllowS3Access"
AssumeRolePolicyDocument:
Version: "2012-10-17"
Statement:
-
Effect: "Allow"
Action: "sts:AssumeRole"
Principal:
Service: "eks.amazonaws.com"
This will have both AssumeRoleWithWebIdentity to assume role from a pod and also sts:AssumeRole for "eks.amazonaws.com" to allow access.