Skip to content

Commit 1f95f3a

Browse files
authored
Add instructions for a privileged build (#13)
1 parent ceb6739 commit 1f95f3a

4 files changed

+79
-3
lines changed

.dockerignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Dockerfile.*

Dockerfile.checkpoint

+4-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1-
ARG JDK_NAME=zulu22.30.13-ca-crac-jdk22.0.1-linux_x64
1+
ARG BASE_IMAGE=ubuntu:24.04
2+
ARG JDK_NAME=zulu22.32.17-ca-crac-jdk22.0.2-linux_x64
23

3-
FROM ubuntu:24.04 as builder
4+
FROM $BASE_IMAGE AS builder
45
ARG JDK_NAME
56

67
RUN apt-get update && apt-get install -y wget
@@ -9,7 +10,7 @@ RUN tar zxf ./crac-jdk.tar.gz -C /usr/share
910

1011
# End of builder
1112

12-
FROM ubuntu:24.04
13+
FROM $BASE_IMAGE
1314
ARG JDK_NAME
1415
ARG FAT_JAR=
1516

Dockerfile.privileged

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
# syntax=docker/dockerfile:1.3-labs
2+
3+
###
4+
# This Dockerfile requires priviledged builder and as such cannot be run usin
5+
# regular `docker build ...`. You need to first create a buildx builder and
6+
# then use it to perform the build:
7+
#
8+
# docker buildx create --buildkitd-flags '--allow-insecure-entitlement security.insecure' --name privileged-builder
9+
# docker buildx build --load --builder privileged-builder --allow=security.insecure -f Dockerfile.privileged -t example-spring-boot .
10+
#
11+
###
12+
13+
ARG BASE_IMAGE=ubuntu:24.04
14+
ARG JDK_NAME=zulu22.32.17-ca-crac-jdk22.0.2-linux_x64
15+
16+
FROM $BASE_IMAGE AS builder
17+
ARG JDK_NAME
18+
ENV JAVA_HOME=/usr/share/$JDK_NAME
19+
ENV ENDPOINT=http://localhost:8080
20+
21+
RUN apt-get update && apt-get install -y curl maven siege wget
22+
RUN wget -O crac-jdk.tar.gz https://cdn.azul.com/zulu/bin/$JDK_NAME.tar.gz
23+
RUN tar zxf ./crac-jdk.tar.gz -C /usr/share
24+
25+
ADD . /example-spring-boot
26+
RUN cd /example-spring-boot && mvn -B install && mv target/example-spring-boot-0.0.1-SNAPSHOT.jar /example-spring-boot.jar
27+
28+
RUN --security=insecure <<END_OF_SCRIPT
29+
#!/bin/bash
30+
31+
# Start the process in background. We are adding -XX:CRaCMinPid to explicitly
32+
# offset PIDs (needed for unprivileged restore) as the process started from
33+
# this script won't be running as PID 1, disabling the built-in offsetting.
34+
$JAVA_HOME/bin/java -XX:CPUFeatures=generic -XX:CRaCMinPid=128 -XX:CRaCCheckpointTo=/cr -jar /example-spring-boot.jar &
35+
PID=$!
36+
# Wait until the connection is opened
37+
until curl --output /dev/null --silent --head --fail $ENDPOINT; do
38+
sleep 0.1
39+
done
40+
# Warm-up the server by executing 100k requests against it
41+
siege -c 1 -r 100000 -b $ENDPOINT
42+
# Do the checkpoint
43+
$JAVA_HOME/bin/jcmd /example-spring-boot.jar JDK.checkpoint
44+
# Wait until the process completes, returning success (wait would return exit code 137)
45+
wait $PID || true
46+
47+
END_OF_SCRIPT
48+
49+
FROM $BASE_IMAGE
50+
ARG JDK_NAME
51+
ENV JAVA_HOME=/usr/share/$JDK_NAME
52+
ENV PATH="$PATH:$JAVA_HOME/bin"
53+
54+
COPY --from=builder /usr/share/${JDK_NAME} /usr/share/${JDK_NAME}
55+
COPY --from=builder /cr /cr
56+
COPY --from=builder /example-spring-boot.jar /example-spring-boot.jar
57+
CMD [ "java", "-XX:CRaCRestoreFrom=/cr" ]

README.md

+17
Original file line numberDiff line numberDiff line change
@@ -73,3 +73,20 @@ docker run -it --rm -p 8080:8080 example-spring-boot
7373
# In another terminal
7474
curl localhost:8080
7575
```
76+
77+
## Checkpoint as Dockerfile build step
78+
79+
In order to perform a checkpoint in a container we need those extra capabilites mentioned in the commands above. Regular `docker build ...` does not include these and therefore it is not possible to do the checkpoint as a build step - unless we create a docker buildx builder that will include these. See [Dockerfile reference](https://docs.docker.com/reference/dockerfile/#run---security) for more details. Note that you require a recent version of Docker BuildKit to do so.
80+
81+
```
82+
docker buildx create --buildkitd-flags '--allow-insecure-entitlement security.insecure' --name privileged-builder
83+
docker buildx build --load --builder privileged-builder --allow=security.insecure -f Dockerfile.privileged -t example-spring-boot .
84+
```
85+
86+
Now you can start the example as before with
87+
```
88+
docker run -it --rm -p 8080:8080 example-spring-boot
89+
```
90+
91+
The most important part of the Dockerfile is invoking the checkpoint with `RUN --security=insecure`. Also, when creating your own Dockerfiles don't forget to enable the experimental syntax using `# syntax=docker/dockerfile:1.3-labs`.
92+

0 commit comments

Comments
 (0)