A credential helper for Terraform Cloud/Enterprise, or to store other secrets, securely in the operating system's credential vault or through a third party vault. No longer keep secrets in a plain text configuration file!
We all know storing secrets in plain text can pose major security threats, and Terraform doesn't come pre-packaged with a credential helper, so we decided to create one and to share it with the greater Terraform/DevOps community to help enable stronger security practices
- Windows (Credential Manager)
- MacOS (Keychain)
- Linux (gnome-keyring) Tested on Ubuntu 20.04
- AWS Secrets Manager
- Azure Key Vault
- Google Secret Manager
- HashiCorp Vault
The fastest way to install terracreds
on Windows is via our Chocolatey package:
choco install terracreds --version "2.0.0" -y
Once installed run the following command to verify terracreds
was installed properly:
terracreds -v
To upgrade terracreds
to the latest version with Chocolatey run the the following command:
choco upgrade terracreds --version "2.0.0" -y
We are currently working on a homebrew
package, however, to install the package simply download our latest release from this repository,
extract the package, and then place it in a directory available on $HOME
You'll need to download the latest binary from our release page and place it anywhere on $PATH
of your system. You can also copy and run the following commands:
wget https://github.com/tonedefdev/terracreds/releases/download/v2.0.0/terracreds_2.0.0_linux_amd64.tar.gz && \
tar -xvf terracreds_2.0.0_linux_amd64.tar.gz && \
sudo mv -f terracreds /usr/bin/terracreds && \
rm -f terracreds_2.0.0_linux_amd64.tar.gz README.md
The terracreds
Linux implementation uses gnome-keyring
in conjunction with gnome-keyring-daemon
to utilize the credential storage engine
In order to leverage terracreds
to have access to the default Login
collection you'll need to unlock
the collection with gnome-keyring-daemon
using an empty password:
echo "" | gnome-keyring-daemon --unlock
You do have the option of setting a password by passing it in with
echo
but every call toterracreds get
will require the unlock password
The command, if successful, should return the following:
SSH_AUTH_SOCK=/run/user/1000/keyring/ssh
You can verify that it's running properly with:
ps -ef | grep 'gnome-keyring-daemon'
Download the source files by entering the following command:
go get github.com/tonedefev/terracreds
Ensure you have the environment variable GO111MODULE
enabled since this project leverages go.mod
For Windows:
$env:GO111MODULE='on'
For macOS and Linux:
export GO111MODULE='on'
Once the files have been downloaded navigate to the terracreds
directory in the and then run:
go install -v
Navigate to the root of the project directory and you should see the terracreds.exe
binary for Windows or terracreds
for macOS and Linux. On Windows, copy the .exe
to any directory of your choosing. Be sure to add the directory on $env:PATH
for Windows to make using the application easier. On macOS and Linux we recommend you place the binary in /usr/bin
as this directory should already be on the $PATH
environment variable
If you're upgrading to the latest version of terracreds
from a previous version use one of the methods above to install the latest binary. Once successfully installed on your system you just need to run terracreds generate
to copy the latest version to the correct plugins
directory for your operating system
In order for terracreds
to act as your credential provider you'll need to generate the binary and the plugin directory in the default location that Terraform looks for plugins. Specifically, for credential helpers, and for Windows, the directory is %APPDATA%\terraform.d\plugins
and for macOS and Linux $HOME/.terraform.d/.terraformrc
To make things as simple as possible we created a helper command to generate everthing needed to use the app. All you need to do is run the following command in terracreds
to generate the plugin directory, and the correctly formatted binary that Terraform will use:
terracreds generate
This command will generate the binary as terraform-credentials-terracreds.exe
for Windows or terraform-credentials-terracreds
for macOS and Linux which is the valid naming convention for Terraform to recognize this plugin as a credential helper
In addition to the binary and plugin a terraform.rc
file is required for Windows or .terraformrc
for macOS and Linux with a credentials_helper
block which instructs Terraform to use the specified credential helper. If you don't already have a terraform.rc
or a .terraformrc
file you can pass in --create-cli-config
to create the file with the credentials helper block already generated for use with the terracreds
binary for your OS
However, if you already have a terraform.rc
or .terraformrc
file you will need to add the following block to your file instead:
credentials_helper "terracreds" {
args = []
}
Once you have moved all of your tokens from this file to the Windows Credential Manager
or KeyChain
via terracreds
you can remove the tokens from the file. If you don't remove the tokens, and you add the credentials_helper
block to this file, Terraform will still use the tokens instead of terracreds
to retreive the tokens, so be sure to remove your tokens from this file once you have used the create
or terraform login
command to create the credentials in terracreds
so you can actually leverage the credential helper
For Terraform to properly use the credentials stored in your credential manager they need to be stored a specific way. The name of the credential object must be the domain name of the Terraform Cloud or Enterprise server. For instance app.terraform.io
which is the default name terraform login
will use
The value for the password will correspond to the API token associated for that specific Terraform Cloud or Enterprise server
The entire process is kicked off directly from the Terraform CLI. Run terraform login
to start the login process with Terraform Cloud. If you're using Terraform Enterprise you'll need to pass the hostname of the server as an additional argument terraform login my.tfe.com
You'll be sent to your Terraform Cloud instance where you'll be requested to sign-in with your account, and then sent to create an API token. Create the API token with any name you'd like for this example we'll use terracreds
Once completed, copy the generated token, paste it into your terminal, and then hit enter. Terraform will then leverage terracreds
to store the credentials in the operating system's credential manager. If all went well you should receive the following success message:
Success! Terraform has obtained and saved an API token.
In the background terraform
calls terracreds
as its credential helper, terraform
passes in a JSON token credential object, and then terracreds
decodes that object from STDIN for storage in the operating system's credential manager. The following command is what is called by terraform
during this process:
terraform-credentials-terracreds store app.terraform.io
When Terraform leverages terracreds
as the credential provider it will run the following command to get the credentials value:
terraform-credentials-terracreds get app.terraform.io
Alternatively, you can run the same command using either binary to return the credentials. The response is formatted as a JSON object as required by Terraform to use the token:
terracreds get app.terraform.io
Example output:
{"token":"reallybigtokenyoudontevenknow"}
To update a credential in your credential manager simply go through the same terraform login
process and it will generate a new token and save it for you!
If the token was updated successfully the following message is returned:
Success! Terraform has obtained and saved an API token.
Additionally, you can check the terracreds.log
if logging is enabled for more information
You can delete the credential object at any time by running:
terraform logout
In the background terraform
calls terracreds
to perform:
terracreds forget app.terraform.io
If the credential was successfully deleted terraform
will return:
Success! Terraform has removed the stored API token for app.terraform.io.
Additionally, you can check the terracreds.log
if logging is enabled for more information
You can reference example configs in our repo plus we have example terraform code you can reference in order to setup your
AWS
orAzure
VMs to useterracreds
for a CI/CD piepline agent or a development workstation
Currently, we only support using an
EC2 Instance Role
for authentication. This ensures the highest level of security by alleviating thesecret zero
dilemma
In order to leverage terracreds
to manage secrets in AWS Secrets Manager
the following block needs to be provided in the configuration file:
aws:
description: my_terraform_api_token
region: us-west-2
secretName: my-secret-name
Value | Description | Required |
---|---|---|
description |
A brief description to provide for the secret object viewable in Secrets Manager |
yes |
region |
The Secrets Manager instance's region where the secret will be stored |
yes |
secretName |
A name for the secret. If omitted and using terraform login the hostname of the TFC\TFE server will be used for the name instead |
no |
The following role permissions are required in order for the EC2 Instance Role
to levearge terracreds
with AWS Secrets Manager
:
Action = [
"secretsmanager:CreateSecret",
"secretsmanager:DeleteSecret",
"secretsmanager:GetSecretValue",
"secretsmanager:PutSecretValue"
]
Currently, we only support using a
Managed Service Identity
for authentication. This ensures the highest level of security by alleviating thesecret zero
dilemma
In order to leverage terracreds
to manage secrets in Azure Key Vault
the following block needs to be provided in the configuration file:
azure:
secretName: my-secret-name
useMSI: true
vaultUri: https://keyvault.azure.net
Value | Description | Required |
---|---|---|
secretName |
A name for the secret. If omitted and using terraform login the hostname of the TFC\TFE server will be used for the name instead |
no |
useMSI |
A flag to choose whether or not to use Manged Service Identity . Currently, true is required |
yes |
vaultUri |
The URI for the Azure Key Vault where you want to store or retrieve your credentials |
yes |
The following Azure Key Vault Access Policies
are required to be given to the Managed Service Identity
for it to leverage terracreds
:
secret_permissions = [
"Get",
"List",
"Set",
"Delete"
]
Since
Azure Key Vault
doesn't support the period character in a secret name a helper function will replace any periods with dashes so they can be successfully stored. This means aterraform
API token name that would usually beapp.terraform.io
will becomeapp-terraform-io
In order to leverage terracreds
to manage secrets in HashiCorp Vault
the following block needs to be provided in the configuration file:
hcvault:
environmentTokenName: HASHI_TOKEN
keyVaultPath: kv
secretName: my-secret-name
secretPath: tfe
vaultUri: http://localhost:8200
Value | Description | Required |
---|---|---|
environmentTokenName |
The name of the environment variable that contains the token value to authenticate with HashiCorp Vault |
yes |
keyVaultPath |
The path to the Key Vault object within the vault |
yes |
secretName |
A name for the secret. If omitted and using terraform login the hostname of the TFC\TFE server will be used for the name instead |
no |
secretPath |
The path of the secret within HashiCorp Vault |
yes |
vaultUri |
The URI for the HashiCorp Vault instance |
yes |
In order to add some protection terracreds
adds a username to the credential object to secrets stored in the local operating system, and checks to ensure that the user requesting access to the token is the same user as the token's creator. This means that only the user account used to create the token can view the token from terracreds
which ensures that the token can only be read by the account used to create it. Any attempt to access or modify this token from terracreds
outside of the user that created the credentail will lead to denial messages. Additionally, if the credential name is not found, the same access denied message will be provided in lieu of a generic not found message to help prevent brute force attempts
Wherever either binary is stored terracreds
or terraform-credential-terracreds
a config.yaml
file is generated on first launch of the binary. Currently, this configuration file only enables/disables logging and sets the log path. If logging is enabled you'll find the log named terracreds.log
at the provided path
It's important to note that you'll have two configuration files due to Terraform requiring that the credential helper have a very specific binary name, so when troubleshooting credential issues with Terraform remember to setup the configuration file in the
%APPDATA%\terraform.d\plugins
directory for Windows and$HOME/.terraform.d/plugins
directory for macOS and Linux
To enable logging for Windows setup the config.yaml
as follows:
logging:
enabled: true
path: C:\Temp\
To enable logging for macOS and Linux:
logging:
enabled: true
path: /home/username/
The log is helpful in understanding if an object was found, deleted, updated or added, and will be found at the path defined in the configuration file as terracreds.log
In addition all error messages returned by the underlying libraries will be logged when logging is enabled and an error is encountered
If you are having trouble viewing, deleting, or saving credentials on Linux systems using gnome-keyring
you must ensure that you have unlocked the collection using gnome-keyring-daemon --unlock
otherwise you will see the following error message in the logs:
ERROR: <TIMESTAMP> - failed to unlock correct collection '/org/freedesktop/secrets/collection/login'
If the daemon has unlocked the collection but you're still getting prompted for credentials -- check to make sure that only a single instance of the daemon is running:
ps -ef | grep gnome-keyring
If more than one daemon is running, take note of the pid, and use kill
to terminate the additional daemon. Try your previous command again
and it should now be working