cargo-chef

Cache the dependencies of your Rust project and speed up your Docker builds.


Crates.io version Download


Table of Contents

  1. How to install
  2. How to use
  3. Benefits vs Limitations
  4. License

How To Install

You can install cargo-chef from crates.io with

bash cargo install cargo-chef

How to use

:warning: cargo-chef is not meant to be run locally
Its primary use-case is to speed up container builds by running BEFORE the actual source code is copied over. Don't run it on existing codebases to avoid having files being overwritten.

cargo-chef exposes two commands: prepare and cook:

bash cargo chef --help

```text

cargo-chef

USAGE: cargo chef

SUBCOMMANDS: cook Re-hydrate the minimum project skeleton identified by cargo chef prepare and build it to cache dependencies prepare Analyze the current project to determine the minimum subset of files (Cargo.lock and Cargo.toml manifests) required to build it and cache dependencies ```

prepare examines your project and builds a recipe that captures the set of information required to build your dependencies.

bash cargo chef prepare --recipe-path recipe.json

Nothing too mysterious going on here, you can examine the recipe.json file: it contains the skeleton of your project (e.g. all the Cargo.toml files with their relative path, the Cargo.lock file is available) plus a few additional pieces of information.
In particular it makes sure that all libraries and binaries are explicitly declared in their respective Cargo.toml files even if they can be found at the canonical default location (src/main.rs for a binary, src/lib.rs for a library).

The recipe.json is the equivalent of the Python requirements.txt file - it is the only input required for cargo chef cook, the command that will build out our dependencies:

bash cargo chef cook --recipe-path recipe.json

If you want to build in --release mode:

bash cargo chef cook --release --recipe-path recipe.json

You can leverage it in a Dockerfile:

```dockerfile FROM lukemathwalker/cargo-chef as planner WORKDIR app COPY . . RUN cargo chef prepare --recipe-path recipe.json

FROM lukemathwalker/cargo-chef as cacher WORKDIR app COPY --from=planner /app/recipe.json recipe.json RUN cargo chef cook --release --recipe-path recipe.json

FROM rust as builder WORKDIR app COPY . .

Copy over the cached dependencies

COPY --from=cacher /app/target target COPY --from=cacher $CARGOHOME $CARGOHOME RUN cargo build --release --bin app

FROM rust as runtime WORKDIR app COPY --from=builder /app/target/release/app /usr/local/bin ENTRYPOINT ["/usr/local/bin/app"] ```

We are using four stages: the first computes the recipe file, the second caches our dependencies, the third builds the binary and the fourth is our runtime environment.
As long as your dependencies do not change the recipe.json file will stay the same, therefore the outcome of cargo cargo chef cook --release --recipe-path recipe.json will be cached, massively speeding up your builds (up to 5x measured on some commercial projects).

If you do not want to use the lukemathwalker/cargo-chef image, you can simply install the CLI within the Dockerfile:

```dockerfile FROM rust as planner WORKDIR app

We only pay the installation cost once,

it will be cached from the second build onwards

RUN cargo install cargo-chef COPY . . RUN cargo chef prepare --recipe-path recipe.json

FROM rust as cacher WORKDIR app RUN cargo install cargo-chef COPY --from=planner /app/recipe.json recipe.json RUN cargo chef cook --release --recipe-path recipe.json

FROM rust as builder WORKDIR app COPY . .

Copy over the cached dependencies

COPY --from=cacher /app/target target COPY --from=cacher $CARGOHOME $CARGOHOME RUN cargo build --release --bin app

FROM rust as runtime WORKDIR app COPY --from=builder /app/target/release/app /usr/local/bin ENTRYPOINT ["/usr/local/bin/app"] ```

Benefits vs Limitations

cargo-chef has been tested on a few OpenSource projects and some of commercial projects, but our testing has definitely not exhausted the range of possibilities when it comes to cargo build customisations and we are sure that there are a few rough edges that will have to be smoothed out - please file issues on GitHub.

Benefits of cargo-chef:

A common alternative is to load a minimal main.rs into a container with Cargo.toml and Cargo.lock to build a Docker layer that consists of only your dependencies (more info here). This is fragile compared to cargo-chef which will instead:

Limitations and caveats:

License

Licensed under either of Apache License, Version 2.0 or MIT license at your option. Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in this crate by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.