diff --git a/src/java.logging/share/classes/java/util/logging/LogManager.java b/src/java.logging/share/classes/java/util/logging/LogManager.java index e864f5ebabf..736f07694f0 100644 --- a/src/java.logging/share/classes/java/util/logging/LogManager.java +++ b/src/java.logging/share/classes/java/util/logging/LogManager.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2018, 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 @@ -396,12 +396,6 @@ public class LogManager { // Create and retain Logger for the root of the namespace. owner.addLogger(root); - // For backward compatibility: add any handlers configured using - // ".handlers" - owner.createLoggerHandlers("", ".handlers") - .stream() - .forEach(root::addHandler); - // Initialize level if not yet initialized if (!root.isLevelInitialized()) { root.setLevel(defaultLevel); @@ -995,7 +989,8 @@ public class LogManager { } } - private List createLoggerHandlers(final String name, final String handlersPropertyName) + private List createLoggerHandlers(final String name, + final String handlersPropertyName) { String names[] = parseClassNames(handlersPropertyName); List handlers = new ArrayList<>(names.length); @@ -1198,7 +1193,7 @@ public class LogManager { } drainLoggerRefQueueBounded(); LoggerContext cx = getUserContext(); - if (cx.addLocalLogger(logger)) { + if (cx.addLocalLogger(logger) || forceLoadHandlers(logger)) { // Do we have a per logger handler too? // Note: this will add a 200ms penalty loadLoggerHandlers(logger, name, name + ".handlers"); @@ -1208,6 +1203,26 @@ public class LogManager { } } + + // Checks whether the given logger is a special logger + // that still requires handler initialization. + // This method will only return true for the root and + // global loggers and only if called by the thread that + // performs initialization of the LogManager, during that + // initialization. Must only be called by addLogger. + @SuppressWarnings("deprecation") + private boolean forceLoadHandlers(Logger logger) { + // Called just after reading the primordial configuration, in + // the same thread that reads it. + // The root and global logger would already be present in the context + // by this point, but we would not have called loadLoggerHandlers + // yet. + return (logger == rootLogger || logger == Logger.global) + && !initializationDone + && initializedCalled + && configurationLock.isHeldByCurrentThread(); + } + // Private method to set a level on a logger. // If necessary, we raise privilege before doing the call. private static void doSetLevel(final Logger logger, final Level level) { diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/BadRootLoggerHandlers.java b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/BadRootLoggerHandlers.java new file mode 100644 index 00000000000..70c217f8a51 --- /dev/null +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/BadRootLoggerHandlers.java @@ -0,0 +1,304 @@ +/* + * Copyright (c) 2018, 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.ByteArrayOutputStream; +import java.io.IOException; +import java.io.OutputStream; +import java.io.PrintStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.nio.file.StandardCopyOption; +import java.util.Collections; +import java.util.Enumeration; +import java.util.List; +import java.util.Objects; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentMap; +import java.util.logging.Handler; +import java.util.logging.Level; +import java.util.logging.LogManager; +import java.util.logging.Logger; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +/** + * @test + * @bug 8191033 + * @build custom.DotHandler custom.Handler + * @run main/othervm -Dlogging.properties=badlogging.properties -Dclz=1custom.DotHandler BadRootLoggerHandlers CUSTOM + * @run main/othervm -Dlogging.properties=badlogging.properties -Dclz=1custom.DotHandler BadRootLoggerHandlers DEFAULT + * @run main/othervm -Dlogging.properties=badglobal.properties -Dclz=1custom.GlobalHandler BadRootLoggerHandlers CUSTOM + * @run main/othervm -Dlogging.properties=badglobal.properties -Dclz=1custom.GlobalHandler BadRootLoggerHandlers DEFAULT + * @run main/othervm/java.security.policy==test.policy -Dlogging.properties=badlogging.properties -Dclz=1custom.DotHandler BadRootLoggerHandlers CUSTOM + * @run main/othervm/java.security.policy==test.policy -Dlogging.properties=badlogging.properties -Dclz=1custom.DotHandler BadRootLoggerHandlers DEFAULT + * @run main/othervm/java.security.policy==test.policy -Dlogging.properties=badglobal.properties -Dclz=1custom.GlobalHandler BadRootLoggerHandlers CUSTOM + * @run main/othervm/java.security.policy==test.policy -Dlogging.properties=badglobal.properties -Dclz=1custom.GlobalHandler BadRootLoggerHandlers DEFAULT + * @author danielfuchs + */ +public class BadRootLoggerHandlers { + + public static final Path SRC_DIR = + Paths.get(System.getProperty("test.src", "src")); + public static final Path USER_DIR = + Paths.get(System.getProperty("user.dir", ".")); + public static final Path CONFIG_FILE = Paths.get( + Objects.requireNonNull(System.getProperty("logging.properties"))); + public static final String BAD_HANDLER_NAME = + Objects.requireNonNull(System.getProperty("clz")); + + static enum TESTS { CUSTOM, DEFAULT} + public static final class CustomLogManager extends LogManager { + final ConcurrentMap loggers = new ConcurrentHashMap<>(); + @Override + public boolean addLogger(Logger logger) { + return loggers.putIfAbsent(logger.getName(), logger) == null; + } + + @Override + public Enumeration getLoggerNames() { + return Collections.enumeration(loggers.keySet()); + } + + @Override + public Logger getLogger(String name) { + return loggers.get(name); + } + } + + public static class SystemErr extends OutputStream { + final ByteArrayOutputStream baos = new ByteArrayOutputStream(); + final OutputStream wrapped; + public SystemErr(OutputStream out) { + this.wrapped = out; + } + + @Override + public void write(int b) throws IOException { + baos.write(b); + wrapped.write(b); + } + + public void close() throws IOException { + flush(); + super.close(); + } + + public void flush() throws IOException { + super.flush(); + wrapped.flush(); + } + + } + + // Uncomment this to run the test on Java 8. Java 8 does not have + // List.of(...) + // static final class List { + // static java.util.List of(T... items) { + // return Collections.unmodifiableList(Arrays.asList(items)); + // } + // } + + public static void main(String[] args) throws IOException { + Path initialProps = SRC_DIR.resolve(CONFIG_FILE); + Path loggingProps = USER_DIR.resolve(CONFIG_FILE); + if (args.length != 1) { + throw new IllegalArgumentException("expected (only) one of " + List.of(TESTS.values())); + } + + TESTS test = TESTS.valueOf(args[0]); + System.setProperty("java.util.logging.config.file", loggingProps.toString()); + if (test == TESTS.CUSTOM) { + System.setProperty("java.util.logging.manager", CustomLogManager.class.getName()); + } + + Files.copy(initialProps, loggingProps, StandardCopyOption.REPLACE_EXISTING); + + SystemErr err = new SystemErr(System.err); + System.setErr(new PrintStream(err)); + + System.out.println("Root level is: " + Logger.getLogger("").getLevel()); + if (Logger.getLogger("").getLevel() != Level.INFO) { + throw new RuntimeException("Expected root level INFO, got: " + + Logger.getLogger("").getLevel()); + } + + Class logManagerClass = + LogManager.getLogManager().getClass(); + Class expectedClass = + test == TESTS.CUSTOM ? CustomLogManager.class : LogManager.class; + if (logManagerClass != expectedClass) { + throw new RuntimeException("Bad class for log manager: " + logManagerClass + + " expected " + expectedClass + " for " + test); + } + + if (test == TESTS.DEFAULT) { + // Verify that we have two handlers. One was configured with + // handlers=custom.Handler, the other with + // .handlers=custom.DotHandler + // Verify that exactly one of the two handlers is a custom.Handler + // Verify that exactly one of the two handlers is a custom.DotHandler + // Verify that the two handlers have an id of '1' + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), + 1L, + custom.Handler.class, + custom.DotHandler.class); + } else { + // Verify that we have one handler, configured with + // handlers=custom.Handler. + // Verify that it is a custom.Handler + // Verify that the handler have an id of '1' + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), + 1L, + custom.Handler.class); + + } + + // DEFAULT: The log message "hi" should appear twice on the console. + // CUSTOM: The log message "hi" should appear twice on the console. + // We don't check that. This is just for log analysis in case + // of test failure. + Logger.getAnonymousLogger().info("hi (" + test +")"); + + // Change the root logger level to FINE in the properties file + // and reload the configuration. + Files.write(loggingProps, + Files.lines(initialProps) + .map((s) -> s.replace("INFO", "FINE")) + .collect(Collectors.toList())); + LogManager.getLogManager().readConfiguration(); + + System.out.println("Root level is: " + Logger.getLogger("").getLevel()); + if (Logger.getLogger("").getLevel() != Level.FINE) { + throw new RuntimeException("Expected root level FINE, got: " + + Logger.getLogger("").getLevel()); + } + + // Verify that we have now only one handler, configured with + // handlers=custom.Handler, and that the other configured with + // .handlers=custom.DotHandler was ignored. + // Verify that the handler is a custom.Handler + // Verify that the handler has an id of '2' + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), + 2L, + custom.Handler.class); + + // The log message "there" should appear only once on the console. + // We don't check that. This is just for log analysis in case + // of test failure. + Logger.getAnonymousLogger().info("there!"); + + // Change the root logger level to FINER in the properties file + // and reload the configuration. + Files.write(loggingProps, + Files.lines(initialProps) + .map((s) -> s.replace("INFO", "FINER")) + .collect(Collectors.toList())); + LogManager.getLogManager().readConfiguration(); + + System.out.println("Root level is: " + Logger.getLogger("").getLevel()); + if (Logger.getLogger("").getLevel() != Level.FINER) { + throw new RuntimeException("Expected root level FINER, got: " + + Logger.getLogger("").getLevel()); + } + + // Verify that we have only one handler, configured with + // handlers=custom.Handler, and that the other configured with + // .handlers=custom.DotHandler was ignored. + // Verify that the handler is a custom.Handler + // Verify that the handler has an id of '3' + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), + 3L, + custom.Handler.class); + + // The log message "done" should appear only once on the console. + // We don't check that. This is just for log analysis in case + // of test failure. + Logger.getAnonymousLogger().info("done!"); + + byte[] errBytes = err.baos.toByteArray(); + String errText = new String(errBytes); + switch(test) { + case CUSTOM: + if (errText.contains("java.lang.ClassNotFoundException: " + + BAD_HANDLER_NAME)) { + throw new RuntimeException("Error message found on System.err"); + } + System.out.println("OK: ClassNotFoundException error message not found for " + test); + break; + case DEFAULT: + if (!errText.contains("java.lang.ClassNotFoundException: " + + BAD_HANDLER_NAME)) { + throw new RuntimeException("Error message not found on System.err"); + } + System.err.println("OK: ClassNotFoundException error message found for " + test); + break; + default: + throw new InternalError("unknown test case: " + test); + } + } + + static void checkHandlers(Logger logger, Handler[] handlers, Long expectedID, Class... clz) { + // Verify that we have the expected number of handlers. + if (Stream.of(handlers).count() != clz.length) { + throw new RuntimeException("Expected " + clz.length + " handlers, got: " + + List.of(logger.getHandlers())); + } + for (Class cl : clz) { + // Verify that the handlers are of the expected class. + // For each class, we should have exactly one handler + // of that class. + if (Stream.of(handlers) + .map(Object::getClass) + .filter(cl::equals) + .count() != 1) { + throw new RuntimeException("Expected one " + cl +", got: " + + List.of(logger.getHandlers())); + } + } + // Verify that all handlers have the expected ID + if (Stream.of(logger.getHandlers()) + .map(BadRootLoggerHandlers::getId) + .filter(expectedID::equals) + .count() != clz.length) { + throw new RuntimeException("Expected ids to be " + expectedID + ", got: " + + List.of(logger.getHandlers())); + } + } + + static long getId(Handler h) { + if (h instanceof custom.Handler) { + return ((custom.Handler)h).id; + } + if (h instanceof custom.DotHandler) { + return ((custom.DotHandler)h).id; + } + if (h instanceof custom.GlobalHandler) { + return ((custom.GlobalHandler)h).id; + } + return -1; + } +} diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/RootLoggerHandlers.java b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/RootLoggerHandlers.java index 8b7b841f43d..e01216f0308 100644 --- a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/RootLoggerHandlers.java +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/RootLoggerHandlers.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2017, 2018, 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 @@ -24,7 +24,7 @@ import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; -import java.util.Arrays; +import java.nio.file.StandardCopyOption; import java.util.Collections; import java.util.List; import java.util.logging.Handler; @@ -39,6 +39,7 @@ import java.util.stream.Stream; * @bug 8191033 * @build custom.DotHandler custom.Handler * @run main/othervm RootLoggerHandlers + * @run main/othervm/java.security.policy==test.policy RootLoggerHandlers * @author danielfuchs */ public class RootLoggerHandlers { @@ -61,7 +62,7 @@ public class RootLoggerHandlers { Path initialProps = SRC_DIR.resolve(CONFIG_FILE); Path loggingProps = USER_DIR.resolve(CONFIG_FILE); System.setProperty("java.util.logging.config.file", loggingProps.toString()); - Files.copy(initialProps, loggingProps); + Files.copy(initialProps, loggingProps, StandardCopyOption.REPLACE_EXISTING); System.out.println("Root level is: " + Logger.getLogger("").getLevel()); if (Logger.getLogger("").getLevel() != Level.INFO) { throw new RuntimeException("Expected root level INFO, got: " @@ -73,10 +74,15 @@ public class RootLoggerHandlers { // Verify that exactly one of the two handlers is a custom.Handler // Verify that exactly one of the two handlers is a custom.DotHandler // Verify that the two handlers has an id of '1' - checkHandlers(Logger.getLogger("").getHandlers(), + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), 1L, custom.Handler.class, custom.DotHandler.class); + checkHandlers(Logger.getLogger("global"), + Logger.getGlobal().getHandlers(), + 1L, + custom.GlobalHandler.class); // The log message "hi" should appear twice on the console. // We don't check that. This is just for log analysis in case @@ -102,9 +108,13 @@ public class RootLoggerHandlers { // .handlers=custom.DotHandler was ignored. // Verify that the handler is a custom.Handler // Verify that the handler has an id of '2' - checkHandlers(Logger.getLogger("").getHandlers(), + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), 2L, custom.Handler.class); + checkHandlers(Logger.getGlobal(), + Logger.getGlobal().getHandlers(), + 1L); // The log message "there" should appear only once on the console. // We don't check that. This is just for log analysis in case @@ -130,9 +140,45 @@ public class RootLoggerHandlers { // .handlers=custom.DotHandler was ignored. // Verify that the handler is a custom.Handler // Verify that the handler has an id of '3' - checkHandlers(Logger.getLogger("").getHandlers(), + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), 3L, custom.Handler.class); + checkHandlers(Logger.getGlobal(), + Logger.getGlobal().getHandlers(), + 1L); + + LogManager.getLogManager().reset(); + LogManager.getLogManager().updateConfiguration((s) -> (o,n) -> n); + // Verify that we have only one handler, configured with + // handlers=custom.Handler, and that the other configured with + // .handlers=custom.DotHandler was ignored. + // Verify that the handler is a custom.Handler + // Verify that the handler has an id of '4' + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), + 4L, + custom.Handler.class); + checkHandlers(Logger.getGlobal(), + Logger.getGlobal().getHandlers(), + 2L, + custom.GlobalHandler.class); + + LogManager.getLogManager().updateConfiguration((s) -> (o,n) -> n); + // Verify that we have only one handler, configured with + // handlers=custom.Handler, and that the other configured with + // .handlers=custom.DotHandler was ignored. + // Verify that the handler is a custom.Handler + // Verify that the handler has an id of '4' + checkHandlers(Logger.getLogger(""), + Logger.getLogger("").getHandlers(), + 4L, + custom.Handler.class); + checkHandlers(Logger.getGlobal(), + Logger.getGlobal().getHandlers(), + 2L, + custom.GlobalHandler.class); + // The log message "done" should appear only once on the console. // We don't check that. This is just for log analysis in case @@ -140,11 +186,11 @@ public class RootLoggerHandlers { Logger.getAnonymousLogger().info("done!"); } - static void checkHandlers(Handler[] handlers, Long expectedID, Class... clz) { + static void checkHandlers(Logger logger, Handler[] handlers, Long expectedID, Class... clz) { // Verify that we have the expected number of handlers. if (Stream.of(handlers).count() != clz.length) { throw new RuntimeException("Expected " + clz.length + " handlers, got: " - + List.of(Logger.getLogger("").getHandlers())); + + List.of(logger.getHandlers())); } for (Class cl : clz) { // Verify that the handlers are of the expected class. @@ -155,16 +201,16 @@ public class RootLoggerHandlers { .filter(cl::equals) .count() != 1) { throw new RuntimeException("Expected one " + cl +", got: " - + List.of(Logger.getLogger("").getHandlers())); + + List.of(logger.getHandlers())); } } // Verify that all handlers have the expected ID - if (Stream.of(Logger.getLogger("").getHandlers()) + if (Stream.of(logger.getHandlers()) .map(RootLoggerHandlers::getId) .filter(expectedID::equals) .count() != clz.length) { throw new RuntimeException("Expected ids to be " + expectedID + ", got: " - + List.of(Logger.getLogger("").getHandlers())); + + List.of(logger.getHandlers())); } } @@ -175,6 +221,9 @@ public class RootLoggerHandlers { if (h instanceof custom.DotHandler) { return ((custom.DotHandler)h).id; } + if (h instanceof custom.GlobalHandler) { + return ((custom.GlobalHandler)h).id; + } return -1; } } diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/badglobal.properties b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/badglobal.properties new file mode 100644 index 00000000000..5e4e18f3197 --- /dev/null +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/badglobal.properties @@ -0,0 +1,19 @@ +############################################################ +# Global properties +############################################################ + +# "handlers" specifies a comma separated list of log Handler +# classes. These handlers will be installed during VM startup. +#handlers= java.util.logging.ConsoleHandler +handlers= custom.Handler +.handlers= custom.DotHandler +global.handlers = 1custom.GlobalHandler, custom.GlobalHandler + +# Default global logging level. +.level= INFO + +# Other configuration +custom.Handler.level=ALL +custom.DotHandler.level=ALL +java.util.logging.SimpleFormatter.format=%4$s [%1$tc]: %2$s: %5$s%n + diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/badlogging.properties b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/badlogging.properties new file mode 100644 index 00000000000..20b731af5a4 --- /dev/null +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/badlogging.properties @@ -0,0 +1,18 @@ +############################################################ +# Global properties +############################################################ + +# "handlers" specifies a comma separated list of log Handler +# classes. These handlers will be installed during VM startup. +#handlers= java.util.logging.ConsoleHandler +handlers= custom.Handler +.handlers= 1custom.DotHandler,custom.DotHandler + +# Default global logging level. +.level= INFO + +# Other configuration +custom.Handler.level=ALL +custom.DotHandler.level=ALL +java.util.logging.SimpleFormatter.format=%4$s [%1$tc]: %2$s: %5$s%n + diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/custom/GlobalHandler.java b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/custom/GlobalHandler.java new file mode 100644 index 00000000000..b71571d6785 --- /dev/null +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/custom/GlobalHandler.java @@ -0,0 +1,50 @@ +/* + * Copyright (c) 2018, 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. + */ +package custom; + +import java.util.concurrent.atomic.AtomicLong; + +/** + * + * @author danielfuchs + */ +public class GlobalHandler extends java.util.logging.ConsoleHandler { + + public static final AtomicLong IDS = new AtomicLong(); + public final long id = IDS.incrementAndGet(); + public GlobalHandler() { + System.out.println("GlobalHandler(" + id + ") created"); + //new Exception("GlobalHandler").printStackTrace(); + } + + @Override + public void close() { + System.out.println("GlobalHandler(" + id + ") closed"); + super.close(); + } + + @Override + public String toString() { + return this.getClass().getName() + '(' + id + ')'; + } +} diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/logging.properties b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/logging.properties index a432c19a810..fe302b5d99f 100644 --- a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/logging.properties +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/logging.properties @@ -7,6 +7,7 @@ #handlers= java.util.logging.ConsoleHandler handlers= custom.Handler .handlers= custom.DotHandler +global.handlers= custom.GlobalHandler # Default global logging level. .level= INFO @@ -14,5 +15,6 @@ handlers= custom.Handler # Other configuration custom.Handler.level=ALL custom.DotHandler.level=ALL +custom.GlobalHandler.level=ALL java.util.logging.SimpleFormatter.format=%4$s [%1$tc]: %2$s: %5$s%n diff --git a/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/test.policy b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/test.policy new file mode 100644 index 00000000000..1929792d426 --- /dev/null +++ b/test/jdk/java/util/logging/LogManager/Configuration/rootLoggerHandlers/test.policy @@ -0,0 +1,13 @@ +grant { + // permissions for RootLoggerHandlers + permission java.util.PropertyPermission "*", "read"; + permission java.util.PropertyPermission "java.util.logging.config.file", "write"; + permission java.io.FilePermission "<>", "read,write,delete"; + permission java.util.logging.LoggingPermission "control"; + + // additional permissions for BadRootLoggerHandlers + permission java.util.PropertyPermission "java.util.logging.manager", "write"; + permission java.lang.RuntimePermission "setIO"; + permission java.lang.RuntimePermission "shutdownHooks"; + permission java.lang.RuntimePermission "setContextClassLoader"; +};