#container #yaml-config #module #reproducible #declarative #definition #dockerfile

app containeryard

Container Yard is a declarative, reproducible, and reusable decentralized approach for defining containers. Think Nix flakes meets Containerfiles (aka Dockerfiles).

12 releases

0.3.2 Dec 14, 2024
0.3.1 Dec 14, 2024
0.2.7 Oct 27, 2024
0.2.4 Jul 29, 2024
0.1.0 Jul 17, 2024

#59 in Configuration

Download history 2/week @ 2024-10-12 343/week @ 2024-10-26 43/week @ 2024-11-02 3/week @ 2024-11-09 1/week @ 2024-11-16 5/week @ 2024-12-07 367/week @ 2024-12-14 5/week @ 2024-12-21

377 downloads per month

Apache-2.0

68KB
1.5K SLoC

ContainerYard

github crates.io docs.rs

ContainerYard is a declarative, reproducible, and reusable decentralized approach for defining containers. See Why Use ContainerYard for motivation.

ContainerYard breaks a containers definition into modules and composes them with a yard file.

Yard File

A yard file (yard.yaml) composes modules and outputs one or more Containerfiles (aka Dockerfiles). E.g.

# yaml-language-server: $schema=https://raw.githubusercontent.com/mcmah309/containeryard/master/src/schemas/yard-schema.json

inputs:
  # Modules found on local paths
  modules:
    finalizer: local_modules/finalizer.md
  # Modules found in a remote repos
  remotes:
    - url: https://github.com/mcmah309/yard_module_repository
      commit: 59e4aa77ee7e1c40adba40a7ab10e6b4fb9b8420
      modules:
        base: bases/ubuntu/lts.md
        git_config: dependent/git/git_config.md
        bash_flavor: apt/bash_interactive/flavors/mcmah309/mcmah309.md

outputs:
  # Output Containerfile created from modules
  Containerfile:
    # Module "base" from inputs
    - base:
         # Inputs, shell commands `$(..)` and ENV vars `$..` also supported
        version: "24.04"
    # Inline module
    - RUN apt install git
    - git_config:
        user_name: $(git config --get user.name)
        email: $(git config --get user.email)
    - bash_flavor:
    - finalizer:

hooks:
  build:
    # Command executed before the build. Will reload this file after the command is executed
    pre: yard update
    post: podman build . -t git

Simply running yard build in the above case, will output a single Containerfile to your current directory. See more yard.yaml examples here.

Modules

Modules represent specific features of a container. e.g. The rust module defines rust's installation. Modules can be easily reused, improved, and version controlled.

Module Parts

A module can have two parts, a Containerfile (aka Dockerfile) component and an optional configuration component.

Containerfile

Containerfile defines the core of the module. E.g.

# Note: `version` is defined in the configuration in the next section
FROM alpine:{{ version | default (value="latest") }}

RUN apk update \
    && apk upgrade \
    && apk add --no-cache ca-certificates \
    && update-ca-certificates

This file is first treated as a Tera template, then compiled. The result is a pure Containerfile component that can be combined with other modules.

Configuration

The configuration component is a yaml block and provides metadata for what the Containerfile component needs. E.g.

# yaml-language-server: $schema=https://raw.githubusercontent.com/mcmah309/containeryard/master/src/schemas/yard-module-schema.json

description: "This is a modules description"
args:
  required:
  optional:
    - version
# Files to be pulled in with this module
required_files:

yard.yaml provides the values for args: declared in a this block. e.g.

inputs:
  modules:
    module: path/to/module

outputs:
  Containerfile:
    - module:
        version: "3.20.0"

Putting It All Together

Combining the examples from the Module Parts section, the output of yard build would be

FROM alpine:3.20.0

RUN apk update \
    && apk upgrade \
    && apk add --no-cache ca-certificates \
    && update-ca-certificates

For more module examples click here.

Module Layout

A module consists of one file with one to two parts - a Containerfile section and an optional config section.


```dockerfile

# Dockerfile Statements Here

```

```yaml

# Configuration here

```


Click here for an example. Alternatively the yaml configuration block can be omitted. Or if both the yaml and dockerfile/containerfile blocks are omitted, then the file is just interpreted as a regular Containerfile without any configuration.

Installation

Note: yard is the cli tool for ContainerYard.

Debian - Ubuntu, Linux Mint, Pop!_OS, etc.

release_ver=<INSERT_CURRENT_VERSION> # e.g. release_ver='v0.2.7'
deb_file="containeryard_$(echo $release_ver | sed 's/^v//')-1_amd64.deb"
curl -LO https://github.com/mcmah309/containeryard/releases/download/$release_ver/$deb_file
dpkg -i "$deb_file"

Cargo

cargo install containeryard

Consider adding --profile dist for a longer compile time but a more optimal build.

FAQ

Why Use ContainerYard?

Developers constantly rewrite the same Containerfile/Dockerfile configs. Besides taking away developer time, these configs become hard to maintain/upgrade and adding new features feels like starting from scratch again. With ContainerYard, you can write your config once and easily reuse and incrementally improve it over time. Users can then import these various modules with little to no configuration. Want Rust? Just add it to your yard.yaml file. Want Flutter? Do the same. Need the latest version? Easily upgrade with yard update or just modify the commit line. With ContainerYard you should never have to define certain Containerfile configs again. But if you do want to do something custom, ContainerYard does not get in your way, everything is Containerfile based and the output is a pure Containerfile. No need to learn a complex tool, no need to re-invent the wheel, Containerfiles and Tera templates are powerful enough. Just let ContainerYard be the glue.

Why Use ContainerYard Over Nix Flakes?

ContainerYard is heavily inspired by Nix flakes. In fact, ContainerYard can be thought of as Nix flakes meets Containerfiles (aka Dockerfiles).

Nix flakes guarantees reproducibility at the cost of developer flexibility. ContainerYard is decentralized, allowing users to easily use different package managers and upstreams. As such, ContainerYard sacrifices some reproducibility guarantees and gains complete developer flexibility.

ContainerYard is also extremely simple and built on familiar developer tools - Containerfiles and Tera templates.

Contributing

Feel free to open an issue with any suggestions/ideas/bugs you may have and/or create PR's.

ContainerYard builds and uses its own dev container :D see here. Open the project in vscode, click the "open in container" button and you are ready to go! Otherwise just use the provided Containerfile or your own local setup.

Module Repositories

*Feel free to create a PR to add your own!*

Dependencies

~20–32MB
~490K SLoC