# LinuxKit packages A LinuxKit package is a container image which can be used to assemble a bootable Linux image. The LinuxKit project has a number of [core packages](../pkg), but users can create their own packages, as it's very easy. Packages are the unit of customisation in a LinuxKit-based project, if you know how to build a container, you should be able to build a LinuxKit package. All LinuxKit packages are: - Signed with Docker Content Trust. - Enabled with multi-arch manifests to work on multiple architectures. - Derived from well-known (and signed) sources for repeatable builds. - Built with multi-stage builds to minimise their size. ## CI and Package Builds When building and merging packages, it is important to note that our CI process builds packages. The targets `make ci` and `make ci-pr` execute `make -C pkg build`. These in turn execute `linuxkit pkg build` for each package under `pkg/`. This in turn will try to pull the image whose tag matches the tree hash or, failing that, to build it. We do not want the builds to happen with each CI run for two reasons: 1. It is slower to do a package build than to just pull the latest image. 2. If any of the steps of the build fails, e.g. a `curl` download that depends on an intermittent target, it can cause all of CI to fail. Thus, if, as a maintainer, you merge any commits into a `pkg/`, even if the change is documentation alone, please do a `linuxkit package push`. ## Package source A package source consists of a directory containing at least two files: - `build.yml`: contains metadata associated with the package - `Dockerfile`: contains the steps to build the package. `build.yml` contains the following fields: - `image` _(string)_: *(mandatory)* The name of the image to build - `org` _(string)_: The hub/registry organisation to which this package belongs - `arches` _(list of string)_: The architectures which this package should be built for (valid entries are `GOARCH` names) - `extra-sources` _(list of strings)_: Additional sources for the package outside the package directory. The format is `src:dst`, where `src` can be relative to the package directory and `dst` is the destination in the build context. This is useful for sharing files, such as vendored go code, between packages. - `gitrepo` _(string)_: The git repository where the package source is kept. - `network` _(bool)_: Allow network access during the package build (default: no) - `disable-content-trust` _(bool)_: Disable Docker content trust for this package (default: no) - `disable-cache` _(bool)_: Disable build cache for this package (default: no) - `config`: _(struct `github.com/moby/tool/src/moby.ImageConfig`)_: Image configuration, marshalled to JSON and added as `org.mobyproject.config` label on image (default: no label) - `depends`: Contains information on prerequisites which must be satisfied in order to build the package. Has subfields: - `docker-images`: Docker images to be made available (as `tar` files via `docker image save`) within the package build context. Contains the following nested fields: - `from-file` and `list`: _(string and string list respectively)_. Mutually exclusive fields specifying the list of images to include. Each image must include a valid digest (`sha256:...`) in order to maintain determinism. If `from-file` is used then it is a path relative to (and within) the package directory with one image per line (lines with `#` in column 0 and blank lines are ignore). If `list` is used then each entry is an image. - `target` and `target-dir`: _(string)_ Mutually exclusive fields specifying the target location, if `target` is used then it is a path relative to (and within) the package dir which names a `tar` file into which all of the listed images will be saved. If `target-dir` then it is a path relative to (and within) the package directory which names a directory into which each image will be saved (as `«image name»@«digest».tar`). **NB**: The path referenced by `target-dir` will be _removed_ prior to populating (to avoid issues with stale files). ## Building packages ### Prerequisites Before you can build packages you need: - Docker version 17.06 or newer. If you are on a Mac you also need `docker-credential-osxkeychain.bin`, which comes with Docker for Mac. - `make`, `notary`, `base64`, `jq`, and `expect` - A *recent* version of `manifest-tool` which you can build with `make bin/manifest-tool`, or `go get github.com:estesp/manifest-tool`, or via the LinuxKit homebrew tap with `brew install --HEAD manifest-tool`. `manifest-tool` must be in your path. - The LinuxKit tool `linuxkit` which must be in your path. Further, when building packages you need to be logged into hub with `docker login` as some of the tooling extracts your hub credentials during the build. ### Build packages as a maintainer If you have write access to the `linuxkit` organisation on hub, you should also be set up with signing keys for packages and your signing key should have a passphrase, which we call `` throughout. All official LinuxKit packages are multi-arch manifests and most of them are available for `amd64`, `arm64`, and `s390x`. Official images *must* be build on both architectures and they must be build *in sequence*, i.e., they can't be build in parallel. To build a package on an architecture: ``` DOCKER_CONTENT_TRUST_REPOSITORY_PASSPHRASE="" linuxkit pkg push «path-to-package» ``` `«path-to-package»` is the path to the package's source directory (containing at least `build.yml` and `Dockerfile`). It can be `.` if the package is in the current directory. **Note:** You *must* be logged into hub (`docker login`) and the passphrase for the key *must* be supplied as an environment variable. The build process has to resort to using `expect` to drive `notary` so none of the credentials can be entered interactively. This will: - Build a local images as `linuxkit/:-` - Push it to hub - Sign it with your key - Create a manifest called `linuxkit/:` (note no `-`) - Push the manifest to hub - Sign the manifest If you repeat the same on another architecture, a new manifest will be pushed and signed containing the previous and the new architecture. The YAML files should consume the package as: `linuxkit/:`. Since it is not very good to have your passphrase in the clear (or even stashed in your shell history), we recommend using a password manager with a CLI interface, such as LastPass or `pass`. You can then invoke the build like this (for LastPass): ``` DOCKER_CONTENT_TRUST_REPOSITORY_PASSPHRASE=$(lpass show --password) linuxkit pkg push «path-to-package» ``` or alternatively you may add the command to `~/.moby/linuxkit/config.yml` e.g.: ``` pkg: content-trust-passphrase-command: "lpass show --password" ``` ### Build packages as a developer If you want to develop packages or test them locally, it is best to override the hub organisation used. You may also want to disable signing while developing. A typical example would be: ``` linuxkit pkg build -org=wombat -disable-content-trust «path-to-package» ``` This will create a local image: `wombat/:-` which you can use in your local YAML files for testing. If you need to test on other systems you can push the image to your hub account and pull from a different system by issuing: ``` linuxkit pkg build -org=wombat -disable-content-trust push ``` This will push both `wombat/:-` and `wombat/:` to hub. Finally, if you are tired of the long hashes you can override the hash with: ``` linuxkit pkg build -org=wombat -disable-content-trust -hash=foo push ``` and this will create `wombat/:foo-` and `wombat/:foo` for use in your YAML files.