Extension Capabilities
Quarkus extensions may provide certain capabilities and require certain capabilities to be provided by other extensions in an application to function properly.
A capability represents a technical aspect, for example, it could be an implementation of some functionality, a contract or a specification. Each capability has a name which should follow the Java package naming convention, e.g. io.quarkus.rest
.
Capability promises and requirements are described in Quarkus extension descriptors - META-INF/quarkus-extension.properties
entries of the runtime extension JAR artifacts.
Only a single provider of any given capability is allowed in an application. If more than one provider of a capability is detected, the application build will fail with the corresponding error message. |
If one extension requires a certain capability, there must be another one among the application dependencies that provides that capability, otherwise the build will fail with the corresponding error message.
At build time all the capabilities found in the application will be aggregated in an instance of the io.quarkus.deployment.Capabilities
build item that extension build steps can inject to check whether a given capability is available or not.
Declaring capabilities
The quarkus-extension-maven-plugin:extension-descriptor
Maven goal and the extensionDescriptor
Gradle task, that generate extension descriptors, allow configuring provided and required capabilities in the following way:
Conditional capability promises and requirements
A capability may be provided or required only if a certain condition is satisfied, for example, if a certain configuration option is enabled or based on some other condition. Here is how a conditional capability promise can be configured:
In this case, io.quarkus.container.image.openshift.deployment.OpenshiftBuild
should be included in one of the extension deployment dependencies and implement java.util.function.BooleanSupplier
. At build time, the Quarkus bootstrap will create an instance of it and register io.quarkus.container.image.openshift
capability only if its getAsBoolean()
method returns true.
An implementation of the OpenshiftBuild
could look like this:
import java.util.function.BooleanSupplier;
import io.quarkus.container.image.deployment.ContainerImageConfig;
public class OpenshiftBuild implements BooleanSupplier {
private ContainerImageConfig containerImageConfig;
OpenshiftBuild(ContainerImageConfig containerImageConfig) {
this.containerImageConfig = containerImageConfig;
}
@Override
public boolean getAsBoolean() {
return containerImageConfig.builder.map(b -> b.equals(OpenshiftProcessor.OPENSHIFT)).orElse(true);
}
}
CapabilityBuildItem
Each provided capability will be represented with an instance of io.quarkus.deployment.builditem.CapabilityBuildItem
at build time. Theoretically, `CapabilityBuildItem’s could be produced by extension build steps directly, bypassing the corresponding declaration in the extension descriptors. However, this way of providing capabilities should be avoided, unless there is a very good reason not to declare a capability in the descriptor.
Capabilities produced from extension build steps aren’t available for the Quarkus dev tools. As a consequences, such capabilities can not be taken into account when analyzing extension compatibility during project creation or when adding new extensions to a project. |
Querying capabilities
All the capabilities found in an application will be aggregated during the build in an instance of io.quarkus.deployment.Capabilities
build item, which can be injected by extension build steps to check whether a certain capability is present or not. E.g.
@BuildStep
HealthBuildItem addHealthCheck(Capabilities capabilities, DataSourcesBuildTimeConfig dataSourcesBuildTimeConfig) {
if (capabilities.isPresent(Capability.SMALLRYE_HEALTH)) {
return new HealthBuildItem("io.quarkus.agroal.runtime.health.DataSourceHealthCheck",
dataSourcesBuildTimeConfig.healthEnabled);
} else {
return null;
}
}
Capability prefixes
Like a capability name, a capability prefix is a dot-separated string that is composed of either the first capability name element or a dot-separated sequence of the capability name elements starting from the first one. E.g. for capability io.quarkus.resteasy.json.jackson
the following prefixes will be registered:
-
io
-
io.quarkus
-
io.quarkus.resteasy
-
io.quarkus.resteasy.json
Capabilities.isCapabilityWithPrefixPresent(prefix)
could be used to check whether a capability with a given prefix is present.
Given that only a single provider of a given capability is allowed in an application, capability prefixes allow expressing a certain common aspect among different but somewhat related capabilities. E.g. there could be extensions providing the following capabilities:
-
io.quarkus.resteasy.json.jackson
-
io.quarkus.resteasy.json.jackson.client
-
io.quarkus.resteasy.json.jsonb
-
io.quarkus.resteasy.json.jsonb.client
Including any one of those extensions in an application will enable the RESTEasy JSON serializer. In case a build step needs to check whether the RESTEasy JSON serializer is already enabled in an application, instead of checking whether any of those capabilities is present, it could simply check whether an extension with prefix io.quarkus.resteasy.json
is present.