One of my favorite visualization tools for diagraming Amazon Web Services (AWS) environments is Duo CloudMapper. CloudMapper helps you understand visually what exists in your AWS accounts by running a collection against the environment and providing an interactive web page. This is extremely handy for identifying possible network misconfigurations, along with a slew of other benefits. For a full listing why I like this tool check out my post on How to Visualize Your Cloud Deployments with CloudMapper.
Despite it’s power, one of the challenges I have found is to simply get it started and working. CloudMapper is open source built upon other open source products and I have found that there are inevitably build and dependency issues that suck up my time before I can simply use the tool. For these reasons and to make things easier in general, I chose to create and deploy CloudMapper as virtual appliance.
Building the Virtual Appliance
I utilized Packer to provision my CloudMapper virtual appliance. Packer is excellent for creating machine images for multiple platforms from a single source configuration. In this case we will build out an Amazon Machine Image (AMI) with Packer, which will take care of all package installation and dependencies for the build out. You can learn more about all the Packer goodness on the HashiCorp website and Paul Kirby provides a nice overview in his Packer PluralSight course.
Download the cloudmapper.packer template from my GitHub account. (Packer templates are simply JSON files that specify the various components used to create the machine image, and where the build of the image will be saved. In our case we will be creating and deploying our virtual appliance into AWS, but Packer comes with support to build images for Amazon EC2, CloudStack, DigitalOcean, Docker, Google Compute Engine, Microsoft Azure, QEMU, VirtualBox, VMware, and more.)
Specify AWS Credentials for creating our virtual appliance. There are a number of ways to accomplish this but we will use environment variables.
$ export AWS_ACCESS_KEY_ID="awsaccesskey"
$ export AWS_SECRET_ACCESS_KEY="awsecretkey"
Build the image.
$ packer build -var aws_region="us-west-2" -var ami_id="ami-6cd6f714" -var python_version="3.5.6" cloudmapper.packer
# aws_region is where the image will be stored.
# ami_id is the base Amazon Linux image in the region.
# python version of your choice.
There are currently some issues with CloudMapper and Python 3.7, so I am using the recommend version of 3.5.6
The build process will take ~10-15 minutes as it needs to compile and pull down all of the components. Once it is complete, Packer will notify of your unique AMI that can now be used for deployment.
Deploying the Virtual Appliance
Now that the image for our virtual appliance is available in AWS, let deploy it and run CloudMapper. My preferred way to deploy would be using Terraform but for purposes of this post we will step through the manual steps.
Launch an instance using the newly created CloudMapper image. You can accept the defaults providing your instance a public IP with SSH access.
Configure CloudMapper by logging in via SSH and performing the final initialization steps. (While these could be automated and built into the image, I get sensitive about saving AWS credentials anywhere even if my image is private. I prefer to specify them when needed.)
$ aws configure
You can specify a full access account to run CloudMapper but I like least privilege so have setup a “Visualization” IAM user with the privileges specified in the CloudMapper readme.
Configure CloudMapper’s account information in the config.json file to match aws credentials:
$ cd ~/cloudmapper
$ pipenv run python3 cloudmapper.py configure add-account --config-file config.json --name AWS_USERNAME --id AWS_ACCESS_KEY_ID
#AWS_USERNAME is “friendly name” tied to IAM account
#AWS_ACCESS_KEY_ID is the AWS Access Key ID specified in aws configure.
Run CloudMapper’s collection against the environment. The collection phase can take some time, as it is truly pulling all the metadata information for your entire AWS account across all components and regions.
$ pipenv run python3 cloudmapper.py collect --account AWS_USERNAME
Prepare the results and launch the webserver to display them.
$ pipenv run python3 cloudmapper.py prepare --config config.json --account AWS_USERNAME
$ pipenv run python3 cloudmapper.py webserver --public
Create and attach a security group to the instance to make the site publicly available.
Browse to public DNS address of your virtual appliance on port 8000
Please note that these steps show running this instance with a publicly available website. You can certainly deploy this to a private subnet and access through a bastion server, etc which is recommended. It would also make sense to put this site behind a login which I have noted as an opportunity for further improvement. Be sure to stop this instance when you are done using it.
Further Improvements
Having a readily available virtual appliance that just works is perfect, but there are some further improvements that I think would be handy:
Create a docker image of CloudMapper that can be run as a container. (There are some folks who have built this)
Save the collection data to an external volume so that it doesn’t live in the running appliance.
Create the virtual appliance that can be deployed within other Packer supported platforms, namely vSphere and Azure.
Lock down the website behind a username and password.