|
|
|
---
|
|
|
|
title: "HCL variables and functions"
|
|
|
|
keywords: build, buildx, bake, buildkit, hcl
|
|
|
|
---
|
|
|
|
|
|
|
|
Similar to how Terraform provides a way to [define variables](https://www.terraform.io/docs/configuration/variables.html#declaring-an-input-variable),
|
|
|
|
the HCL file format also supports variable block definitions. These can be used
|
|
|
|
to define variables with values provided by the current environment, or a
|
|
|
|
default value when unset.
|
|
|
|
|
|
|
|
A [set of generally useful functions](https://github.com/docker/buildx/blob/master/bake/hclparser/stdlib.go)
|
|
|
|
provided by [go-cty](https://github.com/zclconf/go-cty/tree/main/cty/function/stdlib)
|
|
|
|
are available for use in HCL files. In addition, [user defined functions](https://github.com/hashicorp/hcl/tree/main/ext/userfunc)
|
|
|
|
are also supported.
|
|
|
|
|
|
|
|
## Using interpolation to tag an image with the git sha
|
|
|
|
|
|
|
|
As shown in the [File definition](file-definition.md#variable) page, `bake`
|
|
|
|
supports variable blocks which are assigned to matching environment variables
|
|
|
|
or default values:
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake.hcl
|
|
|
|
variable "TAG" {
|
|
|
|
default = "latest"
|
|
|
|
}
|
|
|
|
|
|
|
|
group "default" {
|
|
|
|
targets = ["webapp"]
|
|
|
|
}
|
|
|
|
|
|
|
|
target "webapp" {
|
|
|
|
tags = ["docker.io/username/webapp:${TAG}"]
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
alternatively, in json format:
|
|
|
|
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"variable": {
|
|
|
|
"TAG": {
|
|
|
|
"default": "latest"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": ["webapp"]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"tags": ["docker.io/username/webapp:${TAG}"]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake --print webapp
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"webapp"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"tags": [
|
|
|
|
"docker.io/username/webapp:latest"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ TAG=$(git rev-parse --short HEAD) docker buildx bake --print webapp
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"webapp"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"tags": [
|
|
|
|
"docker.io/username/webapp:985e9e9"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Using the `add` function
|
|
|
|
|
|
|
|
You can use [`go-cty` stdlib functions](https://github.com/zclconf/go-cty/tree/main/cty/function/stdlib).
|
|
|
|
Here we are using the `add` function.
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake.hcl
|
|
|
|
variable "TAG" {
|
|
|
|
default = "latest"
|
|
|
|
}
|
|
|
|
|
|
|
|
group "default" {
|
|
|
|
targets = ["webapp"]
|
|
|
|
}
|
|
|
|
|
|
|
|
target "webapp" {
|
|
|
|
args = {
|
|
|
|
buildno = "${add(123, 1)}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake --print webapp
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"webapp"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"args": {
|
|
|
|
"buildno": "124"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Defining an `increment` function
|
|
|
|
|
|
|
|
It also supports [user defined functions](https://github.com/hashicorp/hcl/tree/main/ext/userfunc).
|
|
|
|
The following example defines a simple an `increment` function.
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake.hcl
|
|
|
|
function "increment" {
|
|
|
|
params = [number]
|
|
|
|
result = number + 1
|
|
|
|
}
|
|
|
|
|
|
|
|
group "default" {
|
|
|
|
targets = ["webapp"]
|
|
|
|
}
|
|
|
|
|
|
|
|
target "webapp" {
|
|
|
|
args = {
|
|
|
|
buildno = "${increment(123)}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake --print webapp
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"webapp"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"args": {
|
|
|
|
"buildno": "124"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Only adding tags if a variable is not empty using an `notequal`
|
|
|
|
|
|
|
|
Here we are using the conditional `notequal` function which is just for
|
|
|
|
symmetry with the `equal` one.
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake.hcl
|
|
|
|
variable "TAG" {default="" }
|
|
|
|
|
|
|
|
group "default" {
|
|
|
|
targets = [
|
|
|
|
"webapp",
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
target "webapp" {
|
|
|
|
context="."
|
|
|
|
dockerfile="Dockerfile"
|
|
|
|
tags = [
|
|
|
|
"my-image:latest",
|
|
|
|
notequal("",TAG) ? "my-image:${TAG}": "",
|
|
|
|
]
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake --print webapp
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"webapp"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"tags": [
|
|
|
|
"my-image:latest"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Using variables in functions
|
|
|
|
|
|
|
|
You can refer variables to other variables like the target blocks can. Stdlib
|
|
|
|
functions can also be called but user functions can't at the moment.
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake.hcl
|
|
|
|
variable "REPO" {
|
|
|
|
default = "user/repo"
|
|
|
|
}
|
|
|
|
|
|
|
|
function "tag" {
|
|
|
|
params = [tag]
|
|
|
|
result = ["${REPO}:${tag}"]
|
|
|
|
}
|
|
|
|
|
|
|
|
target "webapp" {
|
|
|
|
tags = tag("v1")
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake --print webapp
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"webapp"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"webapp": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"tags": [
|
|
|
|
"user/repo:v1"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Using variables in variables across files
|
|
|
|
|
|
|
|
When multiple files are specified, one file can use variables defined in
|
|
|
|
another file.
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake1.hcl
|
|
|
|
variable "FOO" {
|
|
|
|
default = upper("${BASE}def")
|
|
|
|
}
|
|
|
|
|
|
|
|
variable "BAR" {
|
|
|
|
default = "-${FOO}-"
|
|
|
|
}
|
|
|
|
|
|
|
|
target "app" {
|
|
|
|
args = {
|
|
|
|
v1 = "pre-${BAR}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake2.hcl
|
|
|
|
variable "BASE" {
|
|
|
|
default = "abc"
|
|
|
|
}
|
|
|
|
|
|
|
|
target "app" {
|
|
|
|
args = {
|
|
|
|
v2 = "${FOO}-post"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake -f docker-bake1.hcl -f docker-bake2.hcl --print app
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"app"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"app": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"args": {
|
|
|
|
"v1": "pre--ABCDEF-",
|
|
|
|
"v2": "ABCDEF-post"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Using typed variables
|
|
|
|
|
|
|
|
Non-string variables are also accepted. The value passed with env is parsed
|
|
|
|
into suitable type first.
|
|
|
|
|
|
|
|
```hcl
|
|
|
|
# docker-bake.hcl
|
|
|
|
variable "FOO" {
|
|
|
|
default = 3
|
|
|
|
}
|
|
|
|
|
|
|
|
variable "IS_FOO" {
|
|
|
|
default = true
|
|
|
|
}
|
|
|
|
|
|
|
|
target "app" {
|
|
|
|
args = {
|
|
|
|
v1 = FOO > 5 ? "higher" : "lower"
|
|
|
|
v2 = IS_FOO ? "yes" : "no"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
```console
|
|
|
|
$ docker buildx bake --print app
|
|
|
|
```
|
|
|
|
```json
|
|
|
|
{
|
|
|
|
"group": {
|
|
|
|
"default": {
|
|
|
|
"targets": [
|
|
|
|
"app"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"target": {
|
|
|
|
"app": {
|
|
|
|
"context": ".",
|
|
|
|
"dockerfile": "Dockerfile",
|
|
|
|
"args": {
|
|
|
|
"v1": "lower",
|
|
|
|
"v2": "yes"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
## Built-in variables
|
|
|
|
|
|
|
|
* `BAKE_CMD_CONTEXT` can be used to access the main `context` for bake command
|
|
|
|
from a bake file that has been [imported remotely](file-definition.md#remote-definition).
|
|
|
|
* `BAKE_LOCAL_PLATFORM` returns the current platform's default platform
|
|
|
|
specification (e.g. `linux/amd64`).
|