-
Notifications
You must be signed in to change notification settings - Fork 169
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Rewrite the plugin using standard javax.tools
API instead of Plexus.
#271
Conversation
Other aspects that are rewritten include incremental build mechanism and how the overwriting of `module-info.java` in tests is handled. For details and impact on users, see pull request description at apache#271
Fwiw, the One possibility would be to nullify the |
Would a Alternatively, should the default value be simply removed? I remember a discussion years ago about having a stable default value. But since the Maven default also need to be upgraded from times to times, if a user didn't specified the target version, aren't we basically replacing his dependency to the Java version installed on his machine by a dependency to the Maven version installed on his machine? |
The parent POM must not be the only location where the defaults are set, as in the case of the IT, it does not inherit from the apache POM. So either you added the null values as a safe belt to all ITs, or we need to find another source.
The problem is that the apache-parent POM may still be used with Maven 3.
For compatibility, I think it's safer to remove those when using JDK 9+ only for now. |
Fwiw, apache/maven#1865 should fix the |
I added these null values because otherwise,
Thanks! I will test at the first opportunity, maybe tomorrow. |
Thanks, all integration tests pass now (tested on Java 17 and 23). Does the plugin need to wait for a Maven 4.0.0-beta-6 release, or is it okay to go forward as things stand now? |
We could disable the failing IT until beta-6 is out. And raise a PR to re-establish the test to not forget it. |
It will be re-enabled after Maven 4.0.0-beta-6 release. apache/maven#1865
Done. With MCOMPILER-346 disabled, all integration tests pass (Java 17 and 23). The |
There is a test failure on Windows in MCOMPILER-192, which I suspect come from the <includes>
<include>dummy/*.java</include>
<include>dummy/license.txt</include>
</includes> The code was supposed to translate |
…syntax. It appears that the GLOB matcher expects `/` even on Windows.
Fixed (I was replacing |
…tead of returning the Boolean `false` value. Adjust what is the cause and what is the suppressed exception.
@desruisseaux the plugin has been released
|
Done. The CI job failed, but the cause seems a temporary network problem (a "GOAWAY received"):
I verified that the integration tests passed on my machine (Java 23, Linux Fedora). |
@desruisseaux anything that needs to be done ? Or should I merge this PR now ? |
I suggest to merge if you don't mind. The first failure today was a network problem, and the two other failures seem random (happening in different OS + Java combinations). I have no idea at this time what could be the cause. If it happens to be an issue in the code, I hope that feedback would allow us to fix it. |
We definitely have intermittent issues, so I really think the code is not a problem here. |
Thanks! |
This is a proposed rewriting of the Maven compiler plugin. This work is for Maven 4 only, the plugin for Maven 3 will stay unaffected. A major goal of this work is to improve the support of Java Module System in Maven.
Incompatible changes
This section describes behaviors of the Maven compiler plugin 4 that changed in incompatibles way compared to versions 3. Some projects building successfully with version 3 of
maven-compiler-plugin
may fail with version 4 because of those changes. This section describes corrections that can be applied to projects for fixing those build failures.Cannot use both
--release
and--source
As a general rule, this new Maven plugin does not modify the compiler options specified in the configuration. This policy may break projects that specify the
<release>
option together with<source>
and<target>
. The previous version of the plugin applied some heuristic rules for deciding which options to pass to the Java compiler. The new version passes all options verbatim, which may result in the following message from thejavac
compiler:Correction if the build fails
Removes the
<source>
and<target>
configuration parameters. They may be specified indirectly through themaven.compiler.source
andmaven.compiler.target
properties. If those configuration parameters or properties are inherited from a super-POM that cannot be modified, clear the<source>
and<target>
parameters as below:Use of dependencies with automatic names
If a dependency does not declare explicitly whether the JAR file should be placed on the class-path or on the module-path, the Maven plugin uses heuristic rules for making its own decision. Those rules are not obvious, and the selected option is not always appropriate for the project. The results may sometime be different between this version and the previous version of
maven-compiler-plugin
. Some JAR files may placed on the class-path when they were previously on the module-path, or conversely.By default, a dependency is placed on
--module-path
if the project is itself modularized (i.e., contains amodule-info.java
) and one of the following conditions is true:module-info.class
entry.META-INF/MANIFEST.MF
entry with anAutomatic-Module-Name
attribute.In all other cases, the dependency is placed on
--class-path
. This default behavior may be adjusted in future Maven plugin versions depending on experience.Correction if the build fails
Developers are strongly encouraged to specify explicitly where they want their non-modular JAR files to appear in a modular project. It can be done by setting the type of a dependency to
modular-jar
orclasspath-jar
. Example:Note that specifying the dependency type is usually not needed in other cases (e.g., non-modular project, or dependencies that are already modular). It should not be needed neither for dependencies with the runtime scope. However, it may be a good practice to be explicit anyway for more guarantees.
Guideline for choosing the dependency type
If a non-modular dependency is forced to the
modular-jar
type, Java will automatically infer a module name from the file name. That name can be shown by thejar --describe-module
command. For example, the following command shows the name of aboveplexus-utils
dependency on a Unix system:jar --describe-module --file ~/.m2/repository/org/codehaus/plexus/plexus-utils/3.0.24/plexus-utils-3.0.24.jar
The command output contains
[email protected] automatic
. That module name must be declared in themodule-info.java
file.Conversely, if a dependency is forced to the
classpath-jar
type, then themodule-info.java
is not modified. Instead, thepom.xml
file should have the following configuration (replacemy.module
by the actual project module name):jpms.args
generationThe
META-INF/jpms.args
file is no longer generated.Changes in compiler parameters
In its current version, the new plugin does not remove any parameters that exist in version 3 of
maven-compiler-plugin
. However, the behavior of some parameters has been modified and some parameters are deprecated. This section lists the changes.Deprecated but still working parameters
The following parameters are deprecated but are still working:
<compilerArgument>
: already replaced by<compilerArgs>
since Maven 3.1. The replacement uses a list of strings instead of a single string.<testCompilerArgument>
: replaced by<testCompilerArgs>
for the same reason as above. Note that the latter is a new parameter introduced in Maven 4, see next point.<testCompilerArguments>
: replaced by<testCompilerArgs>
for consistency with the main plugin. The former parameter was inconsistent in the name ("Arguments" suffix instead of "Args"), but also in the value type (Map<String,String>
instead ofList<String>
).<annotationProcessorPaths>
: replaced by ordinary dependencies with<type>proc</type>
. The latter is a new artifact type introduced in Maven 4-alpha13.<useModulePath>
: replaced by<type>classpath-jar</type>
declarations in dependencies.Deprecated parameters that are no-op
The following parameters are marked as deprecated for removal. They all became no-op in the new plugin:
<forceJavacCompilerUse>
: the documentation is not really explicit, but this parameters seems to be about forcing the use ofjava.lang.Compiler
API instead ofjavax.tools.JavaCompiler
. The former class was deprecated since Java 9 and no longer exists in Java 21.<compilerVersion>
: this parameter was passed toorg.codehaus.plexus.compiler.CompilerConfiguration
, but it is unclear how the Plexus compiler used it. We see no obvious mapping injavax.tools
orjava.lang.Process
API.<compilerReuseStrategy>
: does not apply well to thejavax.tools.JavaCompiler
API that the new plugin is using.<skipMultiThreadWarning>
: deprecated as a consequence of<compilerReuseStrategy>
deprecation.<optimize>
: was already deprecated in Maven 3, reported here for completness.<outputFileName>
: producing the final JAR file can be considered as the task of separated plugins, for example the JAR plugin. Furthermore, this parameter does not work well in the context of Module Source Hierarchy, because the output is not a single JAR file.<outputTimestamp>
: not really applicable to the Maven compiler plugin. It was used only as an heuristic rules for guessing if the developer intended to have a reproducible build.Changes in default values
The Maven compiler plugin has numerous default values, which sometime differ from the standard (JDK) default values. For example, the default value of
-source
and-target
options was hard-coded to "1.8" in Maven 3, while the standard (JDK) default is the version of the JDK used for compiling the code. The following parameters had their default value modified in the new plugin:<source>
and<target>
: removed the default value and added a Javadoc saying "As of Java 9, the --release option is preferred."<release>
: no default value. Therefor, the standard (JDK) default applies.<release>
and<target>
are unspecified, a warning is logged.<debugLevel>
: omitting this parameter now means "JDK default debug info" instead of "all debug info".all
level (Maven-specific) has been added for meaning "all debug info".<createMissingPackageInfoClass>
: default changed fromtrue
tofalse
,because this workaround is no longer necessary for the new incremental build system.
<showWarnings>
: default totrue
for consistency with the JDK default value.Default values not yet changed
The following changes are in consideration, but not yet applied. The intend for those changes would be to have the same default values as the
javac
tool:<meminitial>
and<maxmem>
: if no unit is specified, change the default from 'M' to bytes as interpreted by thejavac
tool.Changes in parameters validation
<debugLevel>
values unknown to the plugin but known by the compiler are now accepted.Changes in implementation
This section lists some noticeable implementation changes.
Plexus dependencies removed
The major change is the replacement of Plexus compiler API by the standard
javax.tools.JavaCompiler
API introduced in Java 6. This changes required an almost full rewrite of the compiler plugin. Other Plexus dependencies such asStringUtils
could also be removed because they now have replacements in the standard Java API.ASM no longer used for generating package-info
The previous compiler plugin used ASM for generating the
package-info.class
files that were omitted by the Java compiler because empty. This issue is moot because thepackage-info.class
files were generated as a workaround for the way that Maven incremental compilation worked. The new plugin no longer generate those classes by default (i.e., the<createMissingPackageInfoClass>
default value has been changed tofalse
). If emptypackage-info.class
files are nevertheless desired, they are generated by the Java compiler itself with the-Xpkginfo:always
option. If that extra option is not supported, then a warning is logged and no option is added.Heuristic removed
The previous plugin tried to be clever by intercepting and modifying the values of user-supplied
--patch-module
compiler arguments, by adding--path-module
and--add-reads <module>=ALL-UNNAMED
arguments on its own initiative, etc. The result was saved in aMETA-INF/jpms.args
file added to the JAR file. Above-cited heuristics have been removed from the new compiler plugin, as this plugin aims to give more configuration control to the users. For example, developers are encouraged to set a dependency type tomodular-jar
orclasspath-jar
instead ofjar
. Some heuristics have been kept for compatibility with current widespread practice, in particular when the dependency type is onlyjar
. But most heuristics should not be applied when developers use the most explicit ways to configure Maven.Include/exclude filters
The Plexus scranners used for include/exclude filters have been replaced by
java.nio.file.PathMatcher
. The set of allowed syntax contains at least "glob" and "regex". SeeFileSystem.getPathMatcher(String)
Javadoc for a description of the "glob" syntax. If no syntax is specified, then the default syntax is "glob" with the following modification:\
, all occurrences of the/
character are replaced by the platform-specific separator.The list of files to process is built by applying the path matcher on each regular (non directory) files. The walk in file trees has the following characteristics:
Incremental compilation
The code for detecting which files to recompile has been rewritten. The previous code had an issue with list of files generated and compared in one case as absolute files, and in other case as relative files. Consequently, the plugin always considered that all files changed, which may explain the performance issue reported in MCOMPILER-209.
The new implementation saves the list of source files, together with their last modification times, in a binary file with the
.cache
extension instead of a text file with the.lst
extension. This custom binary encoding is more compact than the previous text files (because of less redundancies) and has more information. The incremental compilation behavior is modified as below:The last point avoids the problem that the
<createMissingPackageInfoClass>
parameter tried to solve. Because the new algorithm does not depend on.class
timestamps, there is no need to force their creation.If the
<useIncrementalCompilation>
parameter is set tofalse
, the plugin does not use the binary cache file and compare modification times of source files with modification times of class files. This approach reproduces the previous behavior.Detection of changes in dependencies
When checking if a dependency changed, the new implementation compares the JAR timestamps against the start time of the previous build saved in the binary file instead of comparing against the value given by
session.getStartTime()
. The reason for this change is as below:mvn install
is executed on the command-line.A.jar
get a modification time which is after the build start time. Module B and C detect that fact and perform a "clean and build all". This is the current behavior of Maven 3 incremental compilation.But let assume that the compilation failed in module B because of the change in A. The developer fixes the compilarion error in B, then executes
mvn install
a second time:A.jar
is unchanged.A.jar
is before the second build start time.By using the timestamp saved in the binary file instead, the new algorithm will detect that
A.jar
has been updated since the last build of C.Overwriting
module-info
in testsFor modular projects having a
module-info.java
file in their main sources, Maven 3 recommended to declare the test dependencies in anothermodule-info.java
file located in the test sources. The latter was basically a copy of the former with the addition of, for example,requires org.junit.jupiter.api
statements. This approach is deprecated in Maven 4. Java does not let us overwritemodule-info
easily, maybe for security reasons. Maven 3 supported that approach by putting the project upside-down: the test classes were considered the project's main code (so that the test'smodule-info
takes precedence), and the main classes were considered the patch added on top of the tests. This new compiler plugin keeps the main classes as main and the test classes as patch, and instead executes the following steps:module-info.java
file alone.module-info.class
file asmodule-info.class.bak
.module-info.class
file (compiled in step 1) in place of the main one (saved in step 2).module-info.java
file asmodule-info.java.bak
(otherwise the compiler seems to get confused).module-info.java
file which has been compiled at step 1.*.bak
files to their original location.A shutdown hook is registered for executing step 6 even if the user interrupted the compilation with [Ctrl-C]. Above hack works, but is obviously fragile and should be used in last resort only. The preferred approach is to use options formally supported by the JDK such as
--add-reads
. Their use should be easier with this new plugin than with Maven 3, since Maven 4 automatizes some of those options.Compiler options validation
The way to handle compiler options has been modified. Previously, the Maven plugin validated some options before to pass them to the compiler. For example, if the
<debuglevel>
value contains anything else thanlines
,vars
orsource
, the plugin raised an error. The intend was to provide more informative message. But in thejavax.tools.JavaCompiler
interface, there is an API telling us whether an option is supported. Therefor, the new plugin version first asks to the compiler whether the option is supported, and only if the compiler said "no", the validation is performed for producing the error message. Consequently, if the compiler claims to support the-g:foo
option, then the plugin will no longer block the use of thefoo
value in<debuglevel>
even if the plugin does not know that value.Miscellaneous
when the plugin proposes a code snippet (e.g. for specifying the target Java release), the Maven compiler plugin version shown in the snippet is fetched from
META-INF/MANIFEST.MF
instead ofMETA-INF/maven/org.apache.maven.plugins/maven-compiler-plugin/pom.properties
.Changes in tests
This section lists some noticeable changes in the tests.
Changes in POM parameters
Moved or added the following configuration parameters in the
pom.xml
files of some tests:<outputDirectory>
and<testOutputDirectory>
parameters declared under<configuration>
moved to<build>
, because those properties are read-only in the configuration.<source>
and<target>
parameters have been either removed or replaced by<release>
.<type>modular-jar</type>
has been added in the dependency declaration.Changes to met new assumptions
The plugin incremental compilation algorithm depends on the convention that Java source files are located in directories of the same name as their package names, with the
.
separator replaced by path separator (/
or\
). This is a very common convention, but not strictly required by the Java compiler. For example, if thesrc/main/java/MyCode.java
file contains thepackage foo
statement, the compiled class will be located intarget/classes/foo/MyCode.class
— note thefoo
additional directory. In such case, the incremental build algorithm will not track correctly the changes. The following tests have been made compliant with the convention for allowing the algorithm to work:mcompiler-182
in integration tests.Note that due to MCOMPILER-209, the old algorithm was compiling everything without really detecting change. So this issue is maybe not really a regression. To reproduce the old behavior, users can just disable the incremental compilation.
Removed tests
JUnit tests
Removed the following directories and associated test methods:
compiler-one-output-file-test2
because it was redundant withcompiler-one-output-file-test
.The only difference was the addition of include/exclude filters, but that difference had no effect because the compiler mock used in this test was ignoring all sources anyway. This test has been replaced by
compiler-modular-project
.Integration tests
The tests in the following directories were already disabled and have been removed:
MCOMPILER-197
because it ran only on Java 8 while the build now requires Java 17.groovy-project-with-new-plexus-compiler
because it ran only on Java 8 and the plexus compiler has been removed.The tests in the following directores are not supported anymore and have been removed:
release-without-profile
because the plugin no longer try to chose automatically which parameters to use between--source
and--release
. This is justified by the fact that the plugin cannot run on Java 8.release-without-profile-fork
for the same reason as above.