8295803: Console should be usable in jshell and other environments
Reviewed-by: jlaskey, alanb
This commit is contained in:
parent
5d4c71c8bd
commit
8a9911ef17
@ -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.
|
||||
* <p>
|
||||
* 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;
|
||||
}
|
||||
});
|
||||
}
|
||||
private static Console cons;
|
||||
|
||||
@SuppressWarnings("removal")
|
||||
private static Console instantiateConsole(boolean istty) {
|
||||
try {
|
||||
// Try loading providers
|
||||
PrivilegedAction<Console> 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(
|
||||
|
@ -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
|
||||
|
198
src/java.base/share/classes/java/io/ProxyingConsole.java
Normal file
198
src/java.base/share/classes/java/io/ProxyingConsole.java
Normal file
@ -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
|
||||
}
|
||||
}
|
||||
}
|
@ -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);
|
||||
}
|
||||
|
@ -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();
|
||||
}
|
||||
|
@ -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
|
||||
|
48
src/java.base/share/classes/jdk/internal/io/JdkConsole.java
Normal file
48
src/java.base/share/classes/jdk/internal/io/JdkConsole.java
Normal file
@ -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();
|
||||
}
|
@ -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);
|
||||
}
|
@ -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;
|
||||
|
@ -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);
|
||||
|
@ -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);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -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;
|
||||
}
|
||||
|
||||
|
68
test/jdk/java/io/Console/ModuleSelectionTest.java
Normal file
68
test/jdk/java/io/Console/ModuleSelectionTest.java
Normal file
@ -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);
|
||||
}
|
||||
}
|
||||
}
|
61
test/jdk/java/io/Console/RedirectTest.java
Normal file
61
test/jdk/java/io/Console/RedirectTest.java
Normal file
@ -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);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
35
test/jdk/java/io/Console/SecurityManagerTest.java
Normal file
35
test/jdk/java/io/Console/SecurityManagerTest.java
Normal file
@ -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
|
||||
}
|
||||
}
|
3
test/jdk/java/io/Console/input.txt
Normal file
3
test/jdk/java/io/Console/input.txt
Normal file
@ -0,0 +1,3 @@
|
||||
This is line 1
|
||||
This is line 2
|
||||
This is the last line
|
3
test/jdk/java/io/Console/test.policy
Normal file
3
test/jdk/java/io/Console/test.policy
Normal file
@ -0,0 +1,3 @@
|
||||
grant {
|
||||
permission java.io.FilePermission "<<ALL FILES>>","read,write,delete";
|
||||
};
|
Loading…
Reference in New Issue
Block a user