Skip to content
This repository was archived by the owner on Dec 26, 2024. It is now read-only.

Commit 9a6175e

Browse files
feat(common): implementing cargo-chef in dockerfile for better dependencies layer caching (#2228)
* feat(common): implementing cargo-chef fixing docker layer caching * fix(common): fixed typo * docs(common): refactor comments for better documentation * fix(common): moved rustup target to base image, aligned utilities.dockerfile with recent changes * fix(common): moved rustup target to base image, aligned utilities.dockerfile with recent changes * docs(common): added doc for x86_64-unknown-linux-musl
1 parent 4e3c1bd commit 9a6175e

File tree

3 files changed

+62
-77
lines changed

3 files changed

+62
-77
lines changed

Cargo.toml

-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@ members = [
2424
"crates/sequencing/papyrus_consensus",
2525
]
2626

27-
2827
# Sequencing crates are not part of the default members.
2928
default-members = [
3029
"crates/papyrus_base_layer",

Dockerfile

+59-71
Original file line numberDiff line numberDiff line change
@@ -1,97 +1,85 @@
1-
# We split the Dockerfile into two stages:
2-
# Stage 1: We copy all the Cargo.toml files and create empty lib.rs files.
3-
# Stage 2:
4-
# * We copy the files from the first stage
5-
# * We compile all the crates.
6-
# * We copy the rest of the files and compile again.
7-
# The reason we compile twice is to allow caching for the first compilation (that compiles all the
8-
# dependency crates) if no Cargo.toml files were changed.
9-
# The reason we split it into two stages is to first copy all the files and then erase all
10-
# non-Cargo.toml files. This way, non-Cargo.toml files won't affect the cache of the second stage
11-
# (For more on docker stages, read https://docs.docker.com/build/building/multi-stage/).
12-
FROM rust:1.78 AS copy_toml
1+
# Dockerfile with multi-stage builds for efficient dependency caching and lightweight final image.
2+
# For more on Docker stages, visit: https://docs.docker.com/build/building/multi-stage/
133

14-
COPY crates/ /app/crates/
15-
COPY Cargo.toml /app/
16-
17-
WORKDIR /app/
18-
19-
# Erase all non-Cargo.toml files.
20-
RUN find /app \! -name "Cargo.toml" -type f -delete ; \
21-
find /app -empty -type d -delete; \
22-
# Create empty lib.rs files.
23-
# In order for cargo init to work, we need to not have a Cargo.toml file. In each crate, we rename
24-
# Cargo.toml to another name and after running `cargo init` we override the auto-generated
25-
# Cargo.toml with the original.
26-
mv Cargo.toml _Cargo.toml && \
27-
# TODO: Consider moving to a script.
28-
for dir in crates/*; do \
29-
if [ -f "$dir/Cargo.toml" ]; then \
30-
mv $dir/Cargo.toml $dir/_Cargo.toml \
31-
&& cargo init --lib --vcs none $dir \
32-
&& mv -f $dir/_Cargo.toml $dir/Cargo.toml; \
33-
else \
34-
for subdir in $dir/*; do \
35-
mv $subdir/Cargo.toml $subdir/_Cargo.toml \
36-
&& cargo init --lib --vcs none $subdir \
37-
&& mv -f $subdir/_Cargo.toml $subdir/Cargo.toml; \
38-
done; \
39-
fi; \
40-
done && \
41-
mv _Cargo.toml Cargo.toml
42-
43-
COPY Cargo.lock /app/
44-
45-
# Starting a new stage so that the first build layer will be cached if a non-Cargo.toml file was
46-
# changed.
47-
# Use this image to compile the project to an alpine compatible binary.
48-
FROM clux/muslrust:1.78.0-stable AS builder
49-
WORKDIR /app/
4+
# We use Cargo Chef to compile dependencies before compiling the rest of the crates.
5+
# This approach ensures proper Docker caching, where dependency layers are cached until a dependency changes.
6+
# Code changes in our crates won't affect these cached layers, making the build process more efficient.
7+
# More info on Cargo Chef: https://github.com/LukeMathWalker/cargo-chef
508

9+
# We start by creating a base image using 'clux/muslrust' with additional required tools.
10+
FROM clux/muslrust:1.78.0-stable AS chef
11+
WORKDIR /app
5112
RUN apt update && apt install -y clang unzip
13+
RUN cargo install cargo-chef
5214
ENV PROTOC_VERSION=25.1
5315
RUN curl -L "https://github.com/protocolbuffers/protobuf/releases/download/v$PROTOC_VERSION/protoc-$PROTOC_VERSION-linux-x86_64.zip" -o protoc.zip && unzip ./protoc.zip -d $HOME/.local && rm ./protoc.zip
5416
ENV PROTOC=/root/.local/bin/protoc
55-
56-
57-
# Copy all the files from the previous stage (which are Cargo.toml and empty lib.rs files).
58-
COPY --from=copy_toml /app .
59-
60-
# Add the proc_macro code since cargo init puts autogenerated code in lib.rs file that breaks the build.
61-
COPY crates/papyrus_proc_macros /app/crates/papyrus_proc_macros
62-
63-
RUN rustup target add x86_64-unknown-linux-musl && \
64-
CARGO_INCREMENTAL=0 cargo build --target x86_64-unknown-linux-musl --release --package papyrus_node
65-
66-
# Copy the rest of the files.
67-
COPY crates/ /app/crates
68-
69-
# Touching the lib.rs files to mark them for re-compilation. Then re-compile now that all the source
70-
# code is available
71-
RUN touch crates/*/src/lib.rs; \
72-
CARGO_INCREMENTAL=0 cargo build --release --package papyrus_node --bin papyrus_node;
73-
74-
# Starting a new stage so that the final image will contain only the executable.
17+
# Add the x86_64-unknown-linux-musl target to rustup for compiling statically linked binaries.
18+
# This enables the creation of fully self-contained binaries that do not depend on the system's dynamic libraries,
19+
# resulting in more portable executables that can run on any Linux distribution.
20+
RUN rustup target add x86_64-unknown-linux-musl
21+
22+
#####################
23+
# Stage 1 (planer): #
24+
#####################
25+
FROM chef AS planner
26+
COPY . .
27+
# * Running Cargo Chef prepare that will generate recipe.json which will be used in the next stage.
28+
RUN cargo chef prepare
29+
30+
#####################
31+
# Stage 2 (cacher): #
32+
#####################
33+
# Compile all the dependecies using Cargo Chef cook.
34+
FROM chef AS cacher
35+
36+
# Copy recipe.json from planner stage
37+
COPY --from=planner /app/recipe.json recipe.json
38+
39+
# Build dependencies - this is the caching Docker layer!
40+
RUN cargo chef cook --target x86_64-unknown-linux-musl --release --package papyrus_node
41+
42+
######################
43+
# Stage 3 (builder): #
44+
######################
45+
FROM chef AS builder
46+
COPY . .
47+
COPY --from=cacher /app/target target
48+
# Disable incremental compilation for a cleaner build.
49+
ENV CARGO_INCREMENTAL=0
50+
51+
# Compile the papyrus_node crate for the x86_64-unknown-linux-musl target in release mode, ensuring dependencies are locked.
52+
RUN cargo build --target x86_64-unknown-linux-musl --release --package papyrus_node --locked
53+
54+
###########################
55+
# Stage 4 (papyrus_node): #
56+
###########################
57+
# Uses Alpine Linux to run a lightweight and secure container.
7558
FROM alpine:3.17.0 AS papyrus_node
7659
ENV ID=1000
77-
7860
WORKDIR /app
79-
# Copy the node executable and its config.
61+
62+
# Copy the node executable and its configuration.
8063
COPY --from=builder /app/target/x86_64-unknown-linux-musl/release/papyrus_node /app/target/release/papyrus_node
81-
COPY config/ /app/config
64+
COPY config config
8265

66+
# Install tini, a lightweight init system, to call our executable.
8367
RUN set -ex; \
8468
apk update; \
8569
apk add --no-cache tini; \
8670
mkdir data
8771

72+
# Create a new user "papyrus".
8873
RUN set -ex; \
8974
addgroup --gid ${ID} papyrus; \
9075
adduser --ingroup $(getent group ${ID} | cut -d: -f1) --uid ${ID} --gecos "" --disabled-password --home /app papyrus; \
9176
chown -R papyrus:papyrus /app
9277

78+
# Expose RPC and monitoring ports.
9379
EXPOSE 8080 8081
9480

81+
# Switch to the new user.
9582
USER ${ID}
9683

84+
# Set the entrypoint to use tini to manage the process.
9785
ENTRYPOINT ["/sbin/tini", "--", "/app/target/release/papyrus_node"]

papyrus_utilities.Dockerfile

+3-5
Original file line numberDiff line numberDiff line change
@@ -8,22 +8,20 @@
88

99
INCLUDE Dockerfile
1010

11-
1211
# Build papyrus utilities.
1312
FROM builder AS utilities_builder
1413

1514
# Build papyrus_load_test and copy its resources.
16-
RUN CARGO_INCREMENTAL=0 cargo build --target x86_64-unknown-linux-musl --release --package papyrus_load_test --bin papyrus_load_test
15+
RUN cargo build --target x86_64-unknown-linux-musl --release --package papyrus_load_test --bin papyrus_load_test
1716

1817
# Build dump_declared_classes.
19-
RUN CARGO_INCREMENTAL=0 cargo build --target x86_64-unknown-linux-musl --release --package papyrus_storage --features "clap" \
18+
RUN cargo build --target x86_64-unknown-linux-musl --release --package papyrus_storage --features "clap" \
2019
--bin dump_declared_classes
2120

2221
# Build storage_benchmark.
23-
RUN CARGO_INCREMENTAL=0 cargo build --target x86_64-unknown-linux-musl --release --package papyrus_storage \
22+
RUN cargo build --target x86_64-unknown-linux-musl --release --package papyrus_storage \
2423
--features "clap statistical" --bin storage_benchmark
2524

26-
2725
# Starting a new stage so that the final image will contain only the executables.
2826
FROM alpine:3.17.0 AS papyrus_utilities
2927

0 commit comments

Comments
 (0)