blob: 33cbd5950fbb4dd99b18967f208127e7a2bf2fb0 [file] [log] [blame] [view]
# Docker support for Bazel
<div class="toc">
<h2>Rules</h2>
<ul>
<li><a href="#docker_build">docker_build</a></li>
</ul>
</div>
## Overview
These build rules are used for building [Docker](https://www.docker.com)
images. Such images are easy to modify and deploy system image for
deploying application easily on cloud providers.
As traditional Dockerfile-based `docker build`s effectively execute a series
of commands inside of Docker containers, saving the intermediate results as
layers; this approach is unsuitable for use in Bazel for a variety of reasons.
The docker_build rule constructs a tarball that is compatible with
`docker save/load`, and creates a single layer out of each BUILD rule in the chain.
* [Basic Example](#basic-example)
* [Build Rule Reference](#reference)
* [Future work](#future)
<a name="basic-example"></a>
## Basic Example
Consider the following BUILD file in `//third_party/debian`:
```python
load("@bazel_tools//tools/build_defs/docker:docker.bzl", "docker_build")
filegroup(
name = "ca_certificates",
srcs = ["ca_certificates.deb"],
)
# Example when you have all your dependencies in your repository.
# We have an example on how to fetch them from the web later in this
# document.
filegroup(
name = "openjdk-7-jre-headless",
srcs = ["openjdk-7-jre-headless.deb"],
)
docker_build(
name = "wheezy",
tars = ["wheezy.tar"],
)
```
The `wheezy` target in that BUILD file roughly corresponds to the Dockerfile:
```docker
FROM scratch
ADD wheezy.tar /
```
You can then build up subsequent layers via:
```python
docker_build(
name = "base",
base = "//third_party/debian:wheezy",
debs = ["//third_party/debian:ca_certificates"],
)
docker_build(
name = "java",
base = ":base",
debs = ["//third_party/debian:openjdk-7-jre-headless"],
)
```
## Metadata
You can set layer metadata on these same rules by simply adding (supported) arguments to the rule, for instance:
```python
docker_build(
name = "my-layer",
entrypoint = ["foo", "bar", "baz"],
...
)
```
Will have a similar effect as the Dockerfile construct:
```docker
ENTRYPOINT ["foo", "bar", "baz"]
```
For the set of supported metadata, and ways to construct layers, see here.
### Using
Suppose you have a `docker_build` target `//my/image:helloworld`:
```python
docker_build(
name = "helloworld",
...
)
```
You can build this with `bazel build my/image:helloworld.tar`.
This will produce the file `bazel-genfiles/my/image/helloworld.tar`.
You can load this into my local Docker client by running
`docker load -i bazel-genfiles/my/image/helloworld.tar`, or simply
`bazel run my/image:helloworld` (this last command only update the
changed layers and thus is faster).
Upon success you should be able to run `docker images` and see:
```
REPOSITORY TAG IMAGE ID ...
bazel/my_image helloworld d3440d7f2bde ...
```
You can now use this docker image with the name `bazel/my_image:helloworld` or
tag it with another name, for example:
`docker tag bazel/my_image:helloworld gcr.io/my-project/my-awesome-image:v0.9`
You can do all that at once with specifying the tag on the command line of
`bazel run`:
```
bazel run my/image:helloworld gcr.io/my-project/my-awesome-image:v0.9
```
__Nota Bene:__ the `docker images` command will show a really old timestamp
because `docker_build` remove all timestamps from the build to make it
reproducible.
## Pulling images and deb files from the internet
If you do not want to check in base image in your repository, you can use
[external repositories](http://bazel.io/docs/external.html). For instance,
you could create various layer with `external` labels:
```python
load("@bazel_tools//tools/build_defs/docker:docker.bzl", "docker_build")
docker_build(
name = "java",
base = "@docker-debian//:wheezy",
debs = ["@openjdk-7-jre-headless//file"],
)
```
Using the WORKSPACE file to add the actual files:
```python
new_http_archive(
name = "docker-debian",
url = "https://codeload.github.com/tianon/docker-brew-debian/zip/e9bafb113f432c48c7e86c616424cb4b2f2c7a51",
build_file = "debian.BUILD",
type = "zip",
sha256 = "515d385777643ef184729375bc5cb996134b3c1dc15c53acf104749b37334f68",
)
http_file(
name = "openjdk-7-jre-headless",
url = "http://security.debian.org/debian-security/pool/updates/main/o/openjdk-7/openjdk-7-jre-headless_7u79-2.5.5-1~deb7u1_amd64.deb",
sha256 = "b632f0864450161d475c012dcfcc37a1243d9ebf7ff9d6292150955616d71c23",
)
```
With the following `debian.BUILD` file:
```python
load("@bazel_tools//tools/build_defs/docker:docker.bzl", "docker_build")
# Extract .xz files
genrule(
name = "wheezy_tar",
srcs = ["docker-brew-debian-e9bafb113f432c48c7e86c616424cb4b2f2c7a51/wheezy/rootfs.tar.xz"],
outs = ["wheezy_tar.tar"],
cmd = "cat $< | xzcat >$@",
)
docker_build(
name = "wheezy",
tars = [":wheezy_tar"],
visibility = ["//visibility:public"],
)
```
<a name="future"></a>
## Future work
In the future, we would like to provide better integration with docker
repositories: pull and push docker image.
<a name="docker_build"></a>
## docker_build
```python
docker_build(name, base, data_path, directory, files, mode, tars, debs, symlinks, entrypoint, cmd, env, labels, ports, volumes, workdir, repository)
```
<table class="table table-condensed table-bordered table-implicit">
<colgroup>
<col class="col-param" />
<col class="param-description" />
</colgroup>
<thead>
<tr>
<th colspan="2">Implicit output targets</th>
</tr>
</thead>
<tbody>
<tr>
<td><code><i>name</i>.tar</code></td>
<td>
<code>The full Docker image</code>
<p>
A full Docker image containing all the layers, identical to
what <code>docker save</code> would return. This is
only generated on demand.
</p>
</td>
</tr>
<tr>
<td><code><i>name</i>-layer.tar</code></td>
<td>
<code>An image of the current layer</code>
<p>
A Docker image containing only the layer corresponding to
that target. It is used for incremental loading of the layer.
</p>
<p>
<b>Note:</b> this target is not suitable for direct comsumption.
It is used for incremental loading and non-docker rules should
depends on the docker image (<i>name</i>.tar) instead.
</p>
</td>
</tr>
<tr>
<td><code><i>name</i></code></td>
<td>
<code>Incremental image loader</code>
<p>
The incremental image loader. It will load only changed
layers inside the Docker registry.
</p>
</td>
</tr>
</tbody>
</table>
<table class="table table-condensed table-bordered table-params">
<colgroup>
<col class="col-param" />
<col class="param-description" />
</colgroup>
<thead>
<tr>
<th colspan="2">Attributes</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>name</code></td>
<td>
<code>Name, required</code>
<p>A unique name for this rule.</p>
</td>
</tr>
<tr>
<td><code>base</code></td>
<td>
<code>File, optional</code>
<p>
The base layers on top of which to overlay this layer, equivalent to
FROM.
</p>
</td>
</tr>
<tr>
<td><code>data_path</code></td>
<td>
<code>String, optional</code>
<p>Root path of the files.</p>
<p>
The directory structure from the files is preserved inside the
docker image but a prefix path determined by `data_path`
is removed from the directory structure. This path can
be absolute from the workspace root if starting with a `/` or
relative to the rule's directory. A relative path may starts with "./"
(or be ".") but cannot use go up with "..". By default, the
`data_path` attribute is unused and all files are supposed to have no
prefix.
</p>
</td>
</tr>
<tr>
<td><code>directory</code></td>
<td>
<code>String, optional</code>
<p>Target directory.</p>
<p>
The directory in which to expand the specified files, defaulting to '/'.
Only makes sense accompanying one of files/tars/debs.
</p>
</td>
</tr>
<tr>
<td><code>files</code></td>
<td>
<code>List of files, optional</code>
<p>File to add to the layer.</p>
<p>
A list of files that should be included in the docker image.
</p>
</td>
</tr>
<tr>
<td><code>mode</code></td>
<td>
<code>String, default to 0555</code>
<p>
Set the mode of files added by the <code>files</code> attribute.
</p>
</td>
</tr>
<tr>
<td><code>tars</code></td>
<td>
<code>List of files, optional</code>
<p>Tar file to extract in the layer.</p>
<p>
A list of tar files whose content should be in the docker image.
</p>
</td>
</tr>
<tr>
<td><code>debs</code></td>
<td>
<code>List of files, optional</code>
<p>Debian package to install.</p>
<p>
A list of debian packages that will be installed in the docker image.
</p>
</td>
</tr>
<tr>
<td><code>symlinks</code></td>
<td>
<code>Dictionary, optional</code>
<p>Symlinks to create in the docker image.</p>
<p>
<code>
symlinks = {
"/path/to/link": "/path/to/target",
...
},
</code>
</p>
</td>
</tr>
<tr>
<td><code>entrypoint</code></td>
<td>
<code>String or string list, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#entrypoint">List
of entrypoints to add in the layer.</a></p>
</td>
</tr>
<tr>
<td><code>cmd</code></td>
<td>
<code>String or string list, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#cmd">List
of commands to execute in the layer.</a></p>
</td>
</tr>
<tr>
<td><code>env</code></td>
<td>
<code>Dictionary from strings to strings, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#env">Dictionary
from environment variable names to their values when running the
docker image.</a></p>
<p>
<code>
env = {
"FOO": "bar",
...
},
</code>
</p>
</td>
</tr>
<tr>
<td><code>labels</code></td>
<td>
<code>Dictionary from strings to strings, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#label">Dictionary
from custom metadata names to their values. You can also put a
file name prefixed by '@' as a value. Then the value is replaced
with the contents of the file.
<p>
<code>
labels = {
"com.example.foo": "bar",
"com.example.baz": "@metadata.json",
...
},
</code>
</p>
</td>
</tr>
<tr>
<td><code>ports</code></td>
<td>
<code>String list, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#expose">List
of ports to expose.</a></p>
</td>
</tr>
<tr>
<td><code>volumes</code></td>
<td>
<code>String list, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#volumes">List
of volumes to mount.</a></p>
</td>
</tr>
<tr>
<td><code>workdir</code></td>
<td>
<code>String, optional</code>
<p><a href="https://docs.docker.com/reference/builder/#workdir">Initial
working directory when running the docker image.</a></p>
<p>Because building the image never happen inside a docker container,
this working directory does not affect the other actions (e.g.,
adding files).</p>
</td>
</tr>
<tr>
<td><code>repository</code></td>
<td>
<code>String, default to `bazel`</code>
<p>The repository for the default tag for the image.</a></p>
<p>Image generated by `docker_build` are tagged by default to
`bazel/package_name:target` for a `docker_build` target at
`//package/name:target`. Setting this attribute to
`gcr.io/dummy` would set the default tag to
`gcr.io/dummy/package_name:target`.</p>
</td>
</tr>
</tbody>
</table>