/* * Copyright (c) 2001, 2022, 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 4402649 * @summary RMI should use new logging APIs. Unit test to exercise * RMI's use of the java.util.logging API. * @author Laird Dornin * * @library ../../../../../java/rmi/testlibrary * @modules java.rmi/sun.rmi.registry * java.rmi/sun.rmi.server * java.rmi/sun.rmi.transport * java.rmi/sun.rmi.transport.tcp * @build TestLibrary * @run main/othervm -Djava.security.manager=allow CheckLogging */ import java.util.logging.Level; import java.util.logging.LogRecord; import java.util.logging.Logger; import java.util.logging.SimpleFormatter; import java.util.logging.StreamHandler; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.PrintStream; import java.io.OutputStream; import java.rmi.RemoteException; import java.rmi.Remote; import java.rmi.Naming; import java.rmi.registry.LocateRegistry; import java.rmi.server.LogStream; import java.rmi.server.RemoteServer; import java.rmi.registry.Registry; /** * Perform following checks: * * 1. If using java.util.logging, turn on client call logger using * system property, "sun.rmi.client.logCalls". Collect client call * output using a custom stream handler. Verify client call output is * generated and contains the string "outbound call". * * 2. Turn on server call using * RemoteServer.setLog(ByteArrayOutputStream). Invoke some remote * method calls verify logger output is non-null. * * Turn off server call log by doing setLog(null), verify output is * zero length. Verify that RemoteServer.getLog == null * * Use setLog to turn call log back on. Invoke remote method that * throws an exception and contains the string "exception". * * 3. Print directly to return value of RemoteServer.getLog(), verify * logger output is non-null. */ public class CheckLogging { private static int REGISTRY_PORT = -1; private static String LOCATION; private static Logger logger; private static final ByteArrayOutputStream clientCallOut = new ByteArrayOutputStream(); private static final boolean usingOld = Boolean.getBoolean("sun.rmi.log.useOld"); static { System.setProperty("sun.rmi.client.logCalls", "true"); if (usingOld) { System.err.println("set default stream"); LogStream.setDefaultStream(new PrintStream(clientCallOut)); } else { logger = Logger.getLogger("sun.rmi.client.call"); logger.addHandler(new InternalStreamHandler(clientCallOut)); } } /* use registry to generate client & server call log info */ private static Registry registry; static { try { registry = TestLibrary.createRegistryOnEphemeralPort(); REGISTRY_PORT = TestLibrary.getRegistryPort(registry); LOCATION = "rmi://localhost:" + REGISTRY_PORT + "/"; } catch (Exception e) { TestLibrary.bomb("could not create registry", e); } } /** * Used to collect output from specific loggers */ private static class InternalStreamHandler extends StreamHandler { private InternalStreamHandler(OutputStream out) { super(out, new SimpleFormatter()); setLevel(Level.ALL); } public void publish(LogRecord record) { super.publish(record); flush(); } public void close() { flush(); } } /** * Ensure that a log has some output and that it contains a * certain string */ private static void verifyLog(ByteArrayOutputStream bout, String mustContain) { byte[] bytes = bout.toByteArray(); if (bytes.length == 0) { TestLibrary.bomb("log data length is zero"); } else if ((mustContain != null) && (bout.toString().indexOf(mustContain) < 0)) { TestLibrary.bomb("log output did not contain: " + mustContain); } } /** * Check serverCallLog output */ private static void checkServerCallLog() throws Exception { ByteArrayOutputStream serverCallLog = new ByteArrayOutputStream(); RemoteServer.setLog(serverCallLog); Naming.list(LOCATION); verifyLog(serverCallLog, "list"); serverCallLog.reset(); RemoteServer.setLog(null); PrintStream callStream = RemoteServer.getLog(); if (callStream != null) { TestLibrary.bomb("call stream not null after calling " + "setLog(null)"); } else { System.err.println("call stream should be null and it is"); } Naming.list(LOCATION); if (usingOld) { if (serverCallLog.toString().indexOf("UnicastServerRef") >= 0) { TestLibrary.bomb("server call logging not turned off"); } } else if (serverCallLog.toByteArray().length != 0) { TestLibrary.bomb("call log contains output but it " + "should be empty"); } serverCallLog.reset(); RemoteServer.setLog(serverCallLog); try { // generates a notbound exception Naming.lookup(LOCATION + "notthere"); } catch (Exception e) { } verifyLog(serverCallLog, "exception"); serverCallLog.reset(); RemoteServer.setLog(serverCallLog); callStream = RemoteServer.getLog(); callStream.println("bingo, this is a getLog test"); verifyLog(serverCallLog, "bingo"); } private static void checkPermissions() { SecurityException ex = null; try { // should fail for lack of LoggingPermission "control" RemoteServer.setLog(System.err); } catch (SecurityException e) { System.err.println("security excepton caught correctly"); ex = e; } if (ex == null) { TestLibrary.bomb("able to set log without permission"); } } public static void main(String[] args) { try { checkServerCallLog(); if (!usingOld) { verifyLog(clientCallOut, "outbound call"); System.setSecurityManager(new java.lang.SecurityManager()); checkPermissions(); } System.err.println("TEST PASSED"); } catch (Exception e) { if (e instanceof RuntimeException) { throw (RuntimeException) e; } TestLibrary.bomb("unexpected exception", e); } finally { TestLibrary.unexport(registry); } } }