The script uses the standard AWS credentials file ~/.aws/credentials
and it's own configuration file ~/.assumerole
to assume a role on an account (defined in ~/.assumerole
), using the AWS profile in ~/.aws/credentials
referred
to by the aws_profile
property in ~/.assumerole
.
Credentials are cached in ~/.assumerole.d/cache
. When assuming a role using assumerole
,
and the cache exists, the cache is used and the validity of the cache is checked.
If the cache is valid, it is used. If the cache is not valid, the cache entry is cleared, and the user is asked to enter the MFA code.
An example to illustrate this.
The AWS credentials file ~/.aws/credentials
contains:
[acme-bastion]
aws_access_key_id = AKIAXXXXXXXXXXXXXXXX
aws_secret_access_key = bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb
region = eu-central-1
The ~/.assumerole
file contains this:
{
"codeartifact": {
"id": "codeartifact",
"username": "aws",
"domain_owner": "678053966837",
"domain": "ixorartifacts"
},
"assume_roles": {
"acme-sandbox-read": {
"aws_profile": "acme-bastion",
"aws_account": "123456789012",
"aws_mfa_arn": "arn:aws:iam::210987654321:mfa/rtytgat",
"aws_role": "read",
"environment": [
{
"name": "MYVAR1",
"value": "MYVALUE1"
},
{
"name": "MYVAR2",
"value": "MYVALUE2"
}
]
},
...
}
And running the command:
$ /usr/local/bin/assumerole
Select from these available accounts:
...
acme-sandbox-read
acme-otheraccount-read
...
Account: acme-sandbox-read
MFA token: 123456
export AWS_ACCESS_KEY_ID=XXXXXXXXXXXXXXXXXXXX
export AWS_SECRET_ACCESS_KEY=aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
export AWS_SESSION_TOKEN=aaaaaaa/a/very/long/string/aaaaaaaaaa
Does the following:
- use the AWS profile
acme-bastion
- if the user has permissions to assume the role
read
on account123456789012
... - ... temporary credentials are requested for that account
- extract the relevant properties from the returned JSON file
- sets the environment variables
- and also prints the
export
commands tostdout
for the user to copy/paste - sets the environment variables if any are defined in the configuration for that profile
- loads the SSH key, if one is defined for the selected profile
- if the file
~/.m2/settings.xml
exists,assumerole
will add a<server>
entry for AWS CodeArtifact - start a new shell
In the new shell, the permissions linked to the role that is being assumed is available for
the aws
CLI.
Exiting the shell will unset the credential environment variables.
- The script uses
jq
to parse the JSON configuration file~/.assumerole
- When you want to use the Maven/AWS CodeArtifact integration, the
xml
command from XMLStarlet is required as well - The AWS CLI, see here on how to install this
$ assumerole [profile [mfatoken]]
$ assumerole
Select from these available accounts:
cust1-prod-read cust1-staging-power
Account: cust1-prod-read
MFA token: 331956
export AWS_ACCESS_KEY_ID=ASIAXXXXXXXXXXXXXXXX
export AWS_SECRET_ACCESS_KEY=7O801XXXXXXXXXXXXXXXXXahoLwdz8KLtRCc1Bvh
export AWS_SESSION_TOKEN=FQoDYXdXXXXXX...XXXX//
During its execution, assumerole
starts a new shell (${SHELL}
). The environment
variables that build the credentials are set before starting the shell.
The user has to enter 2 values (either on the commandline or interactively):
- The account: this is a account string defined in the configuration file
~/.assumerole
- MFA token: the current value of the MFA token used as multi factor device
If these conditions are met:
- the file
~/.m2/settings.xml
exists codeartifact.domain
is in~/.assumerole
codeartifact.domain_owner
is in~/.assumerole
- the environment variable
ASSUMEROLE_SKIP_CODEARTIFACT
is not set - a CodeArtifact authentication token can be successfully obtained
Then:
- the
password
property for the ID will be updated if a server entry for that ID already exists - a net
<server>
node will be added to<servers>
in~/.m2/settings.xml
if that entry did not already exist
An example AWS CodeArtifact configuration for ~/.assumerole
looks like this:
{
"codeartifact": {
"id": "codeartifact",
"username": "aws",
"domain_owner": "123456789012",
"domain": "my_artifacts"
},
"assume_roles": {
}
}
In order to assume a role, it is required to fill in an MFA token
, this token can be given in the command itself or entered when prompted.
If you have the 1Password CLI installed, it is possible to get the OTP
from the entry (env var placed in ~/.zshrc
)
- 1Password account set as
ONEPASS_ACCOUNT
in~/.zshrc
(looks like[account].1password.eu
) - 1Password entry set as
ONEPASS_ENTRY
in~/.zshrc
(Entry in 1Password that contains theBastion OTP
)
# 1Password Bastion Entry
export ONEPASS_ENTRY="[NameOfTheEntry]"
export ONEPASS_ACCOUNT="[yourAccount].1password.eu"
- 1Password CLI v2 installed and configured (install here)
$ assumerole [profile]
1Password CLI will prompt it's authentication in order to fetch the OTP token.
Bash commandline completion was introduced in version v0.1.1 and can be used to complete account names.
For command completion to work, add these lines to your ~/.bash_profile
file:
# Bash completion for assumerole
_assumerole_accounts=$(assumerole accountlist_text)
complete -W "${_assumerole_accounts}" 'assumerole'
You need to star a new shell (or source the ~/.bash_profile
file) before command completion will be avialable.
Copy the file autocompletion/_assumerole.zsh
in this repo to
~/.oh-my-zsh/completions/_assumerole
.
And these lines to your ~/.zshrc
file if they are not present:
### Commandline completion
autoload -U compinit
compinit
Set this envvar to the number of seconds you wish the temporary credentials to be valid.
Mind though, the assumed role should be configured to allow at least the requested session
duration. If the role's session duration is less than what you specify here, the
assumerole
operation will fail.
To increase the role's maximume CLI/API session duration, use the AWS Console or this CLI command:
aws iam update-role -–role-name name-of-the-role -–max-session-duration 14400
When this environment variable is set and the assumerole
command is executed, it will not start
a new shell, but run the command in the environmnet variable instead, returning to the current
shell after the command finishes.
This can be used to loop over different accounts and have the same command run against those accounts.
When required, custom environment variables can be defined in the configuration file. When that profile is activated, the environment variables are set before the new shell is spawned.
To define custom environment variables, add a list environment
to the profile,
for example:
{
"assume_roles": {
"acme-sandbox-read": {
...
"environment": [
{
"name": "MYVAR1",
"value": "MYVALUE1"
},
{
"name": "MYVAR2",
"value": "MYVALUE2"
}
]
},
...
}
This is an example configuration file:
{
"assume_roles": {
"logical_name_for_stsAssumeRole": {
"aws_profile": "aws_profile_to_use",
"aws_account": "account_number_of_account_where_to_assume_the_role",
"aws_mfa_arn": "arn_of_user_mfa_devoce",
"aws_role": "name_of_role_to_assume",
"sshkey": "/full/path/to/private/key/for/this/profile"
},
"mycompany-prod-read": {
"aws_profile": "mycompany-bastion",
"aws_account": "123456789012",
"aws_mfa_arn": "arn:aws:iam::210987654321:mfa/myuser",
"aws_role": "read",
"environment": [
{
"name": "MYVAR1",
"value": "MYVALUE1"
},
{
"name": "MYVAR2",
"value": "MYVALUE2"
}
]
},
...
}
}
This is a description for the role that will be assumed. It typically contains the name of the account where the role is going to be assumed, and the name of the role to be assumed. This is a string that can be chosen freely.
The assumerole
script will set the environment variable AWS_PROFILE
to this value. That
means that the AWS CLI configuration file ~/.aws/credentials
should contain a named profile
that matches this string.
The numeric account ID of the AWS account where a role is to be assumed.
The name of the role to assume on the remote account.
The ARN of the MFA device of the user on the account that will assume the role. This can be
a bastion account, where only user are defined, and groups that allow sts::AssumeRole
permissions for a selection of accounts.
A list of dicts with name
and value
keys, used to set custom environment in the
spawned shell.
Private key to load when activating the profile.
### Prompt
PROMPT_COMMAND="[[ -n \${AWS_ACCOUNT} ]] && export AWS_PROMPT=\" AWS:\${AWS_ACCOUNT}\" || export AWS_PROMPT=''"
PS1="\s-\v\$AWS_PROMPT $ "
### POWERLEVEL9K config
POWERLEVEL9K_LEFT_PROMPT_ELEMENTS=( dir vcs custom_aws_credentials)
POWERLEVEL9K_CUSTOM_AWS_CREDENTIALS="[[ -n \${AWS_ACCOUNT} ]] && echo AWS \${AWS_ACCOUNT}"
The aws_profile
property in the assumerole
configuration file shouls be a valid profile in
~/.aws/credentials
with a valid and active access key.
If the value of the aws_profile
property is, for example, org-bastion
, the following command should
work and should not fail:
$ AWS_PROFILE=org-bastion aws sts get-caller-identity
{
"UserId": "AIDAXXXXXXXXXXXXXXXXX",
"Account": "123456789012",
"Arn": "arn:aws:iam::123456789012:user/you"
}
If the output does not resemble what you here, verify the validity of your access key or contact your AWS support team. This message is what you get if your access key is invalid:
$ AWS_PROFILE=org-bastion aws sts get-caller-identity
An error occurred (SignatureDoesNotMatch) when calling the GetCallerIdentity operation: The request signature
we calculated does not match the signature you provided. Check your AWS Secret Access Key and signing method.
Consult the service documentation for details.
The assumerole
error message is better than it used to be in previous versions, and basically gives the same error
as aws sts *
commands. For example:
$ assumerole ixor.vdl-tst-admin 188739
INFO: The profile ixor.vdl-tst-admin passed on the commandline is a valid profile.
INFO: Cache found for ixor.vdl-tst-admin, but credentials have expired and will be deleted.
An error occurred (SignatureDoesNotMatch) when calling the AssumeRole operation: The request signature we
calculated does not match the signature you provided. Check your AWS Secret Access Key and signing method.
Consult the service documentation for details.
Another possible error is this one:
$ assumerole assumerole_profile 384573
INFO: The profile assumerole_profile passed on the commandline is a valid profile.
An error occurred (AccessDenied) when calling the AssumeRole operation: User: arn:aws:iam::123456789012:user/you
is not authorized to perform: sts:AssumeRole on resource: arn:aws:iam::210987654321:role/admin
This can be caused by:
- The permission to assume the role has not been assigned to you, maybe you can have that taken care off by bying your AWS admin a bak Duvel.
- The value of
AWS_STS_DURATION_SECONDS
is higher than what is allowed for that role on the target account. UnsetAWS_STS_DURATION_SECONDS
and try again. If it still does not work, remove themax_session_duration
property from the profile in~/.assumerole
. If that does not help either, contact the AWS support team in your organization.