packer-cn/website/content/docs/templates/hcl_templates/variables.mdx

312 lines
11 KiB
Plaintext

---
page_title: Input Variables - HCL Configuration Language
sidebar_title: Variables
description: |-
Input variables are parameters for Packer modules.
This page covers configuration syntax for variables.
---
# Input Variables
`@include 'from-1.5/beta-hcl2-note.mdx'`
There are two kinds of variables in HCL Packer templates: Input variables,
sometimes simply called "variables", and Local variables, also known as
"locals". Input variables may have defaults, but those defaults can
be overridden from the command line or special variable files. Local variables
can be thought of as constants, and are not able to be overridden at runtime.
This page is about input variables. To learn about local variables, see the
[locals](/docs/templates/hcl_templates/locals) page.
Input variables serve as parameters for a Packer build, allowing aspects of the
build to be customized without altering the build's own source code.
When you declare variables in the build of your configuration, you can set
their values using CLI options and environment variables.
Input variable and local variable usage are introduced in the [_Variables
Guide_](/guides/hcl/variables).
-> **Note:** For brevity, input variables are often referred to as just
"variables" or "Packer variables" when it is clear from context what sort of
variable is being discussed. Other kinds of variables in Packer include
_environment variables_ (set by the shell where Packer runs) and _expression
variables_ (used to indirectly represent a value in an
[expression](/docs/templates/hcl_templates/expressions)).
## Declaring an Input Variable
Each input variable accepted by a build must be declared using a `variable`
block :
```hcl
variable "image_id" {
type = string
}
variable "availability_zone_names" {
type = list(string)
default = ["us-west-1a"]
}
variable "docker_ports" {
type = list(object({
internal = number
external = number
protocol = string
}))
default = [
{
internal = 8300
external = 8300
protocol = "tcp"
}
]
}
```
Or a less precise variables block:
```hcl
variables {
foo = "value"
my_secret = "foo"
}
```
The label after the `variable` keyword or a label of a `variables` block is a
name for the variable, which must be unique among all variables in the same
build. This name is used to assign a value to the variable from outside and to
reference the variable's value from within the build.
The `variable` block can optionally include a `type` argument to specify what
value types are accepted for the variable, as described in the following
section.
The `variable` declaration can also include a `default` argument. If present,
the variable is considered to be _optional_ and the default value will be used
if no value is set when calling the build or running Packer. The `default`
argument requires a literal value and cannot reference other objects in the
configuration.
## Using Input Variable Values
Within the build that declared a variable, its value can be accessed from
within [expressions](/docs/templates/hcl_templates/expressions) as `var.<NAME>`, where `<NAME>`
matches the label given in the declaration block:
```hcl
source "googlecompute" "debian" {
zone = var.gcp_zone
tags = var.gcp_debian_tags
}
```
The value assigned to a variable can be accessed only from expressions within
the folder where it was declared.
## Type Constraints
The `type` argument in a `variable` block allows you to restrict the [type of
value](/docs/templates/hcl_templates/expressions#types-and-values) that will be accepted as the value
for a variable. If no type constraint is set then a value of any type is
accepted.
While type constraints are optional, we recommend specifying them; they serve
as easy reminders for users of the build, and allow Packer to return a helpful
error message if the wrong type is used.
Type constraints are created from a mixture of type keywords and type
constructors. The supported type keywords are:
- `string`
- `number`
- `bool`
The type constructors allow you to specify complex types such as collections:
- `list(<TYPE>)`
- `set(<TYPE>)`
- `map(<TYPE>)`
- `object({<ATTR NAME> = <TYPE>, ... })`
- `tuple([<TYPE>, ...])`
The keyword `any` may be used to indicate that any type is acceptable. For more
information on the meaning and behavior of these different types, as well as
detailed information about automatic conversion of complex types, see [Type
Constraints](https://www.terraform.io/docs/configuration/types.html).
If both the `type` and `default` arguments are specified, the given default
value must be convertible to the specified type.
If only `default` is specified, the type of the default value will be used.
When the `type` and `default` are both _not_ specified and you try to set a
variable [from env vars](#environment-variables) or [from the command
line](#variables-on-the-command-line), the variable will always be interpreted
as a string.
## Input Variable Documentation
Because the input variables of a build are part of its user interface, you can
briefly describe the purpose of each variable using the optional `description`
argument:
```hcl
variable "image_id" {
type = string
description = "The id of the machine image (AMI) to use for the server."
}
```
The description should concisely explain the purpose of the variable and what
kind of value is expected. This description string might be included in
documentation about the build, and so it should be written from the perspective
of the user of the build rather than its maintainer. For commentary for build
maintainers, use comments.
`@include 'from-1.5/variables/assignment.mdx'`
The following sections describe these options in more detail.
`@include 'from-1.5/variables/custom-validation.mdx'`
### Variables on the Command Line
To specify individual variables on the command line, use the `-var` option when
running the `packer build` command:
```shell-session
$ packer build -var="image_id=ami-abc123"
$ packer build -var='image_id_list=["ami-abc123","ami-def456"]'
$ packer build -var='image_id_map={"us-east-1":"ami-abc123","us-east-2":"ami-def456"}'
```
The `-var` option can be used any number of times in a single command.
If you plan to assign variables via the command line, we strongly recommend that
you at least set a default type instead of using empty blocks; this helps the
HCL parser understand what is being set. Otherwise, the interpreter will assume
that any variable set on the command line is a string.
### Variable Definitions (`.pkrvars.hcl` and `.auto.pkrvars.hcl`) Files
To set lots of variables, it is more convenient to specify their values in a
_variable definitions file_ (with a filename ending in either `.pkrvars.hcl` or
`.pkrvars.json`) and then specify that file on the command line with
`-var-file`:
```shell-session
$ packer build -var-file="testing.pkrvars.hcl"
```
A variable definitions file uses the same basic syntax as Packer language
files, but consists only of variable name assignments:
```hcl
image_id = "ami-abc123"
availability_zone_names = [
"us-east-1a",
"us-west-1c",
]
```
Packer also automatically loads a number of variable definitions files if they
are present:
- Any files with names ending in `.auto.pkrvars.hcl` or `.auto.pkrvars.json`.
Files whose names end with `.json` are parsed as JSON objects instead of HCL,
with the root object properties corresponding to variable names:
```json
{
"image_id": "ami-abc123",
"availability_zone_names": ["us-west-1a", "us-west-1c"]
}
```
### Environment Variables
As a fallback for the other ways of defining variables, Packer searches the
environment of its own process for environment variables named `PKR_VAR_`
followed by the name of a declared variable.
This can be useful when running Packer in automation, or when running a
sequence of Packer commands in succession with the same variables. For example,
at a `bash` prompt on a Unix system:
```shell-session
$ export PKR_VAR_image_id=ami-abc123
$ packer build gcp/debian/
...
```
On operating systems where environment variable names are case-sensitive,
Packer matches the variable name exactly as given in configuration, and so the
required environment variable name will usually have a mix of upper and lower
case letters as in the above example.
### Complex-typed Values
When variable values are provided in a variable definitions file, Packer's
[usual syntax](/docs/templates/hcl_templates/expressions) can be used to assign
complex-typed values, like lists and maps.
Some special rules apply to the `-var` command line option and to environment
variables. For convenience, Packer defaults to interpreting `-var` and
environment variable values as literal strings, which do not need to be quoted:
```shell-session
$ export PKR_VAR_image_id=ami-abc123
```
However, if a build variable uses a [type constraint](#type-constraints) to
require a complex value (list, set, map, object, or tuple), Packer will instead
attempt to parse its value using the same syntax used within variable
definitions files, which requires careful attention to the string escaping
rules in your shell:
```shell-session
$ export PKR_VAR_availability_zone_names='["us-west-1b","us-west-1d"]'
```
For readability, and to avoid the need to worry about shell escaping, we
recommend always setting complex variable values via variable definitions
files.
### Variable Definition Precedence
The above mechanisms for setting variables can be used together in any
combination.
Packer loads variables in the following order, with later sources taking
precedence over earlier ones:
- Environment variables (lowest priority)
- Any `*.auto.pkrvars.hcl` or `*.auto.pkrvars.json` files, processed in lexical
order of their filenames.
- Any `-var` and `-var-file` options on the command line, in the order they are
provided. (highest priority)
If the same variable is assigned multiple values using different mechanisms,
Packer uses the _last_ value it finds, overriding any previous values. Note
that the same variable cannot be assigned multiple values within a single source.
~> **Important:** Variables with map and object values behave the same way as
other variables: the last value found overrides the previous values.
`@include 'from-1.5/variables/must-be-set.mdx'`
### Setting an unknown variable will not always fail:
| Usage | packer validate | any other packer command |
| :----------------------------: | :---------------------: | :-----------------------: |
| `bar=yz` in .pkrvars.hcl file. | error, "bar undeclared" | warning, "bar undeclared" |
| `var.bar` in .pkr.hcl file | error, "bar undeclared" | error, "bar undeclared" |
| `-var bar=yz` argument | error, "bar undeclared" | error, "bar undeclared" |
| `export PKR_VAR_bar=yz` | - | - |
`@include 'from-1.5/variables/sensitive.mdx'`