Skip to content

Commit dacf83a

Browse files
committed
[MCOMPILER-542] Clean JDK patch version in module-info.class
Signed-off-by: Jorge Solórzano <jorsol@gmail.com>
1 parent eab9d47 commit dacf83a

File tree

7 files changed

+224
-96
lines changed

7 files changed

+224
-96
lines changed

src/it/MCOMPILER-542/pom.xml

+58-50
Original file line numberDiff line numberDiff line change
@@ -1,50 +1,58 @@
1-
<?xml version="1.0" encoding="UTF-8"?>
2-
<!--
3-
~ Licensed to the Apache Software Foundation (ASF) under one
4-
~ or more contributor license agreements. See the NOTICE file
5-
~ distributed with this work for additional information
6-
~ regarding copyright ownership. The ASF licenses this file
7-
~ to you under the Apache License, Version 2.0 (the
8-
~ "License"); you may not use this file except in compliance
9-
~ with the License. You may obtain a copy of the License at
10-
~
11-
~ http://www.apache.org/licenses/LICENSE-2.0
12-
~
13-
~ Unless required by applicable law or agreed to in writing,
14-
~ software distributed under the License is distributed on an
15-
~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16-
~ KIND, either express or implied. See the License for the
17-
~ specific language governing permissions and limitations
18-
~ under the License.
19-
-->
20-
21-
<project xmlns="http://maven.apache.org/POM/4.0.0"
22-
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
23-
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
24-
<modelVersion>4.0.0</modelVersion>
25-
26-
<groupId>org.apache.maven.plugins.compiler.it</groupId>
27-
<artifactId>MCOMPILER-542</artifactId>
28-
<version>1.0-SNAPSHOT</version>
29-
<name>${java.specification.version}</name>
30-
31-
<properties>
32-
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
33-
<project.build.outputTimestamp>2023-08-14T15:12:12Z</project.build.outputTimestamp>
34-
</properties>
35-
36-
<build>
37-
<pluginManagement>
38-
<plugins>
39-
<plugin>
40-
<groupId>org.apache.maven.plugins</groupId>
41-
<artifactId>maven-compiler-plugin</artifactId>
42-
<version>@project.version@</version>
43-
<configuration>
44-
<release>${java.specification.version}</release>
45-
</configuration>
46-
</plugin>
47-
</plugins>
48-
</pluginManagement>
49-
</build>
50-
</project>
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<!--
3+
~ Licensed to the Apache Software Foundation (ASF) under one
4+
~ or more contributor license agreements. See the NOTICE file
5+
~ distributed with this work for additional information
6+
~ regarding copyright ownership. The ASF licenses this file
7+
~ to you under the Apache License, Version 2.0 (the
8+
~ "License"); you may not use this file except in compliance
9+
~ with the License. You may obtain a copy of the License at
10+
~
11+
~ http://www.apache.org/licenses/LICENSE-2.0
12+
~
13+
~ Unless required by applicable law or agreed to in writing,
14+
~ software distributed under the License is distributed on an
15+
~ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16+
~ KIND, either express or implied. See the License for the
17+
~ specific language governing permissions and limitations
18+
~ under the License.
19+
-->
20+
21+
<project xmlns="http://maven.apache.org/POM/4.0.0"
22+
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
23+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
24+
<modelVersion>4.0.0</modelVersion>
25+
26+
<groupId>org.apache.maven.plugins.compiler.it</groupId>
27+
<artifactId>MCOMPILER-542</artifactId>
28+
<version>1.0-SNAPSHOT</version>
29+
<name>${java.specification.version}</name>
30+
31+
<properties>
32+
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
33+
<project.build.outputTimestamp>2023-08-14T15:12:12Z</project.build.outputTimestamp>
34+
</properties>
35+
36+
<dependencies>
37+
<dependency>
38+
<groupId>org.slf4j</groupId>
39+
<artifactId>slf4j-jdk-platform-logging</artifactId>
40+
<version>2.0.9</version>
41+
</dependency>
42+
</dependencies>
43+
44+
<build>
45+
<pluginManagement>
46+
<plugins>
47+
<plugin>
48+
<groupId>org.apache.maven.plugins</groupId>
49+
<artifactId>maven-compiler-plugin</artifactId>
50+
<version>@project.version@</version>
51+
<configuration>
52+
<release>${java.specification.version}</release>
53+
</configuration>
54+
</plugin>
55+
</plugins>
56+
</pluginManagement>
57+
</build>
58+
</project>

src/it/MCOMPILER-542/src/main/java/module-info.java

+5-1
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,8 @@
1616
* specific language governing permissions and limitations
1717
* under the License.
1818
*/
19-
module app {}
19+
module app {
20+
requires java.logging;
21+
requires jdk.zipfs;
22+
requires org.slf4j.jdk.platform.logging;
23+
}

src/it/MCOMPILER-542/src/main/java/org/maven/test/Main.java

-3
Original file line numberDiff line numberDiff line change
@@ -20,9 +20,6 @@
2020

2121
public class Main {
2222

23-
/**
24-
* @param args
25-
*/
2623
public static void main(String[] args) {
2724
System.out.println("Hello World!");
2825
}

src/it/MCOMPILER-542/verify.groovy

+58-30
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,58 @@
1-
/*
2-
* Licensed to the Apache Software Foundation (ASF) under one
3-
* or more contributor license agreements. See the NOTICE file
4-
* distributed with this work for additional information
5-
* regarding copyright ownership. The ASF licenses this file
6-
* to you under the Apache License, Version 2.0 (the
7-
* "License"); you may not use this file except in compliance
8-
* with the License. You may obtain a copy of the License at
9-
*
10-
* http://www.apache.org/licenses/LICENSE-2.0
11-
*
12-
* Unless required by applicable law or agreed to in writing,
13-
* software distributed under the License is distributed on an
14-
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15-
* KIND, either express or implied. See the License for the
16-
* specific language governing permissions and limitations
17-
* under the License.
18-
*/
19-
20-
def proc = 'javap -v target/classes/module-info.class'.execute(null,basedir)
21-
def sout = new StringBuilder(), serr = new StringBuilder()
22-
proc.consumeProcessOutput(sout, serr)
23-
proc.waitForOrKill(1000)
24-
def out = sout.toString()
25-
println "javap -v target/classes/module-info.class>\n$out\nerr> $serr"
26-
27-
def module = out.substring(out.indexOf('Module:'))
28-
def javaVersion = System.getProperty('java.version')
29-
assert module.contains('// "java.base" ACC_MANDATED')
30-
assert !module.contains(javaVersion)
1+
/*
2+
* Licensed to the Apache Software Foundation (ASF) under one
3+
* or more contributor license agreements. See the NOTICE file
4+
* distributed with this work for additional information
5+
* regarding copyright ownership. The ASF licenses this file
6+
* to you under the Apache License, Version 2.0 (the
7+
* "License"); you may not use this file except in compliance
8+
* with the License. You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing,
13+
* software distributed under the License is distributed on an
14+
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
* KIND, either express or implied. See the License for the
16+
* specific language governing permissions and limitations
17+
* under the License.
18+
*/
19+
20+
// Check if the javap tool is available
21+
def javapTool = java.util.spi.ToolProvider.findFirst("javap")
22+
assert javapTool.isPresent() : "javap tool not found. Make sure you have the JDK installed."
23+
24+
def moduleDescriptor = new File(basedir, "target/classes/module-info.class")
25+
// Create a list of arguments to pass to the javap tool
26+
String[] args = ["-v", moduleDescriptor]
27+
28+
def swout = new StringWriter(), swerr = new StringWriter()
29+
// Execute the javap tool with args
30+
def result = javapTool.get().run(new PrintWriter(swout), new PrintWriter(swerr), args)
31+
println swerr.toString().isEmpty() ? "javap output:\n$swout" : "javap error:\n$swerr"
32+
assert (result == 0) : "javap run failed"
33+
34+
// Assertions of module content
35+
def out = swout.toString()
36+
def javaVersion = System.getProperty('java.version')
37+
assert out.contains('// "java.base" ACC_MANDATED') : "module not found in module-info.class"
38+
assert out.contains('// "java.logging"') : "module not found in module-info.class"
39+
assert out.contains('// "jdk.zipfs"') : "module not found in module-info.class"
40+
assert out.contains('// "org.slf4j.jdk.platform.logging"') : "module not found in module-info.class"
41+
assert out.contains('// 2.0.9') : "version of org.slf4j.jdk.platform.logging module not found"
42+
// Validation that the module-info should not contain the java version.
43+
assert !out.contains(' ' + javaVersion) : "ERROR: java version found in module descriptor"
44+
45+
// Additional validation that the checksum is always the same
46+
def javaSpecVersion = System.getProperty('java.specification.version')
47+
48+
def checksumMap = [
49+
'21': 'SHA-256 checksum dd3489f88cb2e5afe569a6711c02f27be8ddb01faf48b9704fb1708a8ccdee6a',
50+
'17': 'SHA-256 checksum ab3ba2a20435f1014706e0dbfa13cc11c697374c54a20d664ab6e07d1b0ed2ac',
51+
'11': 'MD5 checksum 157411fcf56061a19e20922876f9903b'
52+
]
53+
54+
def expectedChecksum = checksumMap[javaSpecVersion]
55+
if (expectedChecksum) {
56+
println "Java version: $javaVersion"
57+
assert out.contains(expectedChecksum) : "checksum doesn't match expected output"
58+
}

src/main/java/org/apache/maven/plugin/compiler/AbstractCompilerMojo.java

+40
Original file line numberDiff line numberDiff line change
@@ -644,6 +644,15 @@ protected final MavenProject getProject() {
644644
return project;
645645
}
646646

647+
protected final Optional<Path> getModuleDeclaration(final Set<File> sourceFiles) {
648+
for (File sourceFile : sourceFiles) {
649+
if ("module-info.java".equals(sourceFile.getName())) {
650+
return Optional.of(sourceFile.toPath());
651+
}
652+
}
653+
return Optional.empty();
654+
}
655+
647656
private boolean targetOrReleaseSet;
648657

649658
@Override
@@ -1177,6 +1186,8 @@ public void execute() throws MojoExecutionException, CompilationFailureException
11771186
}
11781187
}
11791188

1189+
removeJdkModuleVersion(compilerResult, compiler.getCompilerOutputStyle(), sources);
1190+
11801191
if (useIncrementalCompilation) {
11811192
if (incrementalBuildHelperRequest.getOutputDirectory().exists()) {
11821193
getLog().debug("incrementalBuildHelper#afterRebuildExecution");
@@ -1798,4 +1809,33 @@ public void setRelease(String release) {
17981809
final String getImplicit() {
17991810
return implicit;
18001811
}
1812+
1813+
/**
1814+
* Patch module-info.class to remove the java version from the requires java/jdk modules.
1815+
*
1816+
* @param compilerResult should succeed.
1817+
* @param outputStyle of type ONE_OUTPUT_FILE_PER_INPUT_FILE
1818+
* @param sources the list of the source files to check for the "module-info.java"
1819+
*
1820+
* @see <a href="https://issues.apache.org/jira/browse/MCOMPILER-542">MCOMPILER-542</a>
1821+
*/
1822+
private void removeJdkModuleVersion(
1823+
CompilerResult compilerResult, CompilerOutputStyle outputStyle, final Set<File> sources)
1824+
throws MojoExecutionException {
1825+
if (compilerResult.isSuccess()
1826+
&& outputStyle == CompilerOutputStyle.ONE_OUTPUT_FILE_PER_INPUT_FILE
1827+
&& getModuleDeclaration(sources).isPresent()) {
1828+
Path outputDir = getOutputDirectory().toPath();
1829+
Path moduleDescriptor = outputDir.resolve("module-info.class");
1830+
if (Files.isRegularFile(moduleDescriptor)) {
1831+
try {
1832+
final byte[] descriptorOriginal = Files.readAllBytes(moduleDescriptor);
1833+
final byte[] descriptorMod = ModuleInfoTransformer.transform(descriptorOriginal);
1834+
Files.write(moduleDescriptor, descriptorMod);
1835+
} catch (IOException ex) {
1836+
throw new MojoExecutionException("Error reading or writing module-info.class", ex);
1837+
}
1838+
}
1839+
}
1840+
}
18011841
}

src/main/java/org/apache/maven/plugin/compiler/CompilerMojo.java

+5-12
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020

2121
import java.io.File;
2222
import java.io.IOException;
23+
import java.nio.file.Path;
2324
import java.util.ArrayList;
2425
import java.util.Collection;
2526
import java.util.Collections;
@@ -28,6 +29,7 @@
2829
import java.util.List;
2930
import java.util.Map;
3031
import java.util.Map.Entry;
32+
import java.util.Optional;
3133
import java.util.Set;
3234

3335
import org.apache.maven.artifact.Artifact;
@@ -223,18 +225,9 @@ protected Set<String> getExcludes() {
223225
protected void preparePaths(Set<File> sourceFiles) {
224226
// assert compilePath != null;
225227

226-
File moduleDescriptorPath = null;
228+
Optional<Path> moduleDeclaration = getModuleDeclaration(sourceFiles);
227229

228-
boolean hasModuleDescriptor = false;
229-
for (File sourceFile : sourceFiles) {
230-
if ("module-info.java".equals(sourceFile.getName())) {
231-
moduleDescriptorPath = sourceFile;
232-
hasModuleDescriptor = true;
233-
break;
234-
}
235-
}
236-
237-
if (hasModuleDescriptor) {
230+
if (moduleDeclaration.isPresent()) {
238231
// For now only allow named modules. Once we can create a graph with ASM we can specify exactly the modules
239232
// and we can detect if auto modules are used. In that case, MavenProject.setFile() should not be used, so
240233
// you cannot depend on this project and so it won't be distributed.
@@ -249,7 +242,7 @@ protected void preparePaths(Set<File> sourceFiles) {
249242

250243
ResolvePathsRequest<File> request = ResolvePathsRequest.ofFiles(dependencyArtifacts)
251244
.setIncludeStatic(true)
252-
.setMainModuleDescriptor(moduleDescriptorPath);
245+
.setMainModuleDescriptor(moduleDeclaration.get().toFile());
253246

254247
Toolchain toolchain = getToolchain();
255248
if (toolchain instanceof DefaultJavaToolChain) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
/*
2+
* Licensed to the Apache Software Foundation (ASF) under one
3+
* or more contributor license agreements. See the NOTICE file
4+
* distributed with this work for additional information
5+
* regarding copyright ownership. The ASF licenses this file
6+
* to you under the Apache License, Version 2.0 (the
7+
* "License"); you may not use this file except in compliance
8+
* with the License. You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing,
13+
* software distributed under the License is distributed on an
14+
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
* KIND, either express or implied. See the License for the
16+
* specific language governing permissions and limitations
17+
* under the License.
18+
*/
19+
package org.apache.maven.plugin.compiler;
20+
21+
import org.objectweb.asm.ClassReader;
22+
import org.objectweb.asm.ClassVisitor;
23+
import org.objectweb.asm.ClassWriter;
24+
import org.objectweb.asm.ModuleVisitor;
25+
import org.objectweb.asm.Opcodes;
26+
27+
final class ModuleInfoTransformer {
28+
29+
private ModuleInfoTransformer() {}
30+
31+
static byte[] transform(byte[] originalBytecode) {
32+
ClassReader reader = new ClassReader(originalBytecode);
33+
ClassWriter writer = new ClassWriter(0);
34+
35+
ClassVisitor classVisitor = new ClassVisitor(Opcodes.ASM9, writer) {
36+
@Override
37+
public ModuleVisitor visitModule(String name, int access, String version) {
38+
ModuleVisitor originalModuleVisitor = super.visitModule(name, access, version);
39+
return new ModuleVisitor(Opcodes.ASM9, originalModuleVisitor) {
40+
@Override
41+
public void visitRequire(String module, int access, String version) {
42+
// Check if the module name matches the java/jdk modules
43+
if (version != null && (module.startsWith("java.") || module.startsWith("jdk."))) {
44+
// Remove the version from the java.* and jdk.* modules
45+
super.visitRequire(module, access, null);
46+
} else {
47+
// Keep the original require statement
48+
super.visitRequire(module, access, version);
49+
}
50+
}
51+
};
52+
}
53+
};
54+
55+
reader.accept(classVisitor, 0);
56+
return writer.toByteArray();
57+
}
58+
}

0 commit comments

Comments
 (0)