326 lines
9.7 KiB
Text
326 lines
9.7 KiB
Text
= Welcome to Arion documentation
|
||
|
||
== Introduction
|
||
|
||
Arion is a tool for building and running applications that
|
||
consist of multiple docker containers using NixOS modules.
|
||
It has special support for docker images that are built with Nix,
|
||
for a smooth development experience and improved performance.
|
||
|
||
It is built on top of https://docs.docker.com/compose/overview/[Docker
|
||
Compose], which implements the container orchestration functionality.
|
||
|
||
Instead of configuring the compositions in YAML files like
|
||
`docker-compose.yaml`, Arion uses the https://nixos.org/nix/[Nix]
|
||
language to declare the compositions. Because of this, Arion gives you
|
||
the ability to declare your deployments, configuration and packaging
|
||
in the same language. By replacing multiple tools with a single
|
||
language, you decrease your mental load and you can more easily
|
||
refactor and maintain your configurations.
|
||
|
||
Although Arion can be used as a Docker Compose with an improved
|
||
configuration front end, there is more to be gained from integrating
|
||
with Nix. In particular, the more structured approach of Nix compared
|
||
to Dockerfiles allows the following:
|
||
|
||
* Build components of your image in *parallel, automatically*
|
||
* *Share packages between images*, regardless of the order they were
|
||
added
|
||
* Improve performance by *skipping container
|
||
image creation*
|
||
* Work with *structured data instead of strings*,
|
||
templates and a multitude of expression languages
|
||
* Refactor across deployments, configuration and packaging
|
||
|
||
Arion allows to compose containers with different granularity:
|
||
|
||
* <<Minimal: Plain command using nixpkgs>>
|
||
* <<NixOS: run only one systemd service>>
|
||
* <<NixOS: run full OS>>
|
||
* <<Docker image from DockerHub>>
|
||
|
||
Full NixOS is supported on
|
||
* docker-compose + podman with docker socket (NixOS >= 21.05)
|
||
* docker-compose + docker, before cgroupsv2 (NixOS < 21.05)
|
||
|
||
`podman-compose` support is currently WIP on a separate branch.
|
||
|
||
== Installation
|
||
|
||
=== Nix
|
||
|
||
```bash
|
||
$ nix-env -iA arion -f https://github.com/hercules-ci/arion/tarball/master
|
||
```
|
||
|
||
=== NixOS
|
||
|
||
Add this module to your NixOS configuration:
|
||
|
||
```nix
|
||
{ pkgs, ... }: {
|
||
environment.systemPackages = [
|
||
pkgs.arion
|
||
|
||
# Do install the docker CLI to talk to podman.
|
||
# Not needed when virtualisation.docker.enable = true;
|
||
pkgs.docker-client
|
||
];
|
||
|
||
# Arion works with Docker, but for NixOS-based containers, you need Podman
|
||
# since NixOS 21.05.
|
||
virtualisation.docker.enable = false;
|
||
virtualisation.podman.enable = true;
|
||
virtualisation.podman.dockerSocket.enable = true;
|
||
virtualisation.podman.defaultNetwork.dnsname.enable = true;
|
||
|
||
# Use your username instead of `myuser`
|
||
users.extraUsers.myuser.extraGroups = ["podman"];
|
||
}
|
||
```
|
||
|
||
////
|
||
|
||
== Not installing: use it in a project
|
||
|
||
TODO: describe: using nix-shell or in a script, building images as
|
||
part of nix-build, pinning, see also todomvc-nix.
|
||
|
||
TODO: exposed Nix functions: arion.build, arion.eval (a bit of IFD)
|
||
|
||
|
||
////
|
||
|
||
|
||
== Usage
|
||
|
||
Arion is configured declaratively with two files:
|
||
|
||
=== arion-pkgs.nix
|
||
|
||
Arion needs `arion-pkgs.nix` to import nixpkgs, for example:
|
||
|
||
```nix
|
||
import <nixpkgs> { system = "x86_64-linux"; }
|
||
```
|
||
|
||
or more sophisticated (recommended) setup with https://github.com/nmattia/niv[Niv].
|
||
|
||
=== arion-compose.nix
|
||
|
||
Describe containers using NixOS-style modules. There are a few options:
|
||
|
||
==== Minimal: Plain command using nixpkgs
|
||
|
||
`examples/minimal/arion-compose.nix`:
|
||
|
||
```nix
|
||
{ pkgs, ... }:
|
||
{
|
||
config.services = {
|
||
|
||
webserver = {
|
||
service.useHostStore = true;
|
||
service.command = [ "sh" "-c" ''
|
||
cd "$$WEB_ROOT"
|
||
${pkgs.python3}/bin/python -m http.server
|
||
'' ];
|
||
service.ports = [
|
||
"8000:8000" # host:container
|
||
];
|
||
service.environment.WEB_ROOT = "${pkgs.nix.doc}/share/doc/nix/manual";
|
||
};
|
||
};
|
||
}
|
||
```
|
||
|
||
==== NixOS: run only one systemd service
|
||
|
||
`examples/nixos-unit/arion-compose.nix`:
|
||
|
||
```nix
|
||
{
|
||
services.webserver = { config, pkgs, ... }: {
|
||
|
||
nixos.configuration = {config, pkgs, ...}: {
|
||
boot.isContainer = true;
|
||
services.nginx.enable = true;
|
||
services.nginx.virtualHosts.localhost.root = "${pkgs.nix.doc}/share/doc/nix/manual";
|
||
system.build.run-nginx = pkgs.writeScript "run-nginx" ''
|
||
#!${pkgs.bash}/bin/bash
|
||
PATH='${config.systemd.services.nginx.environment.PATH}'
|
||
echo nginx:x:${toString config.users.users.nginx.uid}:${toString config.users.groups.nginx.gid}:nginx web server user:/var/empty:/bin/sh >>/etc/passwd
|
||
echo nginx:x:${toString config.users.groups.nginx.gid}:nginx >>/etc/group
|
||
${config.systemd.services.nginx.runner}
|
||
'';
|
||
};
|
||
service.command = [ config.nixos.build.run-nginx ];
|
||
service.useHostStore = true;
|
||
service.ports = [
|
||
"8000:80" # host:container
|
||
];
|
||
};
|
||
}
|
||
```
|
||
|
||
==== NixOS: run full OS
|
||
|
||
`examples/full-nixos/arion-compose.nix`:
|
||
|
||
```nix
|
||
{
|
||
services.webserver = { pkgs, ... }: {
|
||
nixos.useSystemd = true;
|
||
nixos.configuration.boot.tmpOnTmpfs = true;
|
||
nixos.configuration.services.nginx.enable = true;
|
||
nixos.configuration.services.nginx.virtualHosts.localhost.root = "${pkgs.nix.doc}/share/doc/nix/manual";
|
||
service.useHostStore = true;
|
||
service.ports = [
|
||
"8000:80" # host:container
|
||
];
|
||
};
|
||
}
|
||
```
|
||
|
||
==== Docker image from DockerHub
|
||
|
||
```nix
|
||
{
|
||
services.postgres = {
|
||
service.image = "postgres:10";
|
||
service.volumes = [ "${toString ./.}/postgres-data:/var/lib/postgresql/data" ];
|
||
service.environment.POSTGRES_PASSWORD = "mydefaultpass";
|
||
};
|
||
}
|
||
```
|
||
|
||
=== Run
|
||
|
||
Start containers and watch their logs:
|
||
|
||
```bash
|
||
$ arion up -d
|
||
$ arion logs -f
|
||
```
|
||
|
||
You can go to `examples/*/` and run these commands to give it a quick try.
|
||
|
||
=== Inspect the config
|
||
|
||
While developing an arion project, you can make use of `arion repl`, which launches
|
||
a `nix repl` on the project configuration.
|
||
|
||
```
|
||
$ arion repl
|
||
Launching a repl for you. To get started:
|
||
|
||
To see deployment-wide configuration
|
||
type config. and use tab completion
|
||
To bring the top-level Nixpkgs attributes into scope
|
||
type :a (config._module.args.pkgs) // { inherit config; }
|
||
|
||
Welcome to Nix. Type :? for help.
|
||
|
||
Loading '../../src/nix/eval-composition.nix'...
|
||
Added 5 variables.
|
||
|
||
nix-repl> config.services.webserver.service.command
|
||
[ "sh" "-c" "cd \"$$WEB_ROOT\"\n/nix/store/66fbv9mmx1j4hrn9y06kcp73c3yb196r-python3-3.8.9/bin/python -m http.server\n" ]
|
||
|
||
nix-repl>
|
||
|
||
```
|
||
|
||
== Build with Nix
|
||
|
||
You can build a project with `nix-build` using an expression like
|
||
|
||
```nix
|
||
arion.build { modules = [ ./arion-compose.nix ]; pkgs = import ./arion-pkgs.nix; }
|
||
```
|
||
|
||
If you deploy with xref:hercules-ci-effects:ROOT:reference/nix-functions/runArion.adoc[integrate],
|
||
and your `pkgs` variable is equivalent to `import ./arion-pkgs.nix`, you can use:
|
||
|
||
```nix
|
||
let
|
||
deployment = pkgs.effects.runArion { /* ... */ });
|
||
in deployment.prebuilt
|
||
```
|
||
|
||
== Project Status
|
||
|
||
This project was born out of a process supervision need for local
|
||
development environments while working on
|
||
https://www.hercules-ci.com[Hercules CI]. (It was also born out of
|
||
ancient Greek deities disguised as horses. More on that later.)
|
||
|
||
Arion can be used for simple single host deployments, using Docker's TLS
|
||
client verification, or https://search.nixos.org/options?channel=unstable&show=virtualisation.podman.networkSocket.enable&query=virtualisation.podman[`virtualisation.podman.networkSocket` options].
|
||
Remote deployments do not support `useHostStore`, although an SSH-based deployment method could support this.
|
||
Docker Swarm is not currently supported.
|
||
|
||
Arion has run successfully on Linux distributions other than NixOS, but we only perform CI for Arion on NixOS.
|
||
|
||
|
||
== How it works
|
||
|
||
Arion is essentially a thin wrapper around Nix and docker-compose. When
|
||
it runs, it does the following:
|
||
|
||
* Evaluate the configuration using Nix, producing a
|
||
`docker-compose.yaml` and a garbage collection root
|
||
* Invoke `docker-compose`
|
||
* Clean up the garbage collection root
|
||
|
||
Most of the interesting stuff happens in Arion’s Nix expressions, where
|
||
it runs the module system (known from NixOS) and provides the
|
||
configuration that makes the Docker Compose file do the things it needs
|
||
to do.
|
||
|
||
One of the more interesting built-in modules is the
|
||
link:src/nix/modules/service/host-store.nix[host-store.nix module] which
|
||
performs the bind mounts to make the host Nix store available in the
|
||
container.
|
||
|
||
== FAQ
|
||
|
||
=== Do I need to use Hercules CI?
|
||
|
||
Nope, it’s just Nix and Docker Compose under the hood.
|
||
|
||
It does xref:hercules-ci-effects:ROOT:reference/nix-functions/runArion.adoc[integrate] nicely though.
|
||
|
||
=== What about garbage collection?
|
||
|
||
Arion removes the need for garbage collecting docker images, delegating
|
||
this task to Nix when using `service.useHostStore`.
|
||
|
||
Arion creates a garbage collection root that it cleans up after completing
|
||
the command. This means that `arion up -d` should not be used with `useHostStore`
|
||
in production. Instead, disable `useHostStore`, which will use `dockerTools` to
|
||
generate images that can be used in production.
|
||
|
||
=== Why is my container not running latest code?
|
||
|
||
Rebuild the image using `arion up -d --always-recreate-deps <name>` or simply `arion up -d`.
|
||
|
||
Like `docker-compose restart`, `arion restart` does not update the image before starting.
|
||
|
||
=== What is messing with my environment variables?
|
||
|
||
Docker Compose performs its own environment variable substitution. This
|
||
can be a little annoying in `services.command` for example. Either
|
||
reference a script from `pkgs.writeScript` or escape the dollar sign as
|
||
`$$`.
|
||
|
||
=== Why name it ``Arion``?
|
||
|
||
Arion comes from Greek mythology. Poseidon, the god of Docker -- I mean the seas --
|
||
had his eye on Demeter. Demeter tried to trick him by disguising as a
|
||
horse, but Poseidon saw through the deception and they had Arion.
|
||
|
||
So Arion is a super fast divine horse; the result of some weird mixing.
|
||
Also it talks.
|
||
|
||
(And we felt morally obliged to name our stuff after Greek mythology)
|