balena-cli/docker/DOCKER.md
Kyle Harding 9036ce9af3 docker: Improve handling of Docker-in-Docker errors
The `local` logging driver captures output from container’s stdout/stderr
and writes them to an internal storage that is optimized for performance and disk use.

We also want to capture these logs on startup to wait for success/failure.

Advise the use of `--privileged` when running Docker-in-Docker to avoid
various permissions issues encountered in testing.

Change-type: patch
Changlelog-entry: docker: Improve handling of Docker-in-Docker errors
Signed-off-by: Kyle Harding <kyle@balena.io>
2021-03-25 14:02:22 +00:00

15 KiB

Docker Images for the balena CLI

Docker images with the balena CLI and Docker-in-Docker.

Features Overview

These CLI images are based on the popular Balena base images so they include many of the features you see there.

  • Multiple Architectures:
    • rpi
    • armv7hf
    • aarch64 (debian only)
    • amd64
    • i386
  • Multiple Distributions
    • debian
    • alpine
  • cross-build functionality for building ARM containers on x86.
  • Helpful package installer script called install_packages inspired by minideb.

Note that there are some additional considerations when running the CLI via Docker so pay close attention to the Usage section for examples of different CLI commands.

Image Names

balenalib/<arch>-<distro>-balenacli:<cli_ver>

  • <arch> is the architecture and is mandatory. If using Dockerfile.template, you can replace this with %%BALENA_ARCH%%. For a list of available device names and architectures, see the Device types.
  • <distro> is the Linux distribution and is mandatory. Currently there are 2 distributions, namely debian and alpine.

Image Tags

In the tags, all of the fields are optional, and if they are left out, they will default to their latest pointer.

  • <cli_ver> is the version of the balena CLI, for example, 12.40.2, it can also be substituted for latest.

Examples

balenalib/amd64-debian-balenacli:12.40.2

  • <arch>: amd64 - suitable for running on most workstations
  • <distro>: debian - widely used base distro
  • <cli_ver>: 12.40.2

balenalib/armv7hf-alpine-balenacli

  • <arch>: armv7hf - suitable for running on a Raspberry Pi 3 for example
  • <distro>: alpine - smaller footprint than debian
  • <cli_ver>: omitted - the latest available CLI version will be used

Volumes

Volumes can be used to persist data between instances of the CLI container, or to share files between the host and the container. In most cases these are optional, but some examples will highlight when volumes are required.

  • -v "balena_data:/root/.balena": persist balena credentials and downloads between instances
  • -v "docker_data:/var/lib/docker": persist cache between instances when using Docker-in-Docker (requires -e "DOCKERD=1")
  • -v "$PWD:$PWD" -w "$PWD": bind mount your current working directory in the container to share app sources or balenaOS image files
  • -v "${SSH_AUTH_SOCK}:/ssh-agent": bind mount your host ssh-agent socket with preloaded SSH keys
  • -v "/var/run/docker.sock:/var/run/docker.sock": bind mount your host Docker socket instead of Docker-in-Docker

Environment Variables

These environment variables are available for additional functionality included in the CLI image. In most cases these are optional, but some examples will highlight when environment variables are required.

  • -e "SSH_PRIVATE_KEY=$(</path/to/priv/key)": copy your private SSH key file contents as an environment variable
  • -e "DOCKERD=1": enable the included Docker-in-Docker daemon (requires --privileged)

Keeping the CLI image up to date

Please note that using the :latest tag is not enough to keep the image up to date, because Docker will reuse a locally cached image. To update the image to the latest version, run:

$ docker pull balenalib/<arch>-<distro>-balenacli

Replacing <arch> and <distro> with the image architecture and distribution as described earlier.

If you are using Docker v19.09 or later, you can also add the --pull always flag to docker run commands, so that Docker automatically checks for available updates (new image layers will only be downloaded if a new version is available).

Usage

We've provided some examples of common CLI commands and how they are best used with this image, since some special considerations must be made.

  • login - login to balena
  • push - start a build on the remote balenaCloud build servers, or a local mode device
  • logs - show device logs
  • ssh - SSH into the host or application container of a device
  • apps - list all applications
  • app - display information about a single application
  • devices - list all devices
  • device - show info about a single device
  • tunnel - tunnel local ports to your balenaOS device
  • preload - preload an app on a disk image (or Edison zip archive)
  • build - build a project locally
  • deploy - deploy a single image or a multicontainer project to a balena application
  • join - move a local device to an application on another balena server
  • leave - remove a local device from its balena application
  • scan - scan for balenaOS devices on your local network

For each example we have also linked to the corresponding sections of the balena CLI Documentation here: https://www.balena.io/docs/reference/balena-cli

login

The balena login command can't be used with web authorization and a browser when running in a container. Instead it must be used with --token or --credentials.

Notice that here we've used a named volume balena_data to store credentials for future runs of the CLI image. This is optional but avoids having to run the login command again every time you run the image.

$ docker volume create balena_data
$ docker run --rm -it -v "balena_data:/root/.balena" balenalib/amd64-debian-balenacli /bin/bash
    
> balena login --credentials --email "johndoe@gmail.com" --password "secret"
> balena login --token "..."
> exit

push

In this example we are mounting your current working directory into the container with -v "$PWD:$PWD" -w "$PWD". This will bind mount your current working directory into the container at the same absolute path.

This bind mount is required so the CLI has access to your app sources.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -v "$PWD:$PWD" -w "$PWD" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena push myApp --source .
> balena push 10.0.0.1 --env MY_ENV_VAR=value --env my-service:SERVICE_VAR=value
> exit

logs

$ docker run --rm -it -v "balena_data:/root/.balena" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena logs 23c73a1 --service my-service
> balena logs 23c73a1.local --system --tail
> exit

ssh

The balena ssh command requires an existing SSH key added to your balenaCloud account.

One way to make this key available to the container is to pass the private key file contents as an environment variable.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -e "SSH_PRIVATE_KEY=$(</path/to/priv/key)" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena ssh f49cefd
> balena ssh f49cefd my-service
> balena ssh 192.168.0.1 --verbose
> exit

Another way to share SSH keys with the container is to mount your SSH agent socket with keys preloaded.

$ eval ssh-agent
$ ssh-add /path/to/priv/key

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -v "${SSH_AUTH_SOCK}:/ssh-agent" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena ssh f49cefd
> balena ssh f49cefd my-service
> balena ssh 192.168.0.1 --verbose
> exit

app | apps

$ docker run --rm -it -v "balena_data:/root/.balena" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena apps
> balena app myorg/myapp
> exit

device | devices

$ docker run --rm -it -v "balena_data:/root/.balena" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena devices --application MyApp
> balena device 7cf02a6
> exit

tunnel

The balena tunnel command is easiest used when the host networking stack can be shared with the container and ports can be easily assigned.

However the host networking driver only works on Linux hosts, and is not supported on Docker Desktop for Mac, Docker Desktop for Windows, or Docker EE for Windows Server.

Instead you can bind specific port ranges to the host so you can access the tunnel from outside the container via localhost:[localPort].

Note that when exposing individual ports, you must specify all interfaces in the format [remotePort]:0.0.0.0:[localPort] otherwise the tunnel will only be listening for connections within the container.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -p 22222:22222 \
    -p 12345:54321
    balenalib/amd64-debian-balenacli /bin/bash

> balena tunnel 2ead211 -p 22222:0.0.0.0
> balena tunnel myApp -p 54321:0.0.0.0:12345
> exit

If you have host networking available then you do not need to specify ports in your run command, and the interface 0.0.0.0 is optional in your tunnel command.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    --network host \
    balenalib/amd64-debian-balenacli /bin/bash

> balena tunnel 2ead211 -p 22222
> balena tunnel myApp -p 54321:12345
> exit

preload

The balena preload command requires access to a Docker client and daemon.

The easiest way to run this command is to use the included Docker-in-Docker daemon.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -v "docker_data:/var/lib/docker" \
    -e "DOCKERD=1" --privileged \
    balenalib/amd64-debian-balenacli /bin/bash

> balena os download raspberrypi3 -o raspberry-pi.img
> balena os configure raspberry-pi.img --app MyApp
> balena preload raspberry-pi.img --app MyApp --commit current
> exit

Another way to run the preload command is to use the host OS Docker socket and avoid starting a Docker daemon in the container. This is achieved with -v "/var/run/docker.sock:/var/run/docker.sock".

In this example we are mounting your current working directory into the container with -v "$PWD:$PWD" -w "$PWD". This will bind mount your current working directory into the container at the same absolute path.

This bind mount is required when using the host Docker socket because the absolute path to the balenaOS image file must be the same from both the perspective of the CLI in the container and the host Docker socket.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -v "/var/run/docker.sock:/var/run/docker.sock" \
    -v "$PWD:$PWD" -w "$PWD" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena os download raspberrypi3 -o raspberry-pi.img
> balena os configure raspberry-pi.img --app MyApp
> balena preload raspberry-pi.img --app MyApp --commit current
> exit

build | deploy

The build and deploy commands both require access to a Docker client and daemon.

The easiest way to run these commands is to use the included Docker-in-Docker daemon.

In this example we are mounting your current working directory into the container with -v "$PWD:$PWD" -w "$PWD". This will bind mount your current working directory into the container at the same absolute path.

This bind mount is required so the CLI has access to your app sources.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -v "docker_data:/var/lib/docker" \
    -e DOCKERD=1 --privileged \
    -v "$PWD:$PWD" -w "$PWD" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena build --application myApp
> balena deploy myApp
> exit

Another way to run the build and deploy commands is to use the host OS Docker socket and avoid starting a Docker daemon in the container. This is achieved with -v "/var/run/docker.sock:/var/run/docker.sock".

In this example we are mounting your current working directory into the container with -v "$PWD:$PWD" -w "$PWD". This will bind mount your current working directory into the container at the same absolute path.

This bind mount is required so the CLI has access to your app sources.

$ docker run --rm -it -v "balena_data:/root/.balena" \
    -v "/var/run/docker.sock:/var/run/docker.sock" \
    -v "$PWD:$PWD" -w "$PWD" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena build --application myApp
> balena deploy myApp
> exit

join | leave

$ docker run --rm -it -v "balena_data:/root/.balena" \
    balenalib/amd64-debian-balenacli /bin/bash

> balena join balena.local --application MyApp
> balena leave balena.local
> exit

scan

The balena scan command requires access to the host network interface in order to bind and listen for multicast responses from devices.

However the host networking driver only works on Linux hosts, and is not supported on Docker Desktop for Mac, Docker Desktop for Windows, or Docker EE for Windows Server.

$ docker run --rm -it --network host balenalib/amd64-debian-balenacli scan

Custom images / contributing

The following steps may be used to create custom CLI images or to contribute bug reports, fixes or features.

# the currently supported base images are 'debian' and 'alpine'
export BALENA_DISTRO="debian"

# provide the architecture where you will be testing the image
export BALENA_ARCH="amd64"

# optionally register QEMU binfmt if building for other architectures (eg. armv7hf)
$ docker run --rm --privileged multiarch/qemu-user-static --reset -p yes

# build and tag an image with docker
docker build . -f docker/${BALENA_DISTRO}/Dockerfile \
    --build-arg "BUILD_BASE=balenalib/${BALENA_ARCH}-${BALENA_DISTRO}-node:12.19.1-build" \
    --build-arg "RUN_BASE=balenalib/${BALENA_ARCH}-${BALENA_DISTRO}-node:12.19.1-run" \
    --tag "balenalib/${BALENA_ARCH}-${BALENA_DISTRO}-balenacli"