brad058
A Pattern for Managing Clusters in the Cloud with Ansible Vault
September 8th, 2016
David Kegley is a Software Engineer at B23 LLC
If you’ve ever been tasked with standing up a cluster environment in the cloud, you’ve probably used a tool like Ansible to make your life a little easier. By defining the configuration in code, admins can develop their cloud infrastructure in a maintainable and repeatable way. This allows for iteration through trial and error when developing for cloud.
One drawback of developing in this fashion is that it can be difficult to maintain access to a cluster during development. When large teams require access to a common cloud environment, security is often the first casualty in the development process. Maintaining an individual private key for each developer is impractical while developing a cluster for a large team. Cycling keys is a good idea but it means re-distributing the private key to the entire team every time a key is changed.
Ansible provides some useful functionality that can alleviate some of these maintenance woes. Vault utilizes symmetric key encryption which allows us to store private keys in our repository while still providing an acceptable level of security for our cloud. Note that you should never store unencrypted private keys in source control.
By storing these encrypted private keys in a repository, developers are able to access the cluster even if the private key has changed. We can still cycle the private key frequently without the worry of hindering development. Should a private key be misplaced, cycling keys is as simple as encrypting a new private key and replacing the old one in source control.
So how is this accomplished?
First, we will generate a random key to use as our vault password file. This key should only be shared with trusted individuals and it should never be committed to source control. Generate the key and place it in the keys/directory using the following command:
There are many ways to generate random strings, I’m using date | md5 for the sake of simplicity but any method will work.
Define your cluster for whichever environment you want to launch. Here I’m using dev:
Next, just launch the cluster by running the top-level playbook and pass the extra env variable to define the environment.
You should see your cluster launching through the AWS console. It’s also important to note that the play has saved your cluster private key in the keys/directory. This key can be used to connect to any of the nodes in the cluster over ssh.
There is now an encrypted .vault file in the keys/ directory. This is your encrypted private key file. This file is safe to be committed to version control because it cannot be decrypted without vault-key.txt which we generated in the first step. Once it is in your repository, any team member that possesses the vault-key will be able to access the cluster.
Team members can run
once they have obtained the .vault file (from version control) and the vaNExt blog to doult-key (from you). This will decrypt the vault file and place the cluster private key in their keys/ directory
The tasks for encrypting, decrypting, and saving the private key are as follows:
If the private key already exists on AWS then the contents of the key are not returned, so we only save the private key when a new key is created in AWS. If the key is already present, we know that it should be in source control as an encrypted vault file.
This final task should execute no matter what. The vault file is implicitly decrypted when using the copy module.
Should the vault-key be misplaced, it may be necessary to change your vault-key file and re-distribute it to your team. This can be done using the ansible-vault rekey command. As an additional security measure, you should also cycle your private keys at that time
This is enough to get started but there are many ways this concept could be extended to support different cloud use cases.
The source for this project can be found here: https://github.com/dbkegley/ansibleVaultBlog