⚠️ This approach is deprecated in favour of SSH over AWS SSM even without adding SSH port to security group
ℹ️ See my SSH Proxy Command to tranparently connect to your EC2 instances vie AWS SSM session.
This approach will sync public ssh keys for user and groups from IAM account to an S3 bucket as authorized_keys files. On the ssh daemon side AuthorizedKeysCommand is used to request authorized keys from S3 bucket on demand on ssh connection establishment. So you can manage all ssh key access to your instances via IAM.
Create lambda function to sync SSH keys from IAM to S3 This function will also add IAM account as SSH_KEY_OWNER
environment variable to public keys e.g.
environment="SSH_KEY_OWNER=john@example.org" ssh-rsa AAAAB3NzaC1yc2EAAAADAQA...dOXmwPQ== john@example.org
-
Install Serverless framework
-
Adjust S3 bucketname
<S3Bucket>
within serverless.yml -
Create S3 Bucket
-
Bucket Policy - adjust
<AccountId>
and<S3Bucket>
{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Principal": { "AWS": [ "arn:aws:iam::<AccountId_1>:root", "arn:aws:iam::<AccountId_2>:root", "arn:aws:iam::<AccountId_3>:root", ] }, "Action": [ "s3:ListBucket", "s3:GetObject" ], "Resource": [ "arn:aws:s3:::<S3Bucket>", "arn:aws:s3:::<S3Bucket>/*" ] } ] }
-
serverless deploy -v
-
Create Role Policy in Sub Account - adjust
<S3Bucket>
{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Action": [ "s3:ListBucket", "s3:GetObject" ], "Resource": [ "arn:aws:s3:::<S3Bucket>", "arn:aws:s3:::<S3Bucket>/*" ] } ] }
-
Attach policy to EC2 roles who should be able to access public ssh keys from IAM account
-
Set following script as Instance Userdata
#!/bin/bash #### SSH Daemon Setup #### # adjust ssh daemon config cat >> /etc/ssh/sshd_config <<'EOF' LogLevel VERBOSE PermitUserEnvironment yes AuthorizedKeysCommand /usr/local/sbin/sshd_authorized_keys_command.sh AuthorizedKeysCommandUser nobody EOF # create authorized keys command cat > /usr/local/sbin/sshd_authorized_keys_command.sh <<'EOF' #!/bin/bash SSH_USER=$1 SSH_USER_HOME=$(getent passwd ${SSH_USER} | cut -d: -f6) if [ "$SSH_USER" == 'ec2-user' ]; then IAM_BUCKET='<S3Bucket>' IAM_PRINCIPALS='<IAMPrincipals>' fi for iam_principal in ${IAM_PRINCIPALS//,/ } ; do aws s3 cp s3://${IAM_BUCKET}/${iam_principal}/authorized_keys - done EOF chmod a+x /usr/local/sbin/sshd_authorized_keys_command.sh # configure access logging cat > /etc/ssh/sshrc <<'EOF' #!/bin/bash export SSH_KEY_OWNER=${SSH_KEY_OWNER:-'unknown'} logger -ip authpriv.notice -t sshd "Public key owner is ${SSH_KEY_OWNER} for connection $(tmp=${SSH_CLIENT% *}; echo ${tmp// / port })" echo "Publickey of ${SSH_KEY_OWNER}" EOF chmod a+x /etc/ssh/sshrc # restart ssh daemon service sshd restart
-
Ensure AWS CLI is available on EC2 instance, if not prepand
pip install awscli
to Userdata Script -
Ensure IAM Instance Rolle has acces to IAM S3 Bucket (see above)
-
Configure Userdata Script
-
Set
IAM_BUCKET
, the S3 bucket name where principals are stored e.g. 'company-iam' -
Set
IAM_PRINCIPALS
, a comma separated list in form of groups/[GroupName] and users/[UserName]-
Inline
IAM_PRINCIPALS='groups/Administrators, user/Admin'
-
From File - Single principal per line
IAM_PRINCIPALS=$(cat /home/ec2-user/.ssh/iam_principals | while read line; do echo -n "${line},"; done;)
-
From AWS Parameter Store
INSTANCE_IAM_ROLE=$(curl -fs http://169.254.169.254/latest/meta-data/iam/security-credentials/) INSTANCE_REGION=$(curl -s http://169.254.169.254/latest/meta-data/placement/availability-zone | sed 's/.$//') IAM_PRINCIPALS=$(aws ssm get-parameter --region "${INSTANCE_REGION}" --name "/IAM/${INSTANCE_IAM_ROLE}/principals" --query 'Parameter.Value' --output text)
-
Create Prinicpals Parameter
/IAM/<IAM_INSTANCE_ROLE>/principals
within AWS Parameter Store -
Create Inline Policy for IAM Instance Role to grant access to AWS Parameter Store
{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Action": [ "ssm:GetParameter", "ssm:GetParameters" ], "Resource": [ "arn:aws:ssm:eu-central-1:*:parameter/IAM/<IAM_INSTANCE_ROLE>/*" ] } ] }
-
-
-
-