Skip to content

Commit c9694b1

Browse files
committed
codehaus-plexus#420: fix detection of java version when JAVA_TOOL_OPTIONS is set
1 parent 6950ab9 commit c9694b1

File tree

5 files changed

+98
-2
lines changed

5 files changed

+98
-2
lines changed

plexus-compilers/plexus-compiler-javac/src/main/java/org/codehaus/plexus/compiler/javac/JavacCompiler.java

+32-2
Original file line numberDiff line numberDiff line change
@@ -220,6 +220,16 @@ protected static class Messages {
220220
private final Deque<Class<?>> javacClasses = new ConcurrentLinkedDeque<>();
221221

222222
private static final Pattern JAVA_MAJOR_AND_MINOR_VERSION_PATTERN = Pattern.compile("\\d+(\\.\\d+)?");
223+
private static final String JAVAC_PREFIX = "javac";
224+
225+
/**
226+
* List of variable that may produce an output "Picked up %s: %s\n".
227+
* <p>
228+
* There may be more, but for Jenkins withMaven, this is enough.
229+
*
230+
* @see <a href="https://github.com/openjdk/jdk/blob/81e43114eca5199a0d816c02f50ecb6bc370135b/src/hotspot/share/runtime/arguments.cpp#L3074">arguments.cpp</a>
231+
*/
232+
private static final String[] PICKED_ENV_VARS = {"_JAVA_OPTIONS", "JAVA_TOOL_OPTIONS"};
223233

224234
/** Cache of javac version per executable (never invalidated) */
225235
private static final Map<String, String> VERSION_PER_EXECUTABLE = new ConcurrentHashMap<>();
@@ -258,6 +268,7 @@ private String getOutOfProcessJavacVersion(String executable) throws CompilerExc
258268
* up to 21 (https://docs.oracle.com/en/java/javase/21/docs/specs/man/javac.html#standard-options)
259269
*/
260270
cli.addArguments(new String[] {"-version"}); //
271+
261272
CommandLineUtils.StringStreamConsumer out = new CommandLineUtils.StringStreamConsumer();
262273
try {
263274
int exitCode = CommandLineUtils.executeCommandLine(cli, out, out);
@@ -268,20 +279,39 @@ private String getOutOfProcessJavacVersion(String executable) throws CompilerExc
268279
} catch (CommandLineException e) {
269280
throw new CompilerException("Error while executing the external compiler " + executable, e);
270281
}
271-
version = extractMajorAndMinorVersion(out.getOutput());
282+
283+
/*
284+
* javac will output the content of JAVA_TOOL_OPTIONS which may (sic) contains javac: we are forced to remove it.
285+
*/
286+
String cleanedOutput = cleanPickedUp(out.getOutput(), CommandLineUtils.getSystemEnvVars());
287+
version = extractMajorAndMinorVersion(cleanedOutput);
272288
VERSION_PER_EXECUTABLE.put(executable, version);
273289
}
274290
return version;
275291
}
276292

277293
static String extractMajorAndMinorVersion(String text) {
294+
int javacIndex = text.indexOf("javac");
278295
Matcher matcher = JAVA_MAJOR_AND_MINOR_VERSION_PATTERN.matcher(text);
279-
if (!matcher.find()) {
296+
int start = javacIndex == -1 ? 0 : javacIndex + JAVAC_PREFIX.length();
297+
if (!matcher.find(start)) {
280298
throw new IllegalArgumentException("Could not extract version from \"" + text + "\"");
281299
}
282300
return matcher.group();
283301
}
284302

303+
static String cleanPickedUp(String text, Properties envvars) {
304+
String ls = "\n";
305+
String ntext = StringUtils.unifyLineSeparators(text, ls);
306+
for (String env : PICKED_ENV_VARS) {
307+
String value = StringUtils.unifyLineSeparators(envvars.getProperty(env), ls);
308+
if (value != null) {
309+
ntext = ntext.replace(String.format("Picked up %s: %s\n", env, value), "");
310+
}
311+
}
312+
return ntext;
313+
}
314+
285315
@Override
286316
public CompilerResult performCompile(CompilerConfiguration config) throws CompilerException {
287317
File destinationDir = new File(config.getOutputLocation());

plexus-compilers/plexus-compiler-javac/src/test/java/org/codehaus/plexus/compiler/javac/JavacCompilerTest.java

+51
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,12 @@
22

33
import java.io.BufferedReader;
44
import java.io.IOException;
5+
import java.io.InputStream;
6+
import java.io.InputStreamReader;
57
import java.io.StringReader;
8+
import java.nio.charset.StandardCharsets;
69
import java.util.List;
10+
import java.util.Properties;
711
import java.util.stream.Stream;
812

913
import org.codehaus.plexus.compiler.CompilerMessage;
@@ -13,6 +17,7 @@
1317
import org.junit.jupiter.params.ParameterizedTest;
1418
import org.junit.jupiter.params.provider.Arguments;
1519
import org.junit.jupiter.params.provider.MethodSource;
20+
import org.opentest4j.AssertionFailedError;
1621

1722
import static org.codehaus.plexus.compiler.javac.JavacCompiler.Messages.*;
1823
import static org.hamcrest.CoreMatchers.endsWith;
@@ -121,5 +126,51 @@ void testExtractMajorAndMinorVersion() {
121126
assertEquals("11.0", JavacCompiler.extractMajorAndMinorVersion("javac 11.0.22"));
122127
assertEquals("11.0", JavacCompiler.extractMajorAndMinorVersion("11.0.22"));
123128
assertEquals("21", JavacCompiler.extractMajorAndMinorVersion("javac 21"));
129+
assertEquals("1.8", JavacCompiler.extractMajorAndMinorVersion("1.3.4\njavac 1.8.0_432"));
130+
}
131+
132+
@Test
133+
void testCleanPickedUp() {
134+
assertEquals(
135+
"This text contains CRLF\n",
136+
JavacCompiler.cleanPickedUp("This text contains CRLF\r\n", new Properties()));
137+
138+
// files were generated by using:
139+
// declare -x JAVA_TOOL_OPTIONS=$'-Daaa\n\n-Dxxx'
140+
// declare -x _JAVA_OPTIONS="-Dccc"
141+
// "${JAVA_8_HOME}/bin/javac" -version >
142+
// plexus-compilers/plexus-compiler-javac/src/test/resources/org/codehaus/plexus/compiler/javac/java8_pickedUp.txt 2>&1
143+
// "${JAVA_17_HOME}/bin/javac" -version >
144+
// plexus-compilers/plexus-compiler-javac/src/test/resources/org/codehaus/plexus/compiler/javac/java17_pickedUp.txt 2>&1
145+
// "${JAVA_21_HOME}/bin/javac" -version >
146+
// plexus-compilers/plexus-compiler-javac/src/test/resources/org/codehaus/plexus/compiler/javac/java21_pickedUp.txt 2>&1
147+
Properties envvars = new Properties();
148+
envvars.setProperty("JAVA_TOOL_OPTIONS", "-Daaa\r\n\r\n-Dxxx");
149+
envvars.setProperty("_JAVA_OPTIONS", "-Dccc");
150+
151+
assertEquals(
152+
"javac 1.8.0_432\n", JavacCompiler.cleanPickedUp(readAllLines("java8_pickedUp.txt", "\n"), envvars));
153+
assertEquals(
154+
"javac 17.0.13\n", JavacCompiler.cleanPickedUp(readAllLines("java17_pickedUp.txt", "\r\n"), envvars));
155+
assertEquals("javac 21.0.5\n", JavacCompiler.cleanPickedUp(readAllLines("java21_pickedUp.txt", "\r"), envvars));
156+
}
157+
158+
private String readAllLines(String resource, String ls) {
159+
try (InputStream is = this.getClass().getResourceAsStream(resource)) {
160+
if (is == null) {
161+
throw new AssertionFailedError("No such resource: " + resource + " in class " + this.getClass());
162+
}
163+
try (InputStreamReader isr = new InputStreamReader(is, StandardCharsets.UTF_8);
164+
BufferedReader br = new BufferedReader(isr)) {
165+
StringBuilder sb = new StringBuilder();
166+
for (String line = null; (line = br.readLine()) != null; ) {
167+
sb.append(line).append(ls);
168+
}
169+
return sb.toString();
170+
}
171+
} catch (IOException e) {
172+
throw new AssertionFailedError(
173+
"Could not fetch lines of resource: " + resource + " in class " + this.getClass(), e);
174+
}
124175
}
125176
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
Picked up JAVA_TOOL_OPTIONS: -Daaa
2+
3+
-Dxxx
4+
Picked up _JAVA_OPTIONS: -Dccc
5+
javac 17.0.13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
Picked up JAVA_TOOL_OPTIONS: -Daaa
2+
3+
-Dxxx
4+
Picked up _JAVA_OPTIONS: -Dccc
5+
javac 21.0.5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
Picked up JAVA_TOOL_OPTIONS: -Daaa
2+
3+
-Dxxx
4+
Picked up _JAVA_OPTIONS: -Dccc
5+
javac 1.8.0_432

0 commit comments

Comments
 (0)