How to create a Delegated role for assumption by one or more Primary roles
Status
OUT OF DATE
Problem
You want to configure a new set of fine-grained IAM roles (e.g. for an intern) using iam-primary-roles and iam-delegated-roles and not sure how to go about it.
Solution
Plan the privileges of the new role
A Delegated role grants a collection of privileges. Users associated with a Primary role allowed to assume the Delegated role can access these privileges.
Implement the IAM role
Not to be overlooked is the importance of choosing a good name. It should be informative, and for the sake of
automation, consist solely of letters (no symbols). For this example, we will create the role restricted
. Because we want restricted
to be able to do more than an observer
but less than a poweruser
, we will need to create a new Delegated role. If we are creating this permission set for use with a Primary role that is not yet defined, we will need to create that too, but for the sake of this example, we will assume we have already created a Primary IAM role, intern
, that will be granted permission to assume this new Delegated role.
delegated_roles_config:
admin:
...
restricted:
# role_policy_arns is the list of IAM Policies to attach to this
# role. You can give the full ARN of an existing policy, or you
# can give the name of a custom policy you define elsewhere
# in the project. Here, we use a custom policy name we have already
# defined, to give the restricted role permission to assume roles
# in other accounts.
role_policy_arns: ["delegated_assume_role"]
role_description: "Restricted permissions for AWS and kubernetes"
# We would set this to True were this role to also be used as a Primary role,
# but nobody should be logging in directly as `restricted` so here it is false.
sso_login_enabled: false
# trusted_primary_roles is the list of other roles that can assume this one.
trusted_primary_roles: ["admin", "ops", "poweruser", "intern"]
If we wanted to create a new policy for the new role, we would follow the example of delegated_assume_role
. Create a new file with the role name in components/terraform/iam-primary-roles/
, create the policy in the file, and add the policy to the custom_policy_map
in components/terraform/iam-primary-roles/main.tf
, but we do not need to allow our interns anything special in the identity
account.
Configure or disable the Delegated role in the other accounts
By default, delegated roles are created in every account. This may or may not be desired behavior. If there are any
accounts in which your role is irrelevant or should not be used to grant privileges, disable it by adding it to the
exclude_roles
list in stacks/gbl-$stage.yaml
.
Configure IAM policy in other accounts
By default, Delegated roles will get the same-named policies in other accounts as they got in identity
. In practice, this is rarely correct, but if it is correct in this case, then you need not take any of the following steps. Otherwise, you need to collect the policies to apply to the role and list them in the role_policy_arns
map (which we will get to after collecting the entries).
Identify one or more AWS managed IAM policies to apply
If you want, you can use AWS managed policies (although there is a case to be made against it, which
includes that they are not easily found or comprehensively documented). Their ARNs can be used directly in the role_policy_arns
map.
Create a custom policy for the role
Just as with the iam-primary-roles
project, we can create a custom policy for the new role. We will create a
restricted policy that
lets them modify resources that are tagged
restrictedModify: allowed
. We will combine this with the AWS managed ViewOnly
policy so they can look at things.
(The AWS ReadOnly
policy is more permissive and would let them read secrets, so we do not want to use that.)
-
Create a new file with the policy name under
iam-delegated-roles
calledrestricted-policy
. -
Define the policy, something like (this has not been vetted/tested)
data "aws_iam_policy_document" "restricted" {
statement {
sid = "PowerWhenTaggedForRestricted"
effect = "Allow"
actions = [
"cloudwatch:*",
"ec2:*",
"ecr:*",
"eks:*",
# Lambda does not support tag based access "lambda:*",
"rds:*",
"s3:*",
]
resources = ["*"]
condition {
test = "StringEquals"
values = ["allowed"]
variable = "aws:ResourceTag/restrictedModify"
}
}
}
- Create the policy
resource "aws_iam_policy" "restricted" {
name = format("%s-restricted", module.label.id)
description = "Allow users to modify specially tagged resources"
policy = data.aws_iam_policy_document.restricted.json
}
- Add the policy to the
custom_policy_map
inmain.tf
custom_policy_map = { "restricted" = aws_iam_policy.restricted.arn }
- Override the default policy assignment for the new role
In iam-delegated-roles/default.auto.tfvars
the role_policy_arns
map is just that: a map whose keys are role names
and whose values are lists of policy ARNs (or names in the custom_policy_map
) to assign to that role, overriding the
assignments made in iam-primary-roles
. Add an entry for the new role.
role_policy_arns = {
# Put the ARN first, it will make the Terraform output cleaner
restricted = ["arn:aws:iam::aws:policy/job-function/ViewOnlyAccess", "restricted"]
...
}
Configure custom access to Delegated role
We need to allow one or more Primary IAM roles to assume this Delegated IAM role in one or more stages of our
infrastructure for it to have any effect. As noted above, we will assume here that we have created the intern
Primary
role. Further, let's assume we wish to allow our new intern access to the restricted
role's permissions only in our
sbx
and dev
stages. We can accomplish this by adding a line for our new role to trusted_primary_role_overrides
for
sbx
and dev
as shown below.
In stacks/gbl-sbx.yaml
add
projects:
terraform:
iam-delegated-roles:
vars:
trusted_primary_role_overrides:
restricted: ["dev", "intern"]
In stacks/gbl-dev.yaml
add
projects:
terraform:
iam-delegated-roles:
vars:
trusted_primary_role_overrides:
restricted: ["intern"]
Extra Credit: disable roles entirely in root
and audit
Because we have assigned this role the delegated_assume_role
policy, it cannot get into root
or audit
, but to be extra careful, we can prevent this role from being created in those accounts in the first place.
In stacks/gbl-root.yaml
and in stacks/gbl-audit.yaml
add "restricted" to the list of additionally excluded roles.
Run Terraform to actually provision the accounts and policies
These steps must be run as admin in the root
account. We must run plan and apply once for each account that uses
delegated roles. Perform as below for each stage, replacing $stage with the literal stage name.
assume-role eg-gbl-root-admin bash -l
# Perform the following for each stage
atmos terraform plan iam-delegated-roles --stack=gbl-$stage
atmos terraform apply iam-delegated-roles --stack=gbl-$stage
# leave the root admin subshell
exit
Implement the Kubernetes role
EKS maps IAM roles to Kubernetes groups. Selecting appropriate permissions for the restricted
group in Kubernetes is beyond the scope of this article. We will go through the steps without specifics.
Assign Kubernetes usernames and groups to the Delegated roles in each account
The admin, ops, poweruser, and observer primary IAM roles and the same delegated roles plus the terraform and helm roles required for our automation to function are configured in stacks/eks/default.auto.tfvars.
Additional roles can be configured in the same file for universal permissions, or for each account in the relevant account config file. In this example, we need to add a new Kubernetes user and group and assign the new Delegated role to it. We will use the sbx
account and the uw2
environment, and add the following to stacks/uw2-sbx.yaml
.
Edit stacks/uw2-dev.yaml
and add an entry for the restricted
role to the
projects:terraform:eks:vars:delegated_iam_roles
list:
projects:
...
terraform:
...
eks:
vars:
delegated_iam_roles:
role: "restricted"
groups: ["idp:restricted"]
Do this for every account. Be sure to use the correct stage name and account number in the role ARN and username. We use
the idp:
prefix on the Kubernetes groups we create in order to avoid conflict with any pre-defined groups or roles.
"IDP" is for Identity Provider and was chosen because the Kubernetes group membership is assigned based on an identity
provided by an outside identity provider.
If you want to create a mapping from the Primary role in identity
to a Kubernetes role, you can do that by adding an
entry in the primary_additional_iam_roles
map in the defaults.auto.tfvars
file, but this is not recommended. That
feature is provided as a shortcut for operations personnel, is not required for anything, and should not be extended to
other roles because it can open up unintended privileges.
Note that we assigned the role to the group "idp:restricted". That group does not exist yet; we must create it. However, you can run the Terraform to apply the changes now if you prefer, or you can wait until after the groups are created.
Create Kubernetes roles and bindings in each account
Kubernetes roles and role mappings are defined in /projects/helmfiles/idp-roles/
The helmfile.yaml
uses the
Kubernetes "raw chart" to directly specify Kubernetes resources. You can follow the examples in the existing
helmfile.yaml
for creating ClusterRoles and ClusterRoleBindings, and create new ones as needed. At a minimum, you will
need to create a new ClusterRoleBinding to bind the Group "idp:restricted" to a ClusterRole. The examples bind a User as
well, but that has no practical effect.
Run Terraform and Helm in each account
Be sure to run both the Terraform under eks
and the Helm under helmfiles/idp-roles
for each account with an EKS
cluster. These can be run the normal way, as eg-gbl-identity-ops
.
Update the company AWS config
file with the new Delegated role
If you created a new Delegated role, then people who want to use that role need to add it to their AWS config
files,
normally stored as ~/.aws/config
on individual's computers. We keep the shared configuration in the repo as
/aws-config
.
To update the file,
-
Add "restricted" to the list of delegated_roles in
/rootfs/usr/local/bin/aws-config-gen
-
Run the modified
aws-config-gen
and save the output in/aws-config
Let people know to update their configuration files.