[PREVIEW] Scan for usage of API elements annotated with annotations indicating unstable API
Overview
Libraries such as SmallRye and Hibernate have annotations to indicate that parts of their API are unstable, and may change at any time.
As an example, in the case of SmallRye the annotation is io.smallrye.common.annotation.Experimental
, while Hibernate has org.hibernate.Incubating
. Other libraries have similar annotations.
The aim is to provide a mechanism to tell the user that they are using parts of APIs that are marked as unstable with an annotation.
In a nutshell, this will be done via a two stage process:
-
Server build time: Each feature pack will scan the full classpath looking for usage of the specified annotations. This results in an index file, which the feature pack makes sure will appear in a known location when the server is provisioned.
-
Server run time: When an application is deployed into the server, the application classes will be scanned, looking for usage of API elements that have been recorded in the indices we created in the above step.
The 'server run time' part is all in WildFly Core, while the initial work on the 'server build time' stage is currently in the full WildFly codebase tracked by WFLY-19152. Since these efforts are so closely related, it seems best to deal with them as a whole in this proposal.
Background/Evaluation of other approaches
My initial thought was to use jandex to do this, and this is the first thing mentioned by 'outsiders'. However, Jandex solves a different problem.
Jandex is great for finding classes, methods, fields etc. in a deploymemt that have been annotated with a known annotation.
In our case we want to inspect the user bytecode and find usage of members in libraries provided by the server, when those members have been annotated with one of the mentioned annotations.
While it could be possible to do this at runtime, it would mean needing to inspect the bytecode to find all class and member references and either:
-
load the classes referenced to find if the referenced member has any of the annotations indicating unstable API
-
use the JBoss Modules visibility rules to locate the jars for each reference, and inspect the referenced classes/members for the annotations indicating unstable API
Neither of these approaches were seen as viable. While the inspection of the bytecode part would be the same, the checking of the referenced classes/members would be a lot more costly than the work done to reference the index approach we settled for. While implementing the index, we found that converting the CONSTANT_Utf8_Info
entries in the constant pool from byte arrays to strings added huge overhead. In the current implementation we avoid doing this conversion until needed.
Both the two alternative ideas above would need this conversion to strings, since these CONSTANT_Utf8_Info
entries contain the names of the classes and members.
Issue Metadata
Stability Level
-
Experimental
-
Preview
-
Community
-
default
Dev Contacts
-
Paul Ferraro (SME)
-
Emmanuel Hugonnet (Outside Perspective)
QE Contacts
Testing By
-
Engineering
-
QE
Affected Projects or Components
-
WildFly
-
the 'wildfly' and 'wildfly-ee' feature packs will scan their classpaths and provide the indices of API elements annotated with
io.smallrye.common.annotation.Experimental
andorg.hibernate.Incubating
.
-
-
WildFly Core
-
scans the applications when they are deployed, looking for usage of API elements that exist in the index.
-
-
wildfly/unstable-api-annotation-utils - contains a library providing:
-
A Maven plugin which can be used by the feature packs to do the scanning
-
A highly optimised bytecode scanner and lookup mechanism to find usage in user applications of API elements that appear in the indices created by the feature packs
-
Other Interested Projects
Feature packs external to WildFly, as mentioned in the 'Future Work' section.
Relevant Installation Types
-
Traditional standalone server (unzipped or provisioned by Galleon)
-
Managed domain
-
OpenShift s2i
-
Bootable jar
Requirements
Hard Requirements
-
Note this is at preview level only!
-
There is a maven plugin that can be used in the feature pack builds.
-
It will scan for annotations indicating unstable API, and add the following to the
content/
directory of theorg.wildfly._internal.unstable-api-annotation-index
module (thecontent/
directory is defined as a resource root:-
a file containing the indices for annotations concerning the feature pack.
-
The file may be either a .txt or a .zip
-
-
-
The maven plugin scans the full classpath of the feature pack looking for dependencies to index.
-
For this iteration we will scan
-
the 'wildfly-ee' feature pack for the
org.hibernate.Incubating
annotation. -
the 'wildfly' feature pack for the
io.smallrye.common.annotation.Experimental
annotation. -
It also contains a simple configuration mechanism to narrow down which classpath entries to index (e.g when scanning for
io.smallrye.common.annotation.Experimental
we only check the SmallRye libraries on the classpath) -
To reduce size as much as possible we will add the index files in zip format.
-
-
-
In WildFly Core
-
deployment unit processors will be added to:
-
Iterate all resources from the
org.wildfly._internal.unstable-api-annotation-index
module. These resources are the .zip or .txt files containing the indices for each feature pack -
Use the wildfly/unstable-api-annotation-utils library to scan the bytecode of every class in the user’s deployed application and cross-reference against the loaded indices.
-
-
The deployment unit processors will use the WildFly classloading rules to make sure that in the case of nested archives, that we only scan each class once. E.g. if a .war contains a
WEB-INF/lib/my-library.jar
, classes inWEB-INF/lib/my-library.jar
will only be s scanned as part of theWEB-INF/lib/my-library.jar
resource root scan, and not for the parent .war. -
Once scanned, usage of annotated API elements may be reported depending on configuration in the
core-management
subsystem:-
A new child resource (currently called
service=unstable-api-annotations
) is added with alevel
attribute to configure the reporting-
Since this is a resource/feature with a stability level of preview, if the resource is not present, no scanning or reporting is done.
-
The resource is added to the
preview
flavour of the current version of the model. At the time of writing it I was told there is no need to bump the version of management model constructs coming in lower thancommunity
level.
-
-
If the attribute is set to
log
(this is the default value for the attribute), the usages will be logged to the server.log. In this case the user application will be deployed. -
If the attribute is set to
error
, the deployment will fail with an exception containing the usages.
-
-
-
Nice-to-Have Requirements
Non-Requirements
-
Full indexing of all annotations indicating unstable API
-
See the 'Future Work' section for plans to get full coverage
-
-
Inspection of user classes not in an application deployment, such as:
-
User provided modules
-
Classes provided by the user via the management model with the typical
module
/class
attribute pairs.
-
-
No attempt is made to not add the indices if the server is provisioned a a higher stability level than 'preview'. In other words, if the feature packs contain the indices, they will always end up in the server.
-
Since this is a
preview
feature, and the model version is the same as the original with the addedpreview
qualifier, model transformation is not needed.
Future Work
-
Promote this feature to Community or higher. Once we analyse what a community level feature would contain, the below are some ideas which should be taken into consideration when defining the set of enhancements.
-
Promoting to Community or higher would allow us to enable the scanning and reporting out of the box by default, which (in my opinion) would make it more useful.
-
Investigate the use of layers to provision the feature or not
-
Decide whether to opt in or out of provisioning the scanner utility
-
At the moment, the current feeling is that the zipped index files are small enough to not matter if they are provisioned or not. So the candidate would be the module containing the wildfly/unstable-api-annotation-utils jar.
-
-
Identify all annotations indicating unstable API used by WildFly itself with the help of component leads.
-
Scan for these annotations during feature pack builds, and make them available in a provisioned server.
-
-
Engage with owners of feature packs external to WildFly, and identify all annotations indicating unstable API in those feature packs.
-
Provide instructions on how to do the build time scan, and to make the indices available at runtime once their feature pack has been provisioned.
-
Scan for these annotations during feature pack build, and make them available in a provisioned server.
-
I think an appropriate set of feature packs would be the ones WildFly Glow is aware of.
-
-
Investigate the feasibility of scanning user provided classes from other locations than deployment archives, such as:
-
in user modules
-
in classes referenced via the
module
/class
attribute pairs in the management model
-
-
Investigate adding a deployment marker to force enable/disable the runtime scanning independent of the subsystem setting
-
This could possibly be enhanced with a filter so a user could say to ignore usage of constructs annotated with
org.hibernate.Incubating
(or perhaps an expression to ignore classes matching the expression), while reporting usage of everything else (e.g. constructs annotated withio.smallrye.common.annotation.Experimental
).
-
Backwards Compatibility
Default Configuration
Since this is a preview stability feature, this will not be added to the default configuration.
Importing Existing Configuration
Importing existing configuration will be fine, as this is a new feature.
Deployments
If the feature is enabled, deployments will be scanned for usage of API elements existing in the mentioned indices.
Interoperability
N/A
Security Considerations
I don’t believe there are any security issues.
Test Plan
Tests will be added in both WildFLy and WildFly Core
WildFly Core:
-
The WildFly core feature pack is only used internally for testing, so we will not scan for annotations indicating unstable API with the plugin here. Also, it should be noted that the usage of such annotations in third party components is outside of our control, and likely to change (e.g. an annotated member might 'mature' to no longer have the annotation, or the member might be removed in a future release). Meanwhile, the tests for the scanning mechanism should be in WildFly Core since that is where the deployment unit processors live.
-
Thus, a test feature pack will be created offering an API (including its own annotations for indicating unstable API) and a subsystem providing this API on the classpath.
-
The feature pack will be indexed and the indices will be added to the
org.wildfly._internal.unstable-api-annotation-index
module. -
The test will check that
-
the index file is added to the
$org.wildfly._internal.unstable-api-annotation-index
module, as outlined in the requirements -
when deploying an application using annotated API elements from the test feature pack API, usage of those is reported in line with the configuration in the
subsystem=core-management/service=unstable-api-annotations
resource (as outlined in the requirements). We will test both with-
level=log
- and inspect that the log messages pick up all the usages of annotated API elements. -
level=error
- and inspect that the application fails to deploy, and that the error message picks up all the usages of annotated API elements.
-
-
-
The test will run at the
preview
stability level
-
WildFly:
-
Since we have done the main testing of the mechanism in WildFly Core, we will not test that here
-
The verifier plugin in each feature pack will verify that the
org.wildfly._internal.unstable-api-annotation-index
module in each feature pack contains the expected index files.-
The 'wildfly-ee' feature pack expects
wildfly-ee-feature-pack.zip
-
The 'wildfly' feature pack expects
wildfly-ee-feature-pack.zip
andwildfly-galleon-pack.zip
-
-
A test will be added to
testsuite/integration/microprofile
which runs in an execution provisioning both the 'wildfly-ee' and 'wildfly' feature packs. It will test a number of differently packaged deployments, to make sure that each class only gets scanned once, in order to test that the annotated class/member usage follows the WildFly classloading rules. The scanner outputs the class count when an undocumented system property is set totrue
.-
This test will run at the
preview
stability level
-
Community Documentation
Community documentations will be added as part of the PR to WildFly full. This PR will also contain the parts of the tests which live in WildFly, as mentioned in the previous section
Release Note Content
The core-management subsystem now allows you to enable scanning of your deployments for usage of classes/methods in the SmallRye and Hibernate libraries annotated with org.hibernate.Incubating
and io.smallrye.common.annotation.Experimental
. These annotations indicate that those API elements are likely to change at any time