2023-02-21 15:37:08 +00:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
import java.io.BufferedReader;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.nio.file.Path;
|
|
|
|
import java.util.List;
|
2023-03-03 16:14:45 +00:00
|
|
|
import java.util.logging.LogRecord;
|
|
|
|
import java.util.logging.Logger;
|
|
|
|
import java.util.logging.StreamHandler;
|
2023-02-21 15:37:08 +00:00
|
|
|
import java.util.stream.Stream;
|
|
|
|
|
|
|
|
|
|
|
|
import org.junit.jupiter.params.provider.Arguments;
|
|
|
|
import org.junit.jupiter.params.provider.MethodSource;
|
|
|
|
import org.junit.jupiter.params.ParameterizedTest;
|
|
|
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
|
|
|
import static org.junit.jupiter.api.Assertions.fail;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @summary verify logging of call to System.exit or Runtime.exit.
|
|
|
|
* @run junit/othervm RuntimeExitLogTest
|
|
|
|
*/
|
|
|
|
|
|
|
|
public class RuntimeExitLogTest {
|
|
|
|
|
|
|
|
private static final String TEST_JDK = System.getProperty("test.jdk");
|
|
|
|
private static final String TEST_SRC = System.getProperty("test.src");
|
|
|
|
|
2023-03-03 16:14:45 +00:00
|
|
|
private static Object HOLD_LOGGER;
|
|
|
|
|
2023-02-21 15:37:08 +00:00
|
|
|
/**
|
|
|
|
* Call System.exit() with the parameter (or zero if not supplied).
|
|
|
|
* @param args zero or 1 argument, an exit status
|
|
|
|
*/
|
|
|
|
public static void main(String[] args) throws InterruptedException {
|
|
|
|
int status = args.length > 0 ? Integer.parseInt(args[0]) : 0;
|
2023-03-03 16:14:45 +00:00
|
|
|
if (System.getProperty("ThrowingHandler") != null) {
|
|
|
|
HOLD_LOGGER = ThrowingHandler.installHandler();
|
|
|
|
}
|
2023-02-21 15:37:08 +00:00
|
|
|
System.exit(status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test various log level settings, and none.
|
|
|
|
* @return a stream of arguments for parameterized test
|
|
|
|
*/
|
|
|
|
private static Stream<Arguments> logParamProvider() {
|
|
|
|
return Stream.of(
|
|
|
|
// Logging enabled with level DEBUG
|
|
|
|
Arguments.of(List.of("-Djava.util.logging.config.file=" +
|
2023-03-03 16:14:45 +00:00
|
|
|
Path.of(TEST_SRC, "ExitLogging-FINE.properties").toString()), 1,
|
|
|
|
"Runtime.exit() called with status: 1"),
|
2023-02-21 15:37:08 +00:00
|
|
|
// Logging disabled due to level
|
|
|
|
Arguments.of(List.of("-Djava.util.logging.config.file=" +
|
2023-03-03 16:14:45 +00:00
|
|
|
Path.of(TEST_SRC, "ExitLogging-INFO.properties").toString()), 2,
|
|
|
|
""),
|
2023-02-21 15:37:08 +00:00
|
|
|
// Console logger
|
|
|
|
Arguments.of(List.of("--limit-modules", "java.base",
|
2023-03-03 16:14:45 +00:00
|
|
|
"-Djdk.system.logger.level=DEBUG"), 3,
|
|
|
|
"Runtime.exit() called with status: 3"),
|
2023-02-21 15:37:08 +00:00
|
|
|
// Console logger
|
2023-03-03 16:14:45 +00:00
|
|
|
Arguments.of(List.of(), 4, ""),
|
|
|
|
// Throwing Handler
|
|
|
|
Arguments.of(List.of("-DThrowingHandler",
|
|
|
|
"-Djava.util.logging.config.file=" +
|
|
|
|
Path.of(TEST_SRC, "ExitLogging-FINE.properties").toString()), 5,
|
|
|
|
"Runtime.exit(5) logging failed: Exception in publish")
|
|
|
|
);
|
2023-02-21 15:37:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check that the logger output of a launched process contains the expected message.
|
|
|
|
* @param logProps The name of the log.properties file to set on the command line
|
|
|
|
* @param status the expected exit status of the process
|
2023-03-03 16:14:45 +00:00
|
|
|
* @param expectMessage log should contain the message
|
2023-02-21 15:37:08 +00:00
|
|
|
*/
|
|
|
|
@ParameterizedTest
|
|
|
|
@MethodSource("logParamProvider")
|
2023-03-03 16:14:45 +00:00
|
|
|
public void checkLogger(List<String> logProps, int status, String expectMessage) {
|
2023-02-21 15:37:08 +00:00
|
|
|
ProcessBuilder pb = new ProcessBuilder();
|
|
|
|
pb.redirectErrorStream(true);
|
|
|
|
|
|
|
|
List<String> cmd = pb.command();
|
|
|
|
cmd.add(Path.of(TEST_JDK,"bin", "java").toString());
|
|
|
|
cmd.addAll(logProps);
|
|
|
|
cmd.add(this.getClass().getName());
|
|
|
|
cmd.add(Integer.toString(status));
|
|
|
|
|
|
|
|
try {
|
|
|
|
Process process = pb.start();
|
|
|
|
try (BufferedReader reader = process.inputReader()) {
|
|
|
|
List<String> lines = reader.lines().toList();
|
2023-03-03 16:14:45 +00:00
|
|
|
boolean match = (expectMessage.isEmpty())
|
|
|
|
? lines.size() == 0
|
|
|
|
: lines.stream().filter(s -> s.contains(expectMessage)).findFirst().isPresent();
|
|
|
|
if (!match) {
|
|
|
|
// Output lines for debug
|
|
|
|
System.err.println("Expected: \"" + expectMessage + "\"");
|
|
|
|
System.err.println("---- Actual output begin");
|
2023-02-21 15:37:08 +00:00
|
|
|
lines.forEach(l -> System.err.println(l));
|
2023-03-03 16:14:45 +00:00
|
|
|
System.err.println("---- Actual output end");
|
2023-02-21 15:37:08 +00:00
|
|
|
fail("Unexpected log contents");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int result = process.waitFor();
|
|
|
|
assertEquals(status, result, "Exit status");
|
|
|
|
} catch (IOException | InterruptedException ex) {
|
|
|
|
fail(ex);
|
|
|
|
}
|
|
|
|
}
|
2023-03-03 16:14:45 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A LoggingHandler that throws an Exception.
|
|
|
|
*/
|
|
|
|
public static class ThrowingHandler extends StreamHandler {
|
|
|
|
|
|
|
|
// Install this handler for java.lang.Runtime
|
|
|
|
public static Logger installHandler() {
|
|
|
|
Logger logger = Logger.getLogger("java.lang.Runtime");
|
|
|
|
logger.addHandler(new ThrowingHandler());
|
|
|
|
return logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public synchronized void publish(LogRecord record) {
|
|
|
|
super.publish(record);
|
|
|
|
throw new RuntimeException("Exception in publish");
|
|
|
|
}
|
|
|
|
}
|
2023-02-21 15:37:08 +00:00
|
|
|
}
|