From 8a9911ef1762ae837e427ec9d91b1399ba33b6e4 Mon Sep 17 00:00:00 2001 From: Naoto Sato Date: Wed, 7 Dec 2022 20:49:29 +0000 Subject: [PATCH] 8295803: Console should be usable in jshell and other environments Reviewed-by: jlaskey, alanb --- .../share/classes/java/io/Console.java | 49 +++-- .../share/classes/java/io/PrintWriter.java | 9 + .../classes/java/io/ProxyingConsole.java | 198 ++++++++++++++++++ .../share/classes/java/lang/System.java | 10 +- .../jdk/internal/access/JavaIOAccess.java | 4 +- .../jdk/internal/access/JavaLangAccess.java | 7 + .../classes/jdk/internal/io/JdkConsole.java | 48 +++++ .../jdk/internal/io/JdkConsoleProvider.java | 50 +++++ src/java.base/share/classes/module-info.java | 4 + .../classes/sun/security/util/Password.java | 11 +- .../org/jline/JdkConsoleProviderImpl.java | 131 ++++++++++++ .../share/classes/module-info.java | 5 +- .../java/io/Console/ModuleSelectionTest.java | 68 ++++++ test/jdk/java/io/Console/RedirectTest.java | 61 ++++++ .../java/io/Console/SecurityManagerTest.java | 35 ++++ test/jdk/java/io/Console/input.txt | 3 + test/jdk/java/io/Console/test.policy | 3 + 17 files changed, 673 insertions(+), 23 deletions(-) create mode 100644 src/java.base/share/classes/java/io/ProxyingConsole.java create mode 100644 src/java.base/share/classes/jdk/internal/io/JdkConsole.java create mode 100644 src/java.base/share/classes/jdk/internal/io/JdkConsoleProvider.java create mode 100644 src/jdk.internal.le/share/classes/jdk/internal/org/jline/JdkConsoleProviderImpl.java create mode 100644 test/jdk/java/io/Console/ModuleSelectionTest.java create mode 100644 test/jdk/java/io/Console/RedirectTest.java create mode 100644 test/jdk/java/io/Console/SecurityManagerTest.java create mode 100644 test/jdk/java/io/Console/input.txt create mode 100644 test/jdk/java/io/Console/test.policy diff --git a/src/java.base/share/classes/java/io/Console.java b/src/java.base/share/classes/java/io/Console.java index 287f7d391f5..0f83a72c23d 100644 --- a/src/java.base/share/classes/java/io/Console.java +++ b/src/java.base/share/classes/java/io/Console.java @@ -25,10 +25,13 @@ package java.io; +import java.security.AccessController; +import java.security.PrivilegedAction; import java.util.*; import java.nio.charset.Charset; import jdk.internal.access.JavaIOAccess; import jdk.internal.access.SharedSecrets; +import jdk.internal.io.JdkConsoleProvider; import jdk.internal.util.StaticProperty; import sun.nio.cs.StreamDecoder; import sun.nio.cs.StreamEncoder; @@ -45,7 +48,7 @@ import sun.security.action.GetPropertyAction; * output streams then its console will exist and will typically be * connected to the keyboard and display from which the virtual machine * was launched. If the virtual machine is started automatically, for - * example by a background job scheduler, then it will typically not + * example by a background job scheduler, then it may not * have a console. *

* If this virtual machine has a console then it is represented by a @@ -93,7 +96,7 @@ import sun.security.action.GetPropertyAction; * @since 1.6 */ -public final class Console implements Flushable +public class Console implements Flushable { /** * Retrieves the unique {@link java.io.PrintWriter PrintWriter} object @@ -592,25 +595,43 @@ public final class Console implements Flushable CHARSET = cs; + cons = instantiateConsole(istty); + // Set up JavaIOAccess in SharedSecrets SharedSecrets.setJavaIOAccess(new JavaIOAccess() { public Console console() { - if (istty) { - if (cons == null) - cons = new Console(); - return cons; - } - return null; - } - - public Charset charset() { - return CHARSET; + return cons; } }); } - private static Console cons; + + @SuppressWarnings("removal") + private static Console instantiateConsole(boolean istty) { + try { + // Try loading providers + PrivilegedAction pa = () -> { + var consModName = System.getProperty("jdk.console", + JdkConsoleProvider.DEFAULT_PROVIDER_MODULE_NAME); + return ServiceLoader.load(ModuleLayer.boot(), JdkConsoleProvider.class).stream() + .map(ServiceLoader.Provider::get) + .filter(jcp -> consModName.equals(jcp.getClass().getModule().getName())) + .map(jcp -> jcp.console(istty, CHARSET)) + .filter(Objects::nonNull) + .findAny() + .map(jc -> (Console) new ProxyingConsole(jc)) + .orElse(istty ? new Console() : null); + }; + return AccessController.doPrivileged(pa); + } catch (ServiceConfigurationError ignore) { + // default to built-in Console + return istty ? new Console() : null; + } + } + + private static final Console cons; private static native boolean istty(); - private Console() { + + Console() { readLock = new Object(); writeLock = new Object(); out = StreamEncoder.forOutputStreamWriter( diff --git a/src/java.base/share/classes/java/io/PrintWriter.java b/src/java.base/share/classes/java/io/PrintWriter.java index a83521c3ffe..667e6c92337 100644 --- a/src/java.base/share/classes/java/io/PrintWriter.java +++ b/src/java.base/share/classes/java/io/PrintWriter.java @@ -208,6 +208,15 @@ public class PrintWriter extends Writer { false); } + /* Package private constructor, using the specified lock + * for synchronization. + */ + PrintWriter(Writer out, Object lock) { + super(lock); + this.out = out; + this.autoFlush = false; + } + /* Private constructor */ private PrintWriter(Charset charset, File file) throws FileNotFoundException diff --git a/src/java.base/share/classes/java/io/ProxyingConsole.java b/src/java.base/share/classes/java/io/ProxyingConsole.java new file mode 100644 index 00000000000..4be206a1590 --- /dev/null +++ b/src/java.base/share/classes/java/io/ProxyingConsole.java @@ -0,0 +1,198 @@ +/* + * Copyright (c) 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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 java.io; + +import java.nio.charset.Charset; +import jdk.internal.io.JdkConsole; + +/** + * Console implementation for internal use. Custom Console delegate may be + * provided with jdk.internal.io.JdkConsoleProvider. + */ +final class ProxyingConsole extends Console { + private final JdkConsole delegate; + private final Object readLock; + private final Object writeLock; + private final Reader reader; + private final PrintWriter printWriter; + + ProxyingConsole(JdkConsole delegate) { + this.delegate = delegate; + readLock = new Object(); + writeLock = new Object(); + reader = new WrappingReader(delegate.reader(), readLock); + printWriter = new WrappingWriter(delegate.writer(), writeLock); + } + + /** + * {@inheritDoc} + */ + @Override + public PrintWriter writer() { + return printWriter; + } + + /** + * {@inheritDoc} + */ + @Override + public Reader reader() { + return reader; + } + + /** + * {@inheritDoc} + */ + @Override + public Console format(String fmt, Object ... args) { + synchronized (writeLock) { + delegate.format(fmt, args); + } + return this; + } + + /** + * {@inheritDoc} + */ + @Override + public Console printf(String format, Object ... args) { + synchronized (writeLock) { + delegate.printf(format, args); + } + return this; + } + + /** + * {@inheritDoc} + */ + @Override + public String readLine(String fmt, Object ... args) { + synchronized (writeLock) { + synchronized (readLock) { + return delegate.readLine(fmt, args); + } + } + } + + /** + * {@inheritDoc} + */ + @Override + public String readLine() { + synchronized (readLock) { + return delegate.readLine(); + } + } + + /** + * {@inheritDoc} + */ + @Override + public char[] readPassword(String fmt, Object ... args) { + synchronized (writeLock) { + synchronized (readLock) { + return delegate.readPassword(fmt, args); + } + } + } + + /** + * {@inheritDoc} + */ + @Override + public char[] readPassword() { + synchronized (readLock) { + return delegate.readPassword(); + } + } + + /** + * {@inheritDoc} + */ + @Override + public void flush() { + delegate.flush(); + } + + /** + * {@inheritDoc} + */ + @Override + public Charset charset() { + return delegate.charset(); + } + + private static class WrappingReader extends Reader { + private final Reader r; + private final Object lock; + + WrappingReader(Reader r, Object lock) { + super(lock); + this.r = r; + this.lock = lock; + } + + @Override + public int read(char[] cbuf, int off, int len) throws IOException { + synchronized (lock) { + return r.read(cbuf, off, len); + } + } + + @Override + public void close() { + // no-op, per Console's spec + } + } + + private static class WrappingWriter extends PrintWriter { + private final PrintWriter pw; + private final Object lock; + + public WrappingWriter(PrintWriter pw, Object lock) { + super(pw, lock); + this.pw = pw; + this.lock = lock; + } + + @Override + public void write(char[] cbuf, int off, int len) { + synchronized (lock) { + pw.write(cbuf, off, len); + } + } + + @Override + public void flush() { + pw.flush(); + } + + @Override + public void close() { + // no-op, per Console's spec + } + } +} diff --git a/src/java.base/share/classes/java/lang/System.java b/src/java.base/share/classes/java/lang/System.java index 455a521d11e..cdb5ec4bcd7 100644 --- a/src/java.base/share/classes/java/lang/System.java +++ b/src/java.base/share/classes/java/lang/System.java @@ -189,6 +189,9 @@ public final class System { */ public static final PrintStream err = null; + // Holder for the initial value of `in`, set within `initPhase1()`. + private static InputStream initialIn; + // indicates if a security manager is possible private static final int NEVER = 1; private static final int MAYBE = 2; @@ -2174,7 +2177,8 @@ public final class System { FileInputStream fdIn = new FileInputStream(FileDescriptor.in); FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out); FileOutputStream fdErr = new FileOutputStream(FileDescriptor.err); - setIn0(new BufferedInputStream(fdIn)); + initialIn = new BufferedInputStream(fdIn); + setIn0(initialIn); // stdout/err.encoding are set when the VM is associated with the terminal, // thus they are equivalent to Console.charset(), otherwise the encodings // of those properties default to native.encoding @@ -2485,6 +2489,10 @@ public final class System { return StringCoding.implEncodeAsciiArray(src, srcOff, dst, dstOff, len); } + public InputStream initialSystemIn() { + return initialIn; + } + public void setCause(Throwable t, Throwable cause) { t.setCause(cause); } diff --git a/src/java.base/share/classes/jdk/internal/access/JavaIOAccess.java b/src/java.base/share/classes/jdk/internal/access/JavaIOAccess.java index b054ec00364..532c1f259d4 100644 --- a/src/java.base/share/classes/jdk/internal/access/JavaIOAccess.java +++ b/src/java.base/share/classes/jdk/internal/access/JavaIOAccess.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2005, 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2005, 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 @@ -26,9 +26,7 @@ package jdk.internal.access; import java.io.Console; -import java.nio.charset.Charset; public interface JavaIOAccess { Console console(); - Charset charset(); } diff --git a/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java b/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java index 261be6bbe13..6ccc8276a48 100644 --- a/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java +++ b/src/java.base/share/classes/jdk/internal/access/JavaLangAccess.java @@ -25,6 +25,7 @@ package jdk.internal.access; +import java.io.InputStream; import java.lang.annotation.Annotation; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodType; @@ -363,6 +364,12 @@ public interface JavaLangAccess { */ int decodeASCII(byte[] src, int srcOff, char[] dst, int dstOff, int len); + /** + * Returns the initial `System.in` to determine if it is replaced + * with `System.setIn(newIn)` method + */ + InputStream initialSystemIn(); + /** * Encodes ASCII codepoints as possible from the source array into * the destination byte array, assuming that the encoding is ASCII diff --git a/src/java.base/share/classes/jdk/internal/io/JdkConsole.java b/src/java.base/share/classes/jdk/internal/io/JdkConsole.java new file mode 100644 index 00000000000..a75941fd2a5 --- /dev/null +++ b/src/java.base/share/classes/jdk/internal/io/JdkConsole.java @@ -0,0 +1,48 @@ +/* + * Copyright (c) 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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 jdk.internal.io; + +import java.io.PrintWriter; +import java.io.Reader; +import java.nio.charset.Charset; + +/** + * Delegate interface for custom Console implementations. + * Methods defined here duplicates the ones in Console class. + * Providers should implement jdk.internal.io.JdkConsoleProvider + * to instantiate an implementation of this interface. + */ +public interface JdkConsole { + PrintWriter writer(); + Reader reader(); + JdkConsole format(String fmt, Object ... args); + JdkConsole printf(String format, Object ... args); + String readLine(String fmt, Object ... args); + String readLine(); + char[] readPassword(String fmt, Object ... args); + char[] readPassword(); + void flush(); + Charset charset(); +} diff --git a/src/java.base/share/classes/jdk/internal/io/JdkConsoleProvider.java b/src/java.base/share/classes/jdk/internal/io/JdkConsoleProvider.java new file mode 100644 index 00000000000..192c1745038 --- /dev/null +++ b/src/java.base/share/classes/jdk/internal/io/JdkConsoleProvider.java @@ -0,0 +1,50 @@ +/* + * Copyright (c) 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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 jdk.internal.io; + +import java.nio.charset.Charset; + +/** + * Service provider interface for JdkConsole implementations. + * The provider used for instantiating JdkConsole instance can be + * specified with the system property "jdk.console", whose value + * designates the module name of the implementation, and which defaults + * to "jdk.internal.le" (jline). If no providers is available, + * or instantiation failed, java.base built-in Console implementation + * is used. + */ +public interface JdkConsoleProvider { + /** + * The module name of the JdkConsole default provider. + */ + String DEFAULT_PROVIDER_MODULE_NAME = "jdk.internal.le"; + + /** + * {@return the Console instance, or {@code null} if not available} + * @param isTTY indicates if the jvm is attached to a terminal + * @param charset charset of the platform console + */ + JdkConsole console(boolean isTTY, Charset charset); +} diff --git a/src/java.base/share/classes/module-info.java b/src/java.base/share/classes/module-info.java index a042ecc22dd..d985dec174f 100644 --- a/src/java.base/share/classes/module-info.java +++ b/src/java.base/share/classes/module-info.java @@ -174,6 +174,9 @@ module java.base { jdk.incubator.vector; exports jdk.internal.event to jdk.jfr; + exports jdk.internal.io to + jdk.internal.le, + jdk.jshell; exports jdk.internal.jimage to jdk.jlink; exports jdk.internal.jimage.decompressor to @@ -409,6 +412,7 @@ module java.base { // JDK-internal service types + uses jdk.internal.io.JdkConsoleProvider; uses jdk.internal.logger.DefaultLoggerFinder; uses sun.text.spi.JavaTimeDateTimePatternProvider; uses sun.util.spi.CalendarProvider; diff --git a/src/java.base/share/classes/sun/security/util/Password.java b/src/java.base/share/classes/sun/security/util/Password.java index eba907d0c86..7acece65a57 100644 --- a/src/java.base/share/classes/sun/security/util/Password.java +++ b/src/java.base/share/classes/sun/security/util/Password.java @@ -29,6 +29,7 @@ import java.io.*; import java.nio.*; import java.nio.charset.*; import java.util.Arrays; +import jdk.internal.access.SharedSecrets; /** * A utility class for reading passwords @@ -51,13 +52,15 @@ public class Password { byte[] consoleBytes = null; try { - // Use the new java.io.Console class + // Only use Console if `in` is the initial System.in Console con; - if (!isEchoOn && in == System.in && ((con = System.console()) != null)) { + if (!isEchoOn && + in == SharedSecrets.getJavaLangAccess().initialSystemIn() && + ((con = System.console()) != null)) { consoleEntered = con.readPassword(); - // readPassword returns "" if you just print ENTER, + // readPassword returns "" if you just press ENTER with the built-in Console, // to be compatible with old Password class, change to null - if (consoleEntered != null && consoleEntered.length == 0) { + if (consoleEntered == null || consoleEntered.length == 0) { return null; } consoleBytes = convertToBytes(consoleEntered); diff --git a/src/jdk.internal.le/share/classes/jdk/internal/org/jline/JdkConsoleProviderImpl.java b/src/jdk.internal.le/share/classes/jdk/internal/org/jline/JdkConsoleProviderImpl.java new file mode 100644 index 00000000000..2cafa56a30b --- /dev/null +++ b/src/jdk.internal.le/share/classes/jdk/internal/org/jline/JdkConsoleProviderImpl.java @@ -0,0 +1,131 @@ +/* + * Copyright (c) 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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 jdk.internal.org.jline; + +import java.io.IOException; +import java.io.PrintWriter; +import java.io.Reader; +import java.io.UncheckedIOException; +import java.nio.charset.Charset; + +import jdk.internal.io.JdkConsole; +import jdk.internal.io.JdkConsoleProvider; +import jdk.internal.org.jline.reader.EndOfFileException; +import jdk.internal.org.jline.reader.LineReader; +import jdk.internal.org.jline.reader.LineReaderBuilder; +import jdk.internal.org.jline.terminal.Terminal; +import jdk.internal.org.jline.terminal.TerminalBuilder; + +/** + * JdkConsole/Provider implementations for jline + */ +public class JdkConsoleProviderImpl implements JdkConsoleProvider { + + /** + * {@inheritDoc} + */ + @Override + public JdkConsole console(boolean isTTY, Charset charset) { + return new JdkConsoleImpl(charset); + } + + /** + * An implementation of JdkConsole, which act as a delegate for the + * public Console class. + */ + private static class JdkConsoleImpl implements JdkConsole { + @Override + public PrintWriter writer() { + return terminal.writer(); + } + + @Override + public Reader reader() { + return terminal.reader(); + } + + @Override + public JdkConsole format(String fmt, Object ... args) { + writer().format(fmt, args).flush(); + return this; + } + + @Override + public JdkConsole printf(String format, Object ... args) { + return format(format, args); + } + + @Override + public String readLine(String fmt, Object ... args) { + try { + return jline.readLine(fmt.formatted(args)); + } catch (EndOfFileException eofe) { + return null; + } + } + + @Override + public String readLine() { + return readLine(""); + } + + @Override + public char[] readPassword(String fmt, Object ... args) { + try { + return jline.readLine(fmt.formatted(args), '\0').toCharArray(); + } catch (EndOfFileException eofe) { + return null; + } + } + + @Override + public char[] readPassword() { + return readPassword(""); + } + + @Override + public void flush() { + terminal.flush(); + } + + @Override + public Charset charset() { + return terminal.encoding(); + } + + private final LineReader jline; + private final Terminal terminal; + + public JdkConsoleImpl(Charset charset) { + try { + terminal = TerminalBuilder.builder().encoding(charset).build(); + jline = LineReaderBuilder.builder().terminal(terminal).build(); + } catch (IOException ioe) { + throw new UncheckedIOException(ioe); + } + } + } +} diff --git a/src/jdk.internal.le/share/classes/module-info.java b/src/jdk.internal.le/share/classes/module-info.java index a47e07cd784..127bd7aeb78 100644 --- a/src/jdk.internal.le/share/classes/module-info.java +++ b/src/jdk.internal.le/share/classes/module-info.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2018, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 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 @@ -50,5 +50,8 @@ module jdk.internal.le { uses jdk.internal.org.jline.terminal.spi.JnaSupport; + // Console + provides jdk.internal.io.JdkConsoleProvider with + jdk.internal.org.jline.JdkConsoleProviderImpl; } diff --git a/test/jdk/java/io/Console/ModuleSelectionTest.java b/test/jdk/java/io/Console/ModuleSelectionTest.java new file mode 100644 index 00000000000..c7e66d0c317 --- /dev/null +++ b/test/jdk/java/io/Console/ModuleSelectionTest.java @@ -0,0 +1,68 @@ +/* + * Copyright (c) 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 8295803 + * @summary Tests System.console() returns correct Console (or null) from the expected + * module. + * @modules java.base/java.io:+open + * @run main/othervm ModuleSelectionTest jdk.internal.le + * @run main/othervm -Djdk.console=jdk.internal.le ModuleSelectionTest jdk.internal.le + * @run main/othervm -Djdk.console=java.base ModuleSelectionTest java.base + * @run main/othervm --limit-modules java.base ModuleSelectionTest java.base + */ + +import java.io.Console; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; + +public class ModuleSelectionTest { + public static void main(String... args) throws Throwable { + var con = System.console(); + var pc = Class.forName("java.io.ProxyingConsole"); + var jdkc = Class.forName("jdk.internal.io.JdkConsole"); + var istty = (boolean)MethodHandles.privateLookupIn(Console.class, MethodHandles.lookup()) + .findStatic(Console.class, "istty", MethodType.methodType(boolean.class)) + .invoke(); + var impl = con != null ? MethodHandles.privateLookupIn(pc, MethodHandles.lookup()) + .findGetter(pc, "delegate", jdkc) + .invoke(con) : null; + + var expected = switch (args[0]) { + case "java.base" -> istty ? "java.base" : "null"; + default -> args[0]; + }; + var actual = con == null ? "null" : impl.getClass().getModule().getName(); + + if (!actual.equals(expected)) { + throw new RuntimeException(""" + Console implementation is not the expected one. + Expected: %s + Actual: %s + """.formatted(expected, actual)); + } else { + System.out.printf("%s is the expected implementation. (tty: %s)\n", impl, istty); + } + } +} diff --git a/test/jdk/java/io/Console/RedirectTest.java b/test/jdk/java/io/Console/RedirectTest.java new file mode 100644 index 00000000000..8b55bd488d7 --- /dev/null +++ b/test/jdk/java/io/Console/RedirectTest.java @@ -0,0 +1,61 @@ +/* + * Copyright (c) 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. + */ + +import java.io.File; +import java.nio.file.Files; + +import jdk.test.lib.process.ProcessTools; + +/** + * @test + * @bug 8295803 + * @summary Tests System.console() works with standard input redirection. + * @library /test/lib + */ +public class RedirectTest { + public static void main(String... args) throws Throwable { + if (args.length == 0) { + // no arg will launch the child process that actually perform tests + var pb = ProcessTools.createTestJvm("RedirectTest", "dummy"); + var input = new File(System.getProperty("test.src", "."), "input.txt"); + pb.redirectInput(input); + var oa = ProcessTools.executeProcess(pb); + var output = oa.asLines(); + var expected = Files.readAllLines(input.toPath()); + if (!output.equals(expected)) { + throw new RuntimeException(""" + Standard out had unexpected strings: + Actual output: %s + Expected output: %s + """.formatted(output, expected)); + } + oa.shouldHaveExitValue(0); + } else { + var con = System.console(); + String line; + while ((line = con.readLine()) != null) { + System.out.println(line); + } + } + } +} diff --git a/test/jdk/java/io/Console/SecurityManagerTest.java b/test/jdk/java/io/Console/SecurityManagerTest.java new file mode 100644 index 00000000000..5eed29a7669 --- /dev/null +++ b/test/jdk/java/io/Console/SecurityManagerTest.java @@ -0,0 +1,35 @@ +/* + * Copyright (c) 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 8295803 + * @summary Tests System.console() works with the security manager + * @run main/othervm/java.security.policy=test.policy -Djava.security.manager -Djdk.console=jdk.internal.le SecurityManagerTest + */ +public class SecurityManagerTest { + public static void main(String... args) { + System.console(); + // consider it successful if ServiceConfigurationError was not thrown here + } +} diff --git a/test/jdk/java/io/Console/input.txt b/test/jdk/java/io/Console/input.txt new file mode 100644 index 00000000000..9387db2c697 --- /dev/null +++ b/test/jdk/java/io/Console/input.txt @@ -0,0 +1,3 @@ +This is line 1 +This is line 2 +This is the last line diff --git a/test/jdk/java/io/Console/test.policy b/test/jdk/java/io/Console/test.policy new file mode 100644 index 00000000000..61fda0ca63a --- /dev/null +++ b/test/jdk/java/io/Console/test.policy @@ -0,0 +1,3 @@ +grant { + permission java.io.FilePermission "<>","read,write,delete"; +};