packer-cn/website/pages/docs/templates/user-variables.mdx

422 lines
13 KiB
Plaintext
Raw Normal View History

2013-08-09 19:25:13 -04:00
---
2017-06-14 21:04:16 -04:00
description: |
2020-03-18 18:46:47 -04:00
User variables allow your templates to be further configured with variables
from the command-line, environment variables, or files. This lets you
parameterize your templates so that you can keep secret tokens,
environment-specific data, and other types of information out of your
templates. This maximizes the portability and shareability of the template.
2015-07-22 22:31:00 -04:00
layout: docs
2020-03-18 18:46:47 -04:00
page_title: User Variables - Templates
sidebar_title: User Variables
---
2013-08-09 19:25:13 -04:00
# Template User Variables
2013-08-09 19:25:13 -04:00
2018-10-26 20:02:51 -04:00
User variables allow your templates to be further configured with variables
from the command-line, environment variables, Vault, or files. This lets you
parameterize your templates so that you can keep secret tokens,
environment-specific data, and other types of information out of your
templates. This maximizes the portability of the template.
2013-08-09 19:25:13 -04:00
Using user variables expects you to know how [configuration
2020-03-31 17:40:07 -04:00
templates](/docs/templates/engine) work. If you don't know how
2018-10-26 20:02:51 -04:00
configuration templates work yet, please read that page first.
2013-08-09 19:25:13 -04:00
## Usage
In order to set a user variable, you must define it either within the
`variables` section within your template, or using the command-line `-var` or
`-var-file` flags.
Even if you want a user variable to default to an empty string, it is best to
explicitly define it. This explicitness helps reduce the time it takes for
newcomers to understand what can be modified using variables in your template.
2013-08-09 19:25:13 -04:00
2018-10-26 20:02:51 -04:00
The `variables` section is a key/value mapping of the user variable name to a
default value. A default value can be the empty string. An example is shown
below:
2013-08-09 19:25:13 -04:00
```json
2013-08-09 19:25:13 -04:00
{
"variables": {
"aws_access_key": "",
"aws_secret_key": ""
},
2020-03-18 18:46:47 -04:00
"builders": [
{
"type": "amazon-ebs",
"access_key": "{{user `aws_access_key`}}",
"secret_key": "{{user `aws_secret_key`}}"
// ...
}
]
2013-08-09 19:25:13 -04:00
}
```
2013-08-09 19:25:13 -04:00
2018-10-26 20:02:51 -04:00
In the above example, the template defines two user variables: `aws_access_key`
and `aws_secret_key`. They default to empty values. Later, the variables are
used within the builder we defined in order to configure the actual keys for
the Amazon builder.
2013-08-09 19:25:13 -04:00
2020-03-18 18:46:47 -04:00
If the default value is `null`, then the user variable will be _required_. This
2018-10-26 20:02:51 -04:00
means that the user must specify a value for this variable or template
validation will fail.
2013-08-31 20:38:38 -04:00
User variables are used by calling the `{{user}}` function in the form of
2020-03-23 20:02:12 -04:00
`{{user 'variable'}}`. This function can be used in _any value_ but `type`
within the template: in builders, provisioners, _anywhere outside the `variables` section_.
User variables are available globally within the rest of the template.
2013-08-09 19:25:13 -04:00
## Environment Variables
Environment variables can be used within your template using user variables.
2020-03-18 18:46:47 -04:00
The `env` function is available _only_ within the default value of a user
variable, allowing you to default a user variable to an environment variable.
2015-07-22 22:31:00 -04:00
An example is shown below:
```json
{
"variables": {
2020-03-18 18:46:47 -04:00
"my_secret": "{{env `MY_SECRET`}}"
}
}
```
2020-03-18 18:46:47 -04:00
This will default "my_secret" to be the value of the "MY_SECRET" environment
variable (or an empty string if it does not exist).
2020-03-23 20:02:12 -04:00
-> **Why can't I use environment variables elsewhere?** User variables are
2015-07-22 22:31:00 -04:00
the single source of configurable input to a template. We felt that having
2020-03-18 18:46:47 -04:00
environment variables used _anywhere_ in a template would confuse the user
2018-10-26 20:02:51 -04:00
about the possible inputs to a template. By allowing environment variables only
within default values for user variables, user variables remain as the single
source of input to a template that a user can easily discover using
2015-07-22 22:31:00 -04:00
`packer inspect`.
2020-03-23 20:02:12 -04:00
-> **Why can't I use `~` for home variable?** `~` is an special variable
that is evaluated by shell during a variable expansion. As Packer doesn't run
2016-03-14 13:34:26 -04:00
inside a shell, it won't expand `~`.
2018-08-08 11:44:51 -04:00
## Consul keys
Consul keys can be used within your template using the `consul_key` function.
2020-03-18 18:46:47 -04:00
This function is available _only_ within the default value of a user variable,
2018-08-08 11:44:51 -04:00
for reasons similar to environment variables above.
```json
2018-08-08 11:44:51 -04:00
{
2020-03-18 18:46:47 -04:00
"variables": {
"soft_versions": "{{ consul_key `my_image/softs_versions/next` }}"
}
2018-08-08 11:44:51 -04:00
}
```
2018-10-26 20:02:51 -04:00
This will default `soft_versions` to the value of the key
`my_image/softs_versions/next` in consul.
2018-08-08 11:44:51 -04:00
2018-10-26 20:02:51 -04:00
The configuration for consul (address, tokens, ...) must be specified as
environment variables, as specified in the
[Documentation](https://www.consul.io/docs/commands#environment-variables).
2018-08-08 11:44:51 -04:00
## Vault Variables
Secrets can be read from [Vault](https://www.vaultproject.io/) and used within
2020-03-18 18:46:47 -04:00
your template as user variables. the `vault` function is available _only_
within the default value of a user variable, allowing you to default a user
variable to a vault secret.
An example of using a v2 kv engine:
If you store a value in vault using `vault kv put secret/hello foo=world`, you
can access it using the following template engine:
```json
{
"variables": {
"my_secret": "{{ vault `/secret/data/hello` `foo`}}"
}
}
```
2018-10-26 20:02:51 -04:00
2020-03-18 18:46:47 -04:00
which will assign "my_secret": "world"
An example of using a v1 kv engine:
If you store a value in vault using:
2018-10-26 20:02:51 -04:00
vault secrets enable -version=1 -path=secrets kv
vault kv put secrets/hello foo=world
You can access it using the following template engine:
2018-10-26 20:02:51 -04:00
{
"variables": {
"VAULT_SECRETY_SECRET": "{{ vault `secrets/hello` `foo`}}"
}
}
2018-10-26 20:02:51 -04:00
This example accesses the Vault path `secret/data/foo` and returns the value
2020-03-18 18:46:47 -04:00
stored at the key `bar`, storing it as "my_secret".
In order for this to work, you must set the environment variables `VAULT_TOKEN`
and `VAULT_ADDR` to valid values.
2019-05-09 16:42:48 -04:00
The api tool we use allows for more custom configuration of the Vault client via
environment variables.
The full list of available environment variables is:
2020-03-31 18:47:06 -04:00
```text
2019-05-09 16:42:48 -04:00
"VAULT_ADDR"
"VAULT_AGENT_ADDR"
"VAULT_CACERT"
"VAULT_CAPATH"
"VAULT_CLIENT_CERT"
"VAULT_CLIENT_KEY"
"VAULT_CLIENT_TIMEOUT"
"VAULT_SKIP_VERIFY"
"VAULT_NAMESPACE"
"VAULT_TLS_SERVER_NAME"
"VAULT_WRAP_TTL"
"VAULT_MAX_RETRIES"
"VAULT_TOKEN"
"VAULT_MFA"
"VAULT_RATE_LIMIT"
```
and detailed documentation for usage of each of those variables can be found
[here](https://www.vaultproject.io/docs/commands/#environment-variables).
## AWS Secrets Manager Variables
Secrets can be read from [AWS Secrets Manager](https://aws.amazon.com/secrets-manager/)
and used within your template as user variables. The `aws_secretsmanager` function is
available _only_ within the default value of a user variable, allowing you to default
a user variable to an AWS Secrets Manager secret.
### Plaintext Secrets
```json
{
"variables": {
"password": "{{ aws_secretsmanager `globalpassword` }}"
}
}
```
In the example above it is assumed that the secret `globalpassword` is not
stored as a key pair but as a single non-JSON string value. Which the
`aws_secretsmanager` function will return as a raw string.
### Single Key Secrets
```json
{
"variables": {
"password": "{{ aws_secretsmanager `sample/app/password` }}"
}
}
```
In the example above it is assumed that only one key is stored in
`sample/app/password` if there are multiple keys stored in it then you need
to indicate the specific key you want to fetch as shown below.
### Multiple Key Secrets
```json
{
"variables": {
"db_password": "{{ aws_secretsmanager `sample/app/passwords` `db` }}",
"api_key": "{{ aws_secretsmanager `sample/app/passwords` `api_key` }}"
}
}
```
In order to use this function you have to configure valid AWS credentials using
one of the following methods:
- [Environment Variables](https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-envvars.html)
- [CLI Configuration Files](https://docs.aws.amazon.com/cli/latest/userguide/cli-configure-files.html)
- [Container Credentials](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-iam-roles.html)
- [Instance Profile Credentials](https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/iam-roles-for-amazon-ec2.html)
## Using array values
Some templates call for array values. You can use template variables for these,
too. For example, the `amazon-ebs` builder has a configuration parameter called
`ami_regions`, which takes an array of regions that it will copy the AMI to.
You can parameterize this by using a variable that is a list of regions, joined
by a `,`. For example:
```json
{
2020-03-18 18:46:47 -04:00
"variables": {
"destination_regions": "us-west-1,us-west-2"
},
"builders": [
{
"ami_name": "packer-qs-{{timestamp}}",
"instance_type": "t2.micro",
"region": "us-east-1",
"source_ami_filter": {
"filters": {
"name": "*ubuntu-xenial-16.04-amd64-server-*",
"root-device-type": "ebs",
"virtualization-type": "hvm"
},
"most_recent": true,
"owners": ["099720109477"]
},
"ami_regions": "{{user `destination_regions`}}",
"ssh_username": "ubuntu",
"type": "amazon-ebs"
}
]
}
```
2013-08-09 19:25:13 -04:00
## Setting Variables
2018-10-26 20:02:51 -04:00
Now that we covered how to define and use user variables within a template, the
next important point is how to actually set these variables. Packer exposes two
methods for setting user variables: from the command line or from a file.
2013-08-09 19:25:13 -04:00
### From the Command Line
2018-10-26 20:02:51 -04:00
To set user variables from the command line, the `-var` flag is used as a
parameter to `packer build` (and some other commands). Continuing our example
above, we could build our template using the command below. The command is
split across multiple lines for readability, but can of course be a single
line.
2013-08-09 19:25:13 -04:00
```shell-session
2013-08-09 19:25:13 -04:00
$ packer build \
-var 'aws_access_key=foo' \
-var 'aws_secret_key=bar' \
template.json
```
2015-07-22 22:31:00 -04:00
As you can see, the `-var` flag can be specified multiple times in order to set
2018-10-26 20:02:51 -04:00
multiple variables. Also, variables set later on the command-line override any
earlier set variable of the same name.
2013-08-09 19:25:13 -04:00
2018-10-26 20:02:51 -04:00
**warning** If you are calling Packer from cmd.exe, you should double-quote
your variables rather than single-quoting them. For example:
`packer build -var "aws_secret_key=foo" template.json`
2013-08-09 19:25:13 -04:00
### From a File
2018-10-26 20:02:51 -04:00
Variables can also be set from an external JSON file. The `-var-file` flag
reads a file containing a key/value mapping of variables to values and sets
those variables. An example JSON file may look like this:
2013-08-09 19:25:13 -04:00
```json
2013-08-09 19:25:13 -04:00
{
"aws_access_key": "foo",
"aws_secret_key": "bar"
}
```
2013-08-09 19:25:13 -04:00
2015-07-22 22:31:00 -04:00
It is a single JSON object where the keys are variables and the values are the
variable values. Assuming this file is in `variables.json`, we can build our
template using the following command:
2013-08-09 19:25:13 -04:00
2020-03-18 18:46:47 -04:00
```text
On Linux :
2013-08-09 19:25:13 -04:00
$ packer build -var-file=variables.json template.json
On Windows :
packer build -var-file variables.json template.json
2013-08-09 19:25:13 -04:00
```
2018-10-26 20:02:51 -04:00
The `-var-file` flag can be specified multiple times and variables from
multiple files will be read and applied. As you'd expect, variables read from
files specified later override a variable set earlier.
2013-08-09 19:25:13 -04:00
Combining the `-var` and `-var-file` flags together also works how you'd
2018-10-26 20:02:51 -04:00
expect. Variables set later in the command override variables set earlier. So,
for example, in the following command with the above `variables.json` file:
```shell-session
$ packer build \
-var 'aws_access_key=bar' \
-var-file=variables.json \
-var 'aws_secret_key=baz' \
template.json
```
Results in the following variables:
2020-03-18 18:46:47 -04:00
| Variable | Value |
| -------------- | ----- |
| aws_access_key | foo |
| aws_secret_key | baz |
# Sensitive Variables
If you use the environment to set a variable that is sensitive, you probably
don't want that variable printed to the Packer logs. You can make sure that
sensitive variables won't get printed to the logs by adding them to the
"sensitive-variables" list within the Packer template:
```json
{
"variables": {
"my_secret": "{{env `MY_SECRET`}}",
"not_a_secret": "plaintext",
"foo": "bar"
},
"sensitive-variables": ["my_secret", "foo"],
...
}
```
The above snippet of code will function exactly the same as if you did not set
"sensitive-variables", except that the Packer UI and logs will replace all
2020-03-18 18:46:47 -04:00
instances of "bar" and of whatever the value of "my_secret" is with
`<sensitive>`. This allows you to be confident that you are not printing
secrets in plaintext to our logs by accident.
# Recipes
## Making a provisioner step conditional on the value of a variable
2018-10-26 20:02:51 -04:00
There is no specific syntax in Packer templates for making a provisioner step
conditional, depending on the value of a variable. However, you may be able to
do this by referencing the variable within a command that you execute. For
example, here is how to make a `shell-local` provisioner only run if the
`do_nexpose_scan` variable is non-empty.
```json
{
"type": "shell-local",
"command": "if [ ! -z \"{{user `do_nexpose_scan`}}\" ]; then python -u trigger_nexpose_scan.py; fi"
}
```
2016-03-14 13:34:26 -04:00
## Using HOME Variable
In order to use `$HOME` variable, you can create a `home` variable in Packer:
2016-03-14 13:34:26 -04:00
```json
{
"variables": {
"home": "{{env `HOME`}}"
}
2016-03-14 13:34:26 -04:00
}
```
And this will be available to be used in the rest of the template, i.e.:
2016-03-14 13:34:26 -04:00
```json
2016-03-14 13:34:26 -04:00
{
"builders": [
{
2020-03-18 18:46:47 -04:00
"type": "google",
"account_file": "{{ user `home` }}/.secrets/gcp-{{ user `env` }}.json"
}
]
2016-03-14 13:34:26 -04:00
}
```