8247403: JShell: No custom input (e.g. from GUI) possible with JavaShellToolBuilder
Reviewed-by: vromero
This commit is contained in:
parent
64f0f68958
commit
2c8e94f680
@ -4157,7 +4157,7 @@ public class LineReaderImpl implements LineReader, Flushable
|
||||
} else
|
||||
sb.append(ch);
|
||||
}
|
||||
if (padToWidth > cols) {
|
||||
if (padToWidth > cols && padToWidth > 0) {
|
||||
int padCharCols = WCWidth.wcwidth(padChar);
|
||||
int padCount = (padToWidth - cols) / padCharCols;
|
||||
sb = padPartString;
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2015, 2021, 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
|
||||
@ -47,6 +47,7 @@ import java.util.List;
|
||||
import java.util.ListIterator;
|
||||
import java.util.Map;
|
||||
import java.util.Optional;
|
||||
import java.util.function.Consumer;
|
||||
import java.util.function.Function;
|
||||
import java.util.stream.Collectors;
|
||||
import java.util.stream.Stream;
|
||||
@ -103,8 +104,8 @@ class ConsoleIOContext extends IOContext {
|
||||
|
||||
String prefix = "";
|
||||
|
||||
ConsoleIOContext(JShellTool repl, InputStream cmdin, PrintStream cmdout) throws Exception {
|
||||
this.allowIncompleteInputs = Boolean.getBoolean("jshell.test.allow.incomplete.inputs");
|
||||
ConsoleIOContext(JShellTool repl, InputStream cmdin, PrintStream cmdout,
|
||||
boolean interactive) throws Exception {
|
||||
this.repl = repl;
|
||||
Map<String, Object> variables = new HashMap<>();
|
||||
this.input = new StopDetectingInputStream(() -> repl.stop(),
|
||||
@ -116,8 +117,20 @@ class ConsoleIOContext extends IOContext {
|
||||
}
|
||||
};
|
||||
Terminal terminal;
|
||||
if (System.getProperty("test.jdk") != null) {
|
||||
terminal = new TestTerminal(nonBlockingInput, cmdout);
|
||||
boolean allowIncompleteInputs = Boolean.getBoolean("jshell.test.allow.incomplete.inputs");
|
||||
Consumer<LineReaderImpl> setupReader = r -> {};
|
||||
if (cmdin != System.in) {
|
||||
if (System.getProperty("test.jdk") != null) {
|
||||
terminal = new TestTerminal(nonBlockingInput, cmdout);
|
||||
} else {
|
||||
Size size = null;
|
||||
terminal = new ProgrammaticInTerminal(nonBlockingInput, cmdout, interactive,
|
||||
size);
|
||||
if (!interactive) {
|
||||
setupReader = r -> r.unsetOpt(Option.BRACKETED_PASTE);
|
||||
allowIncompleteInputs = true;
|
||||
}
|
||||
}
|
||||
input.setInputStream(cmdin);
|
||||
} else {
|
||||
terminal = TerminalBuilder.builder().inputStreamWrapper(in -> {
|
||||
@ -125,6 +138,7 @@ class ConsoleIOContext extends IOContext {
|
||||
return nonBlockingInput;
|
||||
}).build();
|
||||
}
|
||||
this.allowIncompleteInputs = allowIncompleteInputs;
|
||||
originalAttributes = terminal.getAttributes();
|
||||
Attributes noIntr = new Attributes(originalAttributes);
|
||||
noIntr.setControlChar(ControlChar.VINTR, 0);
|
||||
@ -179,10 +193,11 @@ class ConsoleIOContext extends IOContext {
|
||||
}
|
||||
};
|
||||
|
||||
setupReader.accept(reader);
|
||||
reader.setOpt(Option.DISABLE_EVENT_EXPANSION);
|
||||
|
||||
reader.setParser((line, cursor, context) -> {
|
||||
if (!allowIncompleteInputs && !repl.isComplete(line)) {
|
||||
if (!ConsoleIOContext.this.allowIncompleteInputs && !repl.isComplete(line)) {
|
||||
int pendingBraces = countPendingOpenBraces(line);
|
||||
throw new EOFError(cursor, cursor, line, null, pendingBraces, null);
|
||||
}
|
||||
@ -230,7 +245,7 @@ class ConsoleIOContext extends IOContext {
|
||||
this.prefix = prefix;
|
||||
try {
|
||||
in.setVariable(LineReader.SECONDARY_PROMPT_PATTERN, continuationPrompt);
|
||||
return in.readLine(firstLinePrompt);
|
||||
return in.readLine(firstLine ? firstLinePrompt : continuationPrompt);
|
||||
} catch (UserInterruptException ex) {
|
||||
throw (InputInterruptedException) new InputInterruptedException().initCause(ex);
|
||||
} catch (EndOfFileException ex) {
|
||||
@ -1276,28 +1291,31 @@ class ConsoleIOContext extends IOContext {
|
||||
return in.getHistory();
|
||||
}
|
||||
|
||||
private static final class TestTerminal extends LineDisciplineTerminal {
|
||||
private static class ProgrammaticInTerminal extends LineDisciplineTerminal {
|
||||
|
||||
private static final int DEFAULT_HEIGHT = 24;
|
||||
protected static final int DEFAULT_HEIGHT = 24;
|
||||
|
||||
private final NonBlockingReader inputReader;
|
||||
private final Size bufferSize;
|
||||
|
||||
public TestTerminal(InputStream input, OutputStream output) throws Exception {
|
||||
super("test", "ansi", output, Charset.forName("UTF-8"));
|
||||
public ProgrammaticInTerminal(InputStream input, OutputStream output,
|
||||
boolean interactive, Size size) throws Exception {
|
||||
this(input, output, interactive ? "ansi" : "dumb",
|
||||
size != null ? size : new Size(80, DEFAULT_HEIGHT),
|
||||
size != null ? size
|
||||
: interactive ? new Size(80, DEFAULT_HEIGHT)
|
||||
: new Size(Integer.MAX_VALUE - 1, DEFAULT_HEIGHT));
|
||||
}
|
||||
|
||||
protected ProgrammaticInTerminal(InputStream input, OutputStream output,
|
||||
String terminal, Size size, Size bufferSize) throws Exception {
|
||||
super("non-system-in", terminal, output, Charset.forName("UTF-8"));
|
||||
this.inputReader = NonBlocking.nonBlocking(getName(), input, encoding());
|
||||
Attributes a = new Attributes(getAttributes());
|
||||
a.setLocalFlag(LocalFlag.ECHO, false);
|
||||
setAttributes(attributes);
|
||||
int h = DEFAULT_HEIGHT;
|
||||
try {
|
||||
String hp = System.getProperty("test.terminal.height");
|
||||
if (hp != null && !hp.isEmpty()) {
|
||||
h = Integer.parseInt(hp);
|
||||
}
|
||||
} catch (Throwable ex) {
|
||||
// ignore
|
||||
}
|
||||
setSize(new Size(80, h));
|
||||
setSize(size);
|
||||
this.bufferSize = bufferSize;
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -1312,6 +1330,31 @@ class ConsoleIOContext extends IOContext {
|
||||
inputReader.close();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Size getBufferSize() {
|
||||
return bufferSize;
|
||||
}
|
||||
}
|
||||
|
||||
private static final class TestTerminal extends ProgrammaticInTerminal {
|
||||
private static Size computeSize() {
|
||||
int h = DEFAULT_HEIGHT;
|
||||
try {
|
||||
String hp = System.getProperty("test.terminal.height");
|
||||
if (hp != null && !hp.isEmpty() && System.getProperty("test.jdk") != null) {
|
||||
h = Integer.parseInt(hp);
|
||||
}
|
||||
} catch (Throwable ex) {
|
||||
// ignore
|
||||
}
|
||||
return new Size(80, h);
|
||||
}
|
||||
public TestTerminal(InputStream input, OutputStream output) throws Exception {
|
||||
this(input, output, computeSize());
|
||||
}
|
||||
private TestTerminal(InputStream input, OutputStream output, Size size) throws Exception {
|
||||
super(input, output, "ansi", size, size);
|
||||
}
|
||||
}
|
||||
|
||||
private static final class CompletionState {
|
||||
|
@ -138,6 +138,8 @@ import static jdk.internal.jshell.tool.ContinuousCompletionProvider.STARTSWITH_M
|
||||
*/
|
||||
public class JShellTool implements MessageHandler {
|
||||
|
||||
private static String PROMPT = "\u0005";
|
||||
private static String CONTINUATION_PROMPT = "\u0006";
|
||||
private static final Pattern LINEBREAK = Pattern.compile("\\R");
|
||||
private static final Pattern ID = Pattern.compile("[se]?\\d+([-\\s].*)?");
|
||||
private static final Pattern RERUN_ID = Pattern.compile("/" + ID.pattern());
|
||||
@ -158,6 +160,7 @@ public class JShellTool implements MessageHandler {
|
||||
final PersistentStorage prefs;
|
||||
final Map<String, String> envvars;
|
||||
final Locale locale;
|
||||
final boolean interactiveTerminal;
|
||||
|
||||
final Feedback feedback = new Feedback();
|
||||
|
||||
@ -177,7 +180,8 @@ public class JShellTool implements MessageHandler {
|
||||
JShellTool(InputStream cmdin, PrintStream cmdout, PrintStream cmderr,
|
||||
PrintStream console,
|
||||
InputStream userin, PrintStream userout, PrintStream usererr,
|
||||
PersistentStorage prefs, Map<String, String> envvars, Locale locale) {
|
||||
PersistentStorage prefs, Map<String, String> envvars, Locale locale,
|
||||
boolean interactiveTerminal) {
|
||||
this.cmdin = cmdin;
|
||||
this.cmdout = cmdout;
|
||||
this.cmderr = cmderr;
|
||||
@ -193,6 +197,7 @@ public class JShellTool implements MessageHandler {
|
||||
this.prefs = prefs;
|
||||
this.envvars = envvars;
|
||||
this.locale = locale;
|
||||
this.interactiveTerminal = interactiveTerminal;
|
||||
}
|
||||
|
||||
private ResourceBundle versionRB = null;
|
||||
@ -974,7 +979,7 @@ public class JShellTool implements MessageHandler {
|
||||
};
|
||||
Runtime.getRuntime().addShutdownHook(shutdownHook);
|
||||
// execute from user input
|
||||
try (IOContext in = new ConsoleIOContext(this, cmdin, console)) {
|
||||
try (IOContext in = new ConsoleIOContext(this, cmdin, console, interactiveTerminal)) {
|
||||
int indent;
|
||||
try {
|
||||
String indentValue = indent();
|
||||
@ -1256,12 +1261,12 @@ public class JShellTool implements MessageHandler {
|
||||
return src;
|
||||
}
|
||||
String firstLinePrompt = interactive()
|
||||
? testPrompt ? " \005"
|
||||
? testPrompt ? PROMPT
|
||||
: feedback.getPrompt(currentNameSpace.tidNext())
|
||||
: "" // Non-interactive -- no prompt
|
||||
;
|
||||
String continuationPrompt = interactive()
|
||||
? testPrompt ? " \006"
|
||||
? testPrompt ? CONTINUATION_PROMPT
|
||||
: feedback.getContinuationPrompt(currentNameSpace.tidNext())
|
||||
: "" // Non-interactive -- no prompt
|
||||
;
|
||||
|
@ -51,6 +51,7 @@ public class JShellToolBuilder implements JavaShellToolBuilder {
|
||||
private PersistentStorage prefs = null;
|
||||
private Map<String, String> vars = null;
|
||||
private Locale locale = Locale.getDefault();
|
||||
private boolean interactiveTerminal;
|
||||
private boolean capturePrompt = false;
|
||||
|
||||
/**
|
||||
@ -208,6 +209,12 @@ public class JShellToolBuilder implements JavaShellToolBuilder {
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public JavaShellToolBuilder interactiveTerminal(boolean terminal) {
|
||||
this.interactiveTerminal = terminal;
|
||||
return this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a tool instance for testing. Not in JavaShellToolBuilder.
|
||||
*
|
||||
@ -221,7 +228,7 @@ public class JShellToolBuilder implements JavaShellToolBuilder {
|
||||
vars = System.getenv();
|
||||
}
|
||||
JShellTool sh = new JShellTool(cmdIn, cmdOut, cmdErr, console, userIn,
|
||||
userOut, userErr, prefs, vars, locale);
|
||||
userOut, userErr, prefs, vars, locale, interactiveTerminal);
|
||||
sh.testPrompt = capturePrompt;
|
||||
return sh;
|
||||
}
|
||||
|
@ -183,6 +183,32 @@ public interface JavaShellToolBuilder {
|
||||
*/
|
||||
JavaShellToolBuilder promptCapture(boolean capture);
|
||||
|
||||
/**
|
||||
* Set to true to specify the inputs and outputs are connected to an interactive terminal
|
||||
* that can interpret the ANSI escape codes. The characters sent to the output streams are
|
||||
* assumed to be interpreted by a terminal and shown to the user, and the exact order and nature
|
||||
* of characters sent to the outputs are unspecified.
|
||||
*
|
||||
* Set to false to specify a legacy simpler behavior whose output can be parsed by automatic
|
||||
* tools.
|
||||
*
|
||||
* When the input stream for this Java Shell is {@code System.in}, this value is ignored,
|
||||
* and the behavior is similar to specifying {@code true} in this method, but is more closely
|
||||
* following the specific terminal connected to {@code System.in}.
|
||||
*
|
||||
* @implSpec If this method is not called, the behavior should be
|
||||
* equivalent to calling {@code interactiveTerminal(false)}. The default implementation of
|
||||
* this method returns {@code this}.
|
||||
*
|
||||
* @param terminal if {@code true}, an terminal that can interpret the ANSI escape codes is
|
||||
* assumed to interpret the output. If {@code false}, a simpler output is selected.
|
||||
* @return the {@code JavaShellToolBuilder} instance
|
||||
* @since 17
|
||||
*/
|
||||
default JavaShellToolBuilder interactiveTerminal(boolean terminal) {
|
||||
return this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Run an instance of the Java shell tool as configured by the other methods
|
||||
* in this interface. This call is not destructive, more than one call of
|
||||
|
123
test/langtools/jdk/jshell/CustomInputToolBuilder.java
Normal file
123
test/langtools/jdk/jshell/CustomInputToolBuilder.java
Normal file
@ -0,0 +1,123 @@
|
||||
/*
|
||||
* Copyright (c) 2021, 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 8247403
|
||||
* @summary Verify JavaShellToolBuilder uses provided inputs
|
||||
* @modules jdk.jshell
|
||||
* @build KullaTesting TestingInputStream
|
||||
* @run testng CustomInputToolBuilder
|
||||
*/
|
||||
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.InputStream;
|
||||
import java.io.PrintStream;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
import jdk.jshell.tool.JavaShellToolBuilder;
|
||||
import org.testng.annotations.Test;
|
||||
|
||||
import static org.testng.Assert.assertTrue;
|
||||
|
||||
@Test
|
||||
public class CustomInputToolBuilder extends KullaTesting {
|
||||
|
||||
private static final String TEST_JDK = "test.jdk";
|
||||
|
||||
public void checkCustomInput() throws Exception {
|
||||
String testJdk = System.getProperty(TEST_JDK);
|
||||
try {
|
||||
System.clearProperty(TEST_JDK);
|
||||
doTest("System.out.println(\"read: \" + System.in.read());",
|
||||
"\u0005System.out.println(\"read: \" + System.in.read());",
|
||||
"read: 97",
|
||||
"\u0005/exit");
|
||||
doTest("1 + 1", "\u00051 + 1", "$1 ==> 2", "\u0005/exit");
|
||||
doTest("for (int i = 0; i < 100; i++) {\nSystem.err.println(i);\n}\n",
|
||||
"\u0005for (int i = 0; i < 100; i++) {",
|
||||
"\u0006System.err.println(i);", "\u0006}",
|
||||
"\u0005/exit");
|
||||
StringBuilder longInput = new StringBuilder();
|
||||
String constant = "1_______________1";
|
||||
longInput.append(constant);
|
||||
for (int i = 0; i < 100; i++) {
|
||||
longInput.append(" + ");
|
||||
longInput.append(constant);
|
||||
}
|
||||
doTest(longInput.toString(), "\u0005" + longInput);
|
||||
} finally {
|
||||
System.setProperty(TEST_JDK, testJdk);
|
||||
}
|
||||
}
|
||||
|
||||
private void doTest(String code, String... expectedLines) throws Exception {
|
||||
doTest(false, code, expectedLines);
|
||||
}
|
||||
|
||||
private void doTest(boolean interactiveTerminal, String code, String... expectedLines) throws Exception {
|
||||
byte[] cmdInputData = (code + "\n/exit\n").getBytes();
|
||||
InputStream cmdInput = new ByteArrayInputStream(cmdInputData);
|
||||
InputStream userInput = new ByteArrayInputStream("a\n".getBytes());
|
||||
ByteArrayOutputStream out = new ByteArrayOutputStream();
|
||||
PrintStream printOut = new PrintStream(out);
|
||||
|
||||
JavaShellToolBuilder.builder()
|
||||
.in(cmdInput, userInput)
|
||||
.out(printOut, printOut, printOut)
|
||||
.interactiveTerminal(interactiveTerminal)
|
||||
.promptCapture(true)
|
||||
.start("--no-startup");
|
||||
|
||||
String actual = new String(out.toByteArray());
|
||||
List<String> actualLines = Arrays.asList(actual.split("\\R"));
|
||||
|
||||
for (String expectedLine : expectedLines) {
|
||||
assertTrue(actualLines.contains(expectedLine),
|
||||
"actual:\n" + actualLines + "\n, expected:\n" + expectedLine);
|
||||
}
|
||||
}
|
||||
|
||||
public void checkInteractiveTerminal() throws Exception {
|
||||
String testJdk = System.getProperty(TEST_JDK);
|
||||
try {
|
||||
System.clearProperty(TEST_JDK);
|
||||
|
||||
//note the exact format of the output is not specified here, and the test mostly validates
|
||||
//the current behavior, and shows the output changes based on the interactiveTerminal setting:
|
||||
doTest(true,
|
||||
"System.out.println(\"read: \" + System.in.read());",
|
||||
"\u001b[?2004h\u0005System.out.println(\"read: \" + System.in.read()\u001b[2D\u001b[2C)\u001b[29D\u001b[29C;",
|
||||
"\u001b[?2004lread: 97",
|
||||
"\u001b[?2004h\u0005/exit");
|
||||
doTest(true,
|
||||
"1 + 1",
|
||||
"\u001b[?2004h\u00051 + 1",
|
||||
"\u001b[?2004l$1 ==> 2",
|
||||
"\u001b[?2004h\u0005/exit");
|
||||
} finally {
|
||||
System.setProperty(TEST_JDK, testJdk);
|
||||
}
|
||||
}
|
||||
}
|
@ -28,6 +28,7 @@
|
||||
* @modules
|
||||
* jdk.compiler/com.sun.tools.javac.api
|
||||
* jdk.compiler/com.sun.tools.javac.main
|
||||
* jdk.jshell/jdk.internal.jshell.tool:open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @library /tools/lib
|
||||
|
@ -29,6 +29,7 @@
|
||||
* @modules
|
||||
* jdk.compiler/com.sun.tools.javac.api
|
||||
* jdk.compiler/com.sun.tools.javac.main
|
||||
* jdk.jshell/jdk.internal.jshell.tool:open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @build toolbox.ToolBox toolbox.JarTask toolbox.JavacTask
|
||||
|
@ -32,6 +32,7 @@
|
||||
* jdk.compiler/com.sun.tools.javac.api
|
||||
* jdk.compiler/com.sun.tools.javac.main
|
||||
* jdk.internal.le/jdk.internal.org.jline.reader.impl
|
||||
* jdk.jshell/jdk.internal.jshell.tool:open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @build toolbox.ToolBox toolbox.JarTask toolbox.JavacTask
|
||||
|
@ -26,6 +26,7 @@
|
||||
* @bug 8182489
|
||||
* @summary test history with multiline snippets
|
||||
* @modules
|
||||
* jdk.jshell/jdk.internal.jshell.tool:open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @build UITesting
|
||||
|
@ -26,6 +26,7 @@
|
||||
* @bug 8166334 8188894
|
||||
* @summary test shift-tab shortcuts "fixes"
|
||||
* @modules
|
||||
* jdk.jshell/jdk.internal.jshell.tool:open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @build UITesting
|
||||
|
@ -27,6 +27,7 @@
|
||||
* @modules
|
||||
* jdk.compiler/com.sun.tools.javac.api
|
||||
* jdk.compiler/com.sun.tools.javac.main
|
||||
* jdk.jshell/jdk.internal.jshell.tool:open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @library /tools/lib
|
||||
|
@ -27,7 +27,7 @@
|
||||
* @modules
|
||||
* jdk.compiler/com.sun.tools.javac.api
|
||||
* jdk.compiler/com.sun.tools.javac.main
|
||||
* jdk.jshell/jdk.internal.jshell.tool
|
||||
* jdk.jshell/jdk.internal.jshell.tool:+open
|
||||
* jdk.jshell/jdk.internal.jshell.tool.resources:open
|
||||
* jdk.jshell/jdk.jshell:open
|
||||
* @library /tools/lib
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2017, 2018, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2017, 2021, 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
|
||||
@ -27,6 +27,7 @@ import java.io.OutputStream;
|
||||
import java.io.OutputStreamWriter;
|
||||
import java.io.PrintStream;
|
||||
import java.io.Writer;
|
||||
import java.lang.reflect.Field;
|
||||
import java.text.MessageFormat;
|
||||
import java.util.HashMap;
|
||||
import java.util.Locale;
|
||||
@ -108,6 +109,13 @@ public class UITesting {
|
||||
runner.start();
|
||||
|
||||
try {
|
||||
Class<?> jshellToolClass = Class.forName("jdk.internal.jshell.tool.JShellTool");
|
||||
Field promptField = jshellToolClass.getDeclaredField("PROMPT");
|
||||
promptField.setAccessible(true);
|
||||
promptField.set(null, PROMPT);
|
||||
Field continuationPromptField = jshellToolClass.getDeclaredField("CONTINUATION_PROMPT");
|
||||
continuationPromptField.setAccessible(true);
|
||||
continuationPromptField.set(null, CONTINUATION_PROMPT);
|
||||
waitOutput(out, PROMPT);
|
||||
test.test(inputSink, out);
|
||||
} finally {
|
||||
|
Loading…
x
Reference in New Issue
Block a user