This repository is maintained by the Develocity Solutions team, as one of several publicly available repositories:
This Maven extension allows to make the Quarkus Maven plugin build
goal cacheable.
This project performs programmatic configuration of the Develocity Build Cache through a Maven extension. See here for more details.
Note:
A native executable can be a very large file. Copying it from/to the local cache, or transferring it from/to the remote cache can be an expensive operation that has to be balanced with the duration of the work being avoided.
Quarkus 3.2.4 and above which brings track-config-changes goal
Note
Although Quarkus 3.2.4 is required, 3.9.0 and above is recommended as it exposes Quarkus extra dependencies which is added as extra input by the current extension.
This additional input is necessary when using snapshot versions (or when overwriting fixed version) of
- The Quarkus dependencies
- A custom Quarkus extension
Only the native
, uber-jar
, jar
and legacy-jar
packaging types can be made cacheable
By default, the native
packaging is cacheable only if the in-container build strategy (quarkus.native.container-build=true
) is configured along with a fixed build image (quarkus.native.builder-image
).
The in-container build strategy means the build is as reproducible as possible. Even so, some timestamps and instruction ordering may be different even when built on the same system in the same environment.
If the build environments are strictly identical, this restriction can be removed by setting DEVELOCITY_QUARKUS_NATIVE_BUILD_IN_CONTAINER_REQUIRED=false
. See configuration section for more details.
Note
When the in-container build strategy is used as a fallback the caching feature will be disabled. The fallback may happen due to GraalVM requirements not met. The recommendation is to explicitly set the in-container strategy (quarkus.native.container-build=true
) to benefit from caching
Reference the extension in .mvn/extensions.xml
(this extension requires the develocity-maven-extension):
<extensions>
<extension>
<groupId>com.gradle</groupId>
<artifactId>develocity-maven-extension</artifactId>
<version>1.23.1</version>
</extension>
<extension>
<groupId>com.gradle</groupId>
<artifactId>quarkus-build-caching-extension</artifactId>
<version>1.8</version>
</extension>
</extensions>
Note on the Compatibility with The Develocity extension:
Extension | Compatible version |
---|---|
com.gradle:develocity-maven-extension |
1.+ |
com.gradle:gradle-enterprise-maven-extension |
0.12 |
Enable Quarkus config tracking in pom.xml
:
<properties>
<quarkus.config-tracking.enabled>true</quarkus.config-tracking.enabled>
</properties>
Add the track-prod-config-changes
execution to the quarkus-maven-plugin
configuration:
<plugin>
<groupId>${quarkus.platform.group-id}</groupId>
<artifactId>quarkus-maven-plugin</artifactId>
<version>${quarkus.platform.version}</version>
<extensions>true</extensions>
<executions>
<execution>
<id>track-prod-config-changes</id>
<phase>process-resources</phase>
<goals>
<goal>track-config-changes</goal>
</goals>
<configuration>
<dumpCurrentWhenRecordedUnavailable>true</dumpCurrentWhenRecordedUnavailable>
</configuration>
</execution>
<execution>
<goals>
<goal>build</goal>
<goal>generate-code</goal>
<goal>generate-code-tests</goal>
</goals>
</execution>
</executions>
</plugin>
After applying the quarkus-maven-plugin configuration step, invoke the Quarkus build
goal to generate the .quarkus/quarkus-prod-config-dump
file.
This file is required to make the Quarkus build
goal cacheable.
The file should be checked-in to the source code repository. in continuous integration environment, an alternative is to have the file restored (see this option for GitHub actions as an example).
If the file has some system dependent properties, it is possible to have different configuration dump to reflect those changes (local, ci, os...) conditionally enabled by Maven profiles.
It is also possible to ignore properties not impacting the produced artifacts.
The quarkus.native.graalvm-home
and quarkus.native.java-home
are some classic examples, the JDK version is already captured as a goal input and the path to the JDK does not impact the produced artifact.
Configuration can be set with (listed in order of precedence ):
The caching can be disabled by setting:
DEVELOCITY_QUARKUS_CACHE_ENABLED=false
By default, the values below are used to compute the dump-config (.quarkus/quarkus-prod-config-dump
) and
config-check (target/quarkus-prod-config-check
) file names:
- build profile: prod
- file prefix: quarkus
- file suffix: config-dump
Those values can be overridden, when CI and local have different Quarkus properties for instance:
DEVELOCITY_QUARKUS_BUILD_PROFILE=prod
DEVELOCITY_QUARKUS_DUMP_CONFIG_PREFIX=quarkus
DEVELOCITY_QUARKUS_DUMP_CONFIG_SUFFIX=config-dump-ci
Some additional outputs can be configured if necessary (when using the quarkus-helm extension for instance). The paths are relative to the target
folder.
Directories can be added (csv list):
DEVELOCITY_QUARKUS_KEY_EXTRA_OUTPUT_DIRS=helm
or Specific files (csv list):
DEVELOCITY_QUARKUS_KEY_EXTRA_OUTPUT_FILES=helm/kubernetes/my-project/Chart.yaml,helm/kubernetes/my-project/values.yaml
The default is to enable caching only when the in-container build strategy is used. If the build environments are strictly identical build over build, the restriction can be removed by setting:
DEVELOCITY_QUARKUS_NATIVE_BUILD_IN_CONTAINER_REQUIRED=false
The same configuration can be achieved with Maven properties:
<properties>
<develocity.quarkus.cache.enabled>true</develocity.quarkus.cache.enabled>
<develocity.quarkus.build.profile>prod</develocity.quarkus.build.profile>
<develocity.quarkus.dump.config.prefix>quarkus</develocity.quarkus.dump.config.prefix>
<develocity.quarkus.dump.config.suffix>config-dump-ci</develocity.quarkus.dump.config.suffix>
<develocity.quarkus.extra.output.dirs>helm</develocity.quarkus.extra.output.dirs>
<develocity.quarkus.extra.output.files>helm/kubernetes/${project.artifactId}/Chart.yaml,helm/kubernetes/${project.artifactId}/values.yaml</develocity.quarkus.extra.output.files>
<develocity.quarkus.native.build.in.container.required>false</develocity.quarkus.native.build.in.container.required>
</properties>
A configuration file can be used instead by defining its location (relative to the project root folder) either:
- as an environment variable:
DEVELOCITY_QUARKUS_CONFIG_FILE=.quarkus/develocity-ci.properties
- as a maven property:
<develocity.quarkus.config.file>.quarkus/extension-local.properties</develocity.quarkus.config.file>
Its content can be created like described in the environment variables section.
It is also possible to configure some properties to be excluded from configuration tracking (more details in the Quarkus documentation). This is relevant when a property is volatile but does not impact the produced artifact, see this section for more details.
<properties>
<quarkus.config-tracking.exclude>quarkus.container-image.tag,quarkus.application.version</quarkus.config-tracking.exclude>
</properties>
Debug logging on the extension can be configured with the following property
mvn -Dorg.slf4j.simpleLogger.log.com.gradle=debug clean install
The logic to make the Quarkus build
goal cacheable is isolated to the QuarkusBuildCache class.
A key component of the caching mechanism is the Quarkus configuration dump file .quarkus/quarkus-prod-config-dump
.
This file is generated by the Quarkus build
goal when the Maven property quarkus.config-tracking.enabled
is true
.
It contains all the Quarkus properties used during the Quarkus build
process.
Some properties are discovered late in the build
phase and can't be determined in advance, thus the need for a first full execution to generate the file.
The presence of the file is required to mark the Quarkus build
goal as cacheable.
The track-config-changes
goal creates a file target/quarkus-prod-config-check
containing all the properties from the .quarkus/quarkus-prod-config-dump
with their actual value.
If property values are identical in the two files, it means that the Quarkus configuration was not changed since the last Quarkus build
, therefore the Quarkus build
goal can be marked cacheable.
When the Quarkus build
goal is marked cacheable, the regular caching process using inputs and outputs kicks in as described here.
Let's illustrate the extension behavior with the following sequence of builds:
Initialization build (one-off step):
-
track-config-changes
does nothing as.quarkus/quarkus-prod-config-dump
is absent -
Quarkus configuration from current and previous build differ
=> The
build
goal is not cacheable -
build
executes and creates.quarkus/quarkus-prod-config-dump
First (post-initialization) build:
-
track-config-changes
createstarget/quarkus-prod-config-check
-
Quarkus configuration from current and previous build are identical (assuming Quarkus configuration was unchanged)
=> The
build
goal is cacheable -
Cache lookup happens: CACHE MISS
-
build
executes and creates.quarkus/quarkus-prod-config-dump
-
output is stored into the cache
Next builds:
-
track-config-changes
createstarget/quarkus-prod-config-check
-
Quarkus configuration from current and previous build are identical (assuming Quarkus configuration was unchanged)
=> The
build
goal is cacheable -
Cache lookup happens: CACHE HIT
-
build
is not executed
This extension makes the Quarkus build goal cacheable by configuring the following goal inputs:
- The compilation classpath
- Generated sources directory
- JDK version
- OS details (name, version, arch)
See here for details
Quarkus' properties are fetched from the config dump populated by the Quarkus build
goal.
The build
goal is cacheable only if the track-config-changes
goal generates a config dump identical to the one generated by the previous build
execution.
This ensures that the local Quarkus configuration hasn't changed since last build, otherwise a new build
execution is required as a configuration can change the produced artifact.
target/quarkus-prod-config-check
is added as a goal input
Some properties are pointing to a file which has to be declared as file input. This allows to have the file content part of the cache key (RELATIVE_PATH
strategy).
quarkus.docker.dockerfile-native-path
quarkus.docker.dockerfile-jvm-path
quarkus.openshift.jvm-dockerfile
quarkus.openshift.native-dockerfile
Quarkus dynamically adds some dependencies to the build which will be listed in the target/quarkus-prod-dependencies.txt
file.
This file is created by the Quarkus track-config-changes
goal and contains the absolute path to each dependency (one dependency per line).
This fileset is added as goal input with a RUNTIME_CLASSPATH
normalization strategy.
Quarkus dynamically adds some dependencies to the build which will be listed in the target/quarkus-prod-dependency-checksums.txt
file.
This file is created by the Quarkus track-config-changes
goal and contains the list of dependencies along with their checksum for snapshot versions (one dependency per line).
This file is added as goal input with a RELATIVE_PATH
normalization strategy.
Here are the files added as output:
target/<project.build.finalName>-runner
target/<project.build.finalName>.jar
target/<project.build.finalName>-runner.jar
target/quarkus-artifact.properties
Note
Some additional outputs can be configured. See the configuration section for more details.
When the test goals (maven-surefire-plugin
and maven-failsafe-plugin
) are running some @QuarkusTest
or @QuarkusIntegrationTest
,
it is important for consistency to add implicit dependencies as goal additional input.
Specifically for maven-failsafe-plugin
, the Quarkus artifact descriptor quarkus-artifact.properties
also needs to be added.
With Quarkus 3.9.0+, This can be achieved by declaring a property addQuarkusInputs
on the test goal:
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<configuration>
<properties>
<addQuarkusInputs>true</addQuarkusInputs>>
</properties>
</configuration>
</plugin>
<plugin>
<artifactId>maven-failsafe-plugin</artifactId>
<configuration>
<properties>
<addQuarkusInputs>true</addQuarkusInputs>>
</properties>
</configuration>
</plugin>
</plugins>
Prior to Quarkus 3.9.0:
<plugins>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<configuration>
<properties>
<addQuarkusPackageInputs>true</addQuarkusPackageInputs>>
</properties>
</configuration>
</plugin>
<plugin>
<artifactId>maven-failsafe-plugin</artifactId>
<configuration>
<properties>
<addQuarkusPackageInputs>true</addQuarkusPackageInputs>>
</properties>
</configuration>
</plugin>
</plugins>