ff240a9135
Reviewed-by: dfuchs
308 lines
13 KiB
Java
308 lines
13 KiB
Java
/*
|
|
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @bug 8314263
|
|
* @summary Signed jars triggering Logger finder recursion and StackOverflowError
|
|
* @library /test/lib ../lib
|
|
* @compile/module=java.base share/classes/jdk/internal/event/EventHelper.java
|
|
* @modules java.base/jdk.internal.logger:+open
|
|
* @build jdk.test.lib.compiler.CompilerUtils
|
|
* jdk.test.lib.process.*
|
|
* jdk.test.lib.util.JarUtils
|
|
* jdk.test.lib.JDKToolLauncher
|
|
* @compile SignedLoggerFinderTest.java SimpleLoggerFinder.java
|
|
* @run main SignedLoggerFinderTest init
|
|
* @run main SignedLoggerFinderTest init sign
|
|
*/
|
|
|
|
import java.io.File;
|
|
import java.nio.file.*;
|
|
import java.security.*;
|
|
import java.util.*;
|
|
import java.util.function.*;
|
|
import java.util.jar.*;
|
|
|
|
import jdk.test.lib.JDKToolFinder;
|
|
import jdk.test.lib.JDKToolLauncher;
|
|
import jdk.test.lib.Utils;
|
|
import jdk.test.lib.process.OutputAnalyzer;
|
|
import jdk.test.lib.process.ProcessTools;
|
|
import jdk.test.lib.util.JarUtils;
|
|
|
|
import static java.nio.file.StandardCopyOption.REPLACE_EXISTING;
|
|
import static java.util.Arrays.asList;
|
|
|
|
public class SignedLoggerFinderTest {
|
|
|
|
/**
|
|
* This test triggers recursion in the broken JDK. The error can
|
|
* manifest in a few different ways.
|
|
* One error seen is "java.lang.NoClassDefFoundError:
|
|
* Could not initialize class jdk.internal.logger.LoggerFinderLoader$ErrorPolicy"
|
|
*
|
|
* The original reported error was a StackOverflow (also seen in different iterations
|
|
* of this run). Running test in signed and unsigned jar mode for sanity coverage.
|
|
* The current bug only manifests when jars are signed.
|
|
*/
|
|
|
|
private static boolean init = false;
|
|
private static boolean signJars = false;
|
|
private static boolean mutliThreadLoad = false;
|
|
private static volatile boolean testComplete = false;
|
|
|
|
private static final String KEYSTORE = "8314263.keystore";
|
|
private static final String ALIAS = "JavaTest";
|
|
private static final String STOREPASS = "changeit";
|
|
private static final String KEYPASS = "changeit";
|
|
private static final String DNAME = "CN=sample";
|
|
private static final String CUSTOM_LOGGER_FINDER_NAME =
|
|
"loggerfinder.SimpleLoggerFinder";
|
|
private static final String CUSTOM_LOGGER_NAME =
|
|
"loggerfinder.SimpleLoggerFinder$SimpleLogger";
|
|
private static final String INTERNAL_LOGGER_FINDER_NAME =
|
|
"sun.util.logging.internal.LoggingProviderImpl";
|
|
private static final String INTERNAL_LOGGER_NAME =
|
|
"sun.util.logging.internal.LoggingProviderImpl$JULWrapper";
|
|
private static final Path jarPath1 =
|
|
Path.of(System.getProperty("test.classes", "."), "SimpleLoggerFinder.jar");
|
|
private static final Path jarPath2 =
|
|
Path.of(System.getProperty("test.classes", "."), "SimpleLoggerFinder2.jar");
|
|
|
|
public static void main(String[] args) throws Throwable {
|
|
init = args.length >=1 && args[0].equals("init");
|
|
signJars = args.length >=2 && args[1].equals("sign");
|
|
|
|
// init only passed in by jtreg test run, initialize the environment
|
|
// for the subsequent test run
|
|
if (init) {
|
|
initialize();
|
|
launchTest(false, false);
|
|
launchTest(false, true);
|
|
launchTest(true, false);
|
|
launchTest(true, true);
|
|
|
|
} else {
|
|
// set up complete. Run the code to trigger the recursion
|
|
// We're in the JVM launched by ProcessTools.executeCommand
|
|
boolean multiThreadLoad = Boolean.getBoolean("multiThreadLoad");
|
|
boolean withCustomLoggerFinder = Boolean.getBoolean("withCustomLoggerFinder");
|
|
|
|
if (multiThreadLoad) {
|
|
long sleep = new Random().nextLong(100L) + 1L;
|
|
System.out.println("multi thread load sleep value: " + sleep);
|
|
new Thread(runnableWithSleep(
|
|
() -> System.getLogger("logger" + System.currentTimeMillis()),
|
|
sleep, "System.getLogger type: ")).start();
|
|
new Thread(runnableWithSleep(
|
|
() -> System.LoggerFinder.getLoggerFinder(),
|
|
sleep, "System.getLoggerFinder type: ")).start();
|
|
}
|
|
|
|
if (withCustomLoggerFinder) {
|
|
JarFile jf = new JarFile(jarPath1.toString(), true);
|
|
jf.getInputStream(jf.getJarEntry("loggerfinder/SimpleLoggerFinder.class"));
|
|
JarFile jf2 = new JarFile(jarPath2.toString(), true);
|
|
jf2.getInputStream(jf.getJarEntry("loggerfinder/SimpleLoggerFinder.class"));
|
|
} else {
|
|
// some other call to prod LoggerFinder loading
|
|
System.getLogger("random" + System.currentTimeMillis());
|
|
System.LoggerFinder.getLoggerFinder();
|
|
}
|
|
Security.setProperty("test_1", "test");
|
|
|
|
// some extra sanity checks
|
|
if (withCustomLoggerFinder) {
|
|
assertEquals(System.LoggerFinder.getLoggerFinder().getClass().getName(),
|
|
CUSTOM_LOGGER_FINDER_NAME);
|
|
System.Logger testLogger = System.getLogger("jdk.event.security");
|
|
assertEquals(testLogger.getClass().getName(), CUSTOM_LOGGER_NAME);
|
|
} else {
|
|
assertEquals(System.LoggerFinder.getLoggerFinder().getClass().getName(),
|
|
INTERNAL_LOGGER_FINDER_NAME);
|
|
System.Logger testLogger = System.getLogger("jdk.event.security");
|
|
assertEquals(testLogger.getClass().getName(), INTERNAL_LOGGER_NAME);
|
|
}
|
|
testComplete = true;
|
|
|
|
// LoggerFinder should be initialized, trigger a simple log call
|
|
Security.setProperty("test_2", "test");
|
|
// allow time to let bootstrap logger flush data
|
|
BootstrapLoggerUtils.awaitPending();
|
|
}
|
|
}
|
|
|
|
// helper to create the inner test. Run config variations with the LoggerFinder jars
|
|
// on the classpath and with other threads running System.Logger calls during load
|
|
private static void launchTest(boolean multiThreadLoad, boolean withCustomLoggerFinder) {
|
|
List<String> cmds = new ArrayList<>();
|
|
cmds.add(JDKToolFinder.getJDKTool("java"));
|
|
cmds.addAll(asList(Utils.getTestJavaOpts()));
|
|
if (withCustomLoggerFinder) {
|
|
cmds.addAll(List.of("-classpath",
|
|
System.getProperty("test.classes") + File.pathSeparator +
|
|
jarPath1.toString() + File.pathSeparator + jarPath2.toString(),
|
|
"-Dtest.classes=" + System.getProperty("test.classes")));
|
|
} else {
|
|
cmds.addAll(List.of("-classpath",
|
|
System.getProperty("test.classes")));
|
|
}
|
|
|
|
Path patches = Paths.get(System.getProperty("test.classes"), "patches", "java.base");
|
|
cmds.addAll(List.of(
|
|
// patch of EventHelper to log at INFO level (for bootstrap logger)
|
|
"--patch-module", "java.base=" + patches.toString(),
|
|
// allow test to access internal bootstrap logger functionality
|
|
"--add-opens=java.base/jdk.internal.logger=ALL-UNNAMED",
|
|
// following debug property seems useful to tickle the issue
|
|
"-Dsun.misc.URLClassPath.debug=true",
|
|
// console logger level to capture event output
|
|
"-Djdk.system.logger.level=DEBUG",
|
|
// useful for debug purposes
|
|
"-Djdk.logger.finder.error=DEBUG",
|
|
// enable logging to verify correct output
|
|
"-Djava.util.logging.config.file=" +
|
|
Path.of(System.getProperty("test.src", "."), "logging.properties")));
|
|
if (multiThreadLoad) {
|
|
cmds.add("-DmultiThreadLoad=true");
|
|
}
|
|
if (withCustomLoggerFinder) {
|
|
cmds.add("-DwithCustomLoggerFinder=true");
|
|
}
|
|
cmds.addAll(List.of(
|
|
"SignedLoggerFinderTest",
|
|
"no-init"));
|
|
|
|
try {
|
|
OutputAnalyzer outputAnalyzer = ProcessTools.executeCommand(cmds.stream()
|
|
.filter(t -> !t.isEmpty())
|
|
.toArray(String[]::new))
|
|
.shouldHaveExitValue(0);
|
|
if (withCustomLoggerFinder) {
|
|
outputAnalyzer
|
|
.shouldContain("TEST LOGGER: [test_1, test]")
|
|
.shouldContain("TEST LOGGER: [test_2, test]");
|
|
} else {
|
|
outputAnalyzer
|
|
.shouldContain("SecurityPropertyModification: key:test_1")
|
|
.shouldContain("SecurityPropertyModification: key:test_2");
|
|
}
|
|
if (withCustomLoggerFinder && signJars) {
|
|
// X509 cert generated during verification of signed jar file
|
|
outputAnalyzer
|
|
.shouldContain(DNAME);
|
|
}
|
|
|
|
} catch (Throwable t) {
|
|
throw new RuntimeException("Unexpected fail.", t);
|
|
}
|
|
}
|
|
|
|
private static Runnable runnableWithSleep(Supplier s, long sleep, String desc) {
|
|
return () -> {
|
|
while(!testComplete) {
|
|
System.out.println(desc + s.get().getClass().getName());
|
|
try {
|
|
Thread.sleep(sleep);
|
|
} catch (InterruptedException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
private static void initialize() throws Throwable {
|
|
if (signJars) {
|
|
genKey();
|
|
}
|
|
|
|
Path classes = Paths.get(System.getProperty("test.classes", ""));
|
|
JarUtils.createJarFile(jarPath1,
|
|
classes,
|
|
classes.resolve("loggerfinder/SimpleLoggerFinder.class"),
|
|
classes.resolve("loggerfinder/SimpleLoggerFinder$SimpleLogger.class"));
|
|
|
|
JarUtils.updateJarFile(jarPath1, Path.of(System.getProperty("test.src")),
|
|
Path.of("META-INF", "services", "java.lang.System$LoggerFinder"));
|
|
if (signJars) {
|
|
signJar(jarPath1.toString());
|
|
}
|
|
// multiple signed jars with services to have ServiceLoader iteration
|
|
Files.copy(jarPath1, jarPath2, REPLACE_EXISTING);
|
|
}
|
|
|
|
private static void genKey() throws Throwable {
|
|
String keytool = JDKToolFinder.getJDKTool("keytool");
|
|
Files.deleteIfExists(Paths.get(KEYSTORE));
|
|
ProcessTools.executeCommand(keytool,
|
|
"-J-Duser.language=en",
|
|
"-J-Duser.country=US",
|
|
"-genkey",
|
|
"-keyalg", "rsa",
|
|
"-alias", ALIAS,
|
|
"-keystore", KEYSTORE,
|
|
"-keypass", KEYPASS,
|
|
"-dname", DNAME,
|
|
"-storepass", STOREPASS
|
|
).shouldHaveExitValue(0);
|
|
}
|
|
|
|
|
|
private static OutputAnalyzer signJar(String jarName) throws Throwable {
|
|
List<String> args = new ArrayList<>();
|
|
args.add("-verbose");
|
|
args.add(jarName);
|
|
args.add(ALIAS);
|
|
|
|
return jarsigner(args);
|
|
}
|
|
|
|
private static OutputAnalyzer jarsigner(List<String> extra)
|
|
throws Throwable {
|
|
JDKToolLauncher launcher = JDKToolLauncher.createUsingTestJDK("jarsigner")
|
|
.addVMArg("-Duser.language=en")
|
|
.addVMArg("-Duser.country=US")
|
|
.addToolArg("-keystore")
|
|
.addToolArg(KEYSTORE)
|
|
.addToolArg("-storepass")
|
|
.addToolArg(STOREPASS)
|
|
.addToolArg("-keypass")
|
|
.addToolArg(KEYPASS);
|
|
for (String s : extra) {
|
|
if (s.startsWith("-J")) {
|
|
launcher.addVMArg(s.substring(2));
|
|
} else {
|
|
launcher.addToolArg(s);
|
|
}
|
|
}
|
|
return ProcessTools.executeCommand(launcher.getCommand());
|
|
}
|
|
|
|
private static void assertEquals(String received, String expected) {
|
|
if (!expected.equals(received)) {
|
|
throw new RuntimeException("Received: " + received);
|
|
}
|
|
}
|
|
}
|