2017-04-06 16:19:33 +02:00
|
|
|
/*
|
2021-05-31 09:25:16 +00:00
|
|
|
* Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved.
|
2017-04-06 16:19:33 +02:00
|
|
|
* 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.IOException;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.OutputStream;
|
|
|
|
import java.io.OutputStreamWriter;
|
|
|
|
import java.io.PrintStream;
|
|
|
|
import java.io.Writer;
|
2021-05-31 09:25:16 +00:00
|
|
|
import java.lang.reflect.Field;
|
2021-06-02 09:55:06 +00:00
|
|
|
import java.nio.charset.StandardCharsets;
|
2017-04-19 11:36:44 +02:00
|
|
|
import java.text.MessageFormat;
|
2017-04-06 16:19:33 +02:00
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.Locale;
|
2017-04-19 11:36:44 +02:00
|
|
|
import java.util.ResourceBundle;
|
2017-04-11 17:26:52 -07:00
|
|
|
import java.util.logging.Level;
|
|
|
|
import java.util.logging.Logger;
|
2017-04-06 16:19:33 +02:00
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
|
|
|
|
2017-04-19 11:36:44 +02:00
|
|
|
import jdk.jshell.JShell;
|
2017-04-06 16:19:33 +02:00
|
|
|
import jdk.jshell.tool.JavaShellToolBuilder;
|
|
|
|
|
|
|
|
public class UITesting {
|
|
|
|
|
2017-12-07 13:23:18 -08:00
|
|
|
protected static final String TAB = "\011";
|
|
|
|
protected static final String INTERRUPT = "\u0003";
|
|
|
|
protected static final String BELL = "\u0007";
|
2018-12-11 11:29:28 +01:00
|
|
|
protected static final String PROMPT = " \u0005";
|
|
|
|
protected static final String CONTINUATION_PROMPT = " \u0006";
|
|
|
|
protected static final String REDRAW_PROMPT = "\n\r?" + PROMPT;
|
|
|
|
protected static final String UP = "\033[A";
|
|
|
|
protected static final String DOWN = "\033[B";
|
2017-06-30 20:03:07 +02:00
|
|
|
private final boolean laxLineEndings;
|
|
|
|
|
|
|
|
public UITesting() {
|
|
|
|
this(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public UITesting(boolean laxLineEndings) {
|
|
|
|
this.laxLineEndings = laxLineEndings;
|
|
|
|
}
|
|
|
|
|
2017-04-06 16:19:33 +02:00
|
|
|
protected void doRunTest(Test test) throws Exception {
|
2017-04-11 17:26:52 -07:00
|
|
|
// turn on logging of launch failures
|
|
|
|
Logger.getLogger("jdk.jshell.execution").setLevel(Level.ALL);
|
|
|
|
|
2017-04-06 16:19:33 +02:00
|
|
|
PipeInputStream input = new PipeInputStream();
|
|
|
|
StringBuilder out = new StringBuilder();
|
|
|
|
PrintStream outS = new PrintStream(new OutputStream() {
|
|
|
|
@Override public void write(int b) throws IOException {
|
|
|
|
synchronized (out) {
|
|
|
|
System.out.print((char) b);
|
|
|
|
out.append((char) b);
|
|
|
|
out.notifyAll();
|
|
|
|
}
|
2017-04-24 18:58:50 +02:00
|
|
|
}
|
|
|
|
@Override public void write(byte[] b, int off, int len) throws IOException {
|
|
|
|
synchronized (out) {
|
|
|
|
String data = new String(b, off, len);
|
|
|
|
System.out.print(data);
|
|
|
|
out.append(data);
|
|
|
|
out.notifyAll();
|
|
|
|
}
|
2017-04-06 16:19:33 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
Thread runner = new Thread(() -> {
|
|
|
|
try {
|
|
|
|
JavaShellToolBuilder.builder()
|
|
|
|
.in(input, input)
|
|
|
|
.out(outS)
|
|
|
|
.err(outS)
|
|
|
|
.promptCapture(true)
|
|
|
|
.persistence(new HashMap<>())
|
|
|
|
.locale(Locale.US)
|
|
|
|
.run("--no-startup");
|
|
|
|
} catch (Exception ex) {
|
|
|
|
throw new IllegalStateException(ex);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2021-06-02 09:55:06 +00:00
|
|
|
Writer inputSink = new OutputStreamWriter(input.createOutput(), StandardCharsets.UTF_8) {
|
2017-04-06 16:19:33 +02:00
|
|
|
@Override
|
|
|
|
public void write(String str) throws IOException {
|
|
|
|
super.write(str);
|
|
|
|
flush();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
runner.start();
|
|
|
|
|
|
|
|
try {
|
2021-05-31 09:25:16 +00:00
|
|
|
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);
|
2017-12-07 13:23:18 -08:00
|
|
|
waitOutput(out, PROMPT);
|
2017-04-06 16:19:33 +02:00
|
|
|
test.test(inputSink, out);
|
|
|
|
} finally {
|
2017-12-07 13:23:18 -08:00
|
|
|
inputSink.write(INTERRUPT + INTERRUPT + "/exit");
|
2017-04-06 16:19:33 +02:00
|
|
|
|
|
|
|
runner.join(1000);
|
|
|
|
if (runner.isAlive()) {
|
|
|
|
runner.stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected interface Test {
|
|
|
|
public void test(Writer inputSink, StringBuilder out) throws Exception;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static final long TIMEOUT;
|
|
|
|
|
|
|
|
static {
|
|
|
|
long factor;
|
|
|
|
|
|
|
|
try {
|
|
|
|
factor = (long) Double.parseDouble(System.getProperty("test.timeout.factor", "1"));
|
|
|
|
} catch (NumberFormatException ex) {
|
|
|
|
factor = 1;
|
|
|
|
}
|
|
|
|
TIMEOUT = 60_000 * factor;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void waitOutput(StringBuilder out, String expected) {
|
2017-12-13 14:21:12 -08:00
|
|
|
waitOutput(out, expected, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if expected is found, false if secondary is found,
|
|
|
|
// otherwise, time out with an IllegalStateException
|
|
|
|
protected boolean waitOutput(StringBuilder out, String expected, String secondary) {
|
2018-12-11 11:29:28 +01:00
|
|
|
expected = expected.replaceAll("\n", laxLineEndings ? "\r*\n" : System.getProperty("line.separator"));
|
2017-04-06 16:19:33 +02:00
|
|
|
Pattern expectedPattern = Pattern.compile(expected, Pattern.DOTALL);
|
2017-12-13 14:21:12 -08:00
|
|
|
Pattern secondaryPattern = null;
|
|
|
|
if (secondary != null) {
|
2018-12-11 11:29:28 +01:00
|
|
|
secondary = secondary.replaceAll("\n", laxLineEndings ? "\r*\n" : System.getProperty("line.separator"));
|
2017-12-13 14:21:12 -08:00
|
|
|
secondaryPattern = Pattern.compile(secondary, Pattern.DOTALL);
|
|
|
|
}
|
2017-04-06 16:19:33 +02:00
|
|
|
synchronized (out) {
|
|
|
|
long s = System.currentTimeMillis();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
Matcher m = expectedPattern.matcher(out);
|
|
|
|
if (m.find()) {
|
2017-08-22 13:10:46 +02:00
|
|
|
out.delete(0, m.end());
|
2017-12-13 14:21:12 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (secondaryPattern != null) {
|
|
|
|
m = secondaryPattern.matcher(out);
|
|
|
|
if (m.find()) {
|
|
|
|
out.delete(0, m.end());
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-06 16:19:33 +02:00
|
|
|
}
|
|
|
|
long e = System.currentTimeMillis();
|
|
|
|
if ((e - s) > TIMEOUT) {
|
|
|
|
throw new IllegalStateException("Timeout waiting for: " + quote(expected) + ", actual output so far: " + quote(out.toString()));
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
out.wait(TIMEOUT);
|
|
|
|
} catch (InterruptedException ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private String quote(String original) {
|
|
|
|
StringBuilder output = new StringBuilder();
|
|
|
|
|
|
|
|
for (char c : original.toCharArray()) {
|
|
|
|
if (c < 32) {
|
|
|
|
output.append(String.format("\\u%04X", (int) c));
|
|
|
|
} else {
|
|
|
|
output.append(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected String clearOut(String what) {
|
2018-06-01 13:04:30 +02:00
|
|
|
return backspace(what.length()) + "\\u001B\\[K";
|
2017-04-06 16:19:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
protected String backspace(int n) {
|
|
|
|
return fill(n, '\010');
|
|
|
|
}
|
|
|
|
|
|
|
|
protected String space(int n) {
|
|
|
|
return fill(n, ' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
private String fill(int n, char c) {
|
|
|
|
StringBuilder result = new StringBuilder(n);
|
|
|
|
|
|
|
|
while (n-- > 0)
|
|
|
|
result.append(c);
|
|
|
|
|
|
|
|
return result.toString();
|
|
|
|
}
|
|
|
|
|
2017-04-19 11:36:44 +02:00
|
|
|
private final ResourceBundle resources;
|
|
|
|
{
|
|
|
|
resources = ResourceBundle.getBundle("jdk.internal.jshell.tool.resources.l10n", Locale.US, JShell.class.getModule());
|
|
|
|
}
|
|
|
|
|
|
|
|
protected String getResource(String key) {
|
|
|
|
return resources.getString(key);
|
|
|
|
}
|
|
|
|
|
2017-12-07 13:23:18 -08:00
|
|
|
protected String resource(String key) {
|
2018-12-11 11:29:28 +01:00
|
|
|
return patternQuote(getResource(key));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected String patternQuote(String str) {
|
|
|
|
//from JDK-6507804:
|
|
|
|
return str.replaceAll("([\\\\\\[\\].^$?*+{}()|])", "\\\\$1");
|
2017-12-07 13:23:18 -08:00
|
|
|
}
|
|
|
|
|
2017-04-19 11:36:44 +02:00
|
|
|
protected String getMessage(String key, Object... args) {
|
|
|
|
return MessageFormat.format(resources.getString(key), args);
|
|
|
|
}
|
2017-04-06 16:19:33 +02:00
|
|
|
private static class PipeInputStream extends InputStream {
|
|
|
|
|
|
|
|
private static final int INITIAL_SIZE = 128;
|
|
|
|
private int[] buffer = new int[INITIAL_SIZE];
|
|
|
|
private int start;
|
|
|
|
private int end;
|
|
|
|
private boolean closed;
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public synchronized int read() throws IOException {
|
|
|
|
if (start == end && !closed) {
|
|
|
|
inputNeeded();
|
|
|
|
}
|
|
|
|
while (start == end) {
|
|
|
|
if (closed) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
wait();
|
|
|
|
} catch (InterruptedException ex) {
|
|
|
|
//ignore
|
|
|
|
}
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
return buffer[start];
|
|
|
|
} finally {
|
|
|
|
start = (start + 1) % buffer.length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public synchronized int read(byte[] b, int off, int len) throws IOException {
|
|
|
|
if (b == null) {
|
|
|
|
throw new NullPointerException();
|
|
|
|
} else if (off < 0 || len < 0 || len > b.length - off) {
|
|
|
|
throw new IndexOutOfBoundsException();
|
|
|
|
} else if (len == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int c = read();
|
|
|
|
if (c == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
b[off] = (byte)c;
|
|
|
|
|
|
|
|
int totalRead = 1;
|
|
|
|
while (totalRead < len && start != end) {
|
|
|
|
int r = read();
|
|
|
|
if (r == (-1))
|
|
|
|
break;
|
|
|
|
b[off + totalRead++] = (byte) r;
|
|
|
|
}
|
|
|
|
return totalRead;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void inputNeeded() throws IOException {}
|
|
|
|
|
|
|
|
private synchronized void write(int b) {
|
|
|
|
if (closed) {
|
|
|
|
throw new IllegalStateException("Already closed.");
|
|
|
|
}
|
|
|
|
int newEnd = (end + 1) % buffer.length;
|
|
|
|
if (newEnd == start) {
|
|
|
|
//overflow:
|
|
|
|
int[] newBuffer = new int[buffer.length * 2];
|
|
|
|
int rightPart = (end > start ? end : buffer.length) - start;
|
|
|
|
int leftPart = end > start ? 0 : start - 1;
|
|
|
|
System.arraycopy(buffer, start, newBuffer, 0, rightPart);
|
|
|
|
System.arraycopy(buffer, 0, newBuffer, rightPart, leftPart);
|
|
|
|
buffer = newBuffer;
|
|
|
|
start = 0;
|
|
|
|
end = rightPart + leftPart;
|
|
|
|
newEnd = end + 1;
|
|
|
|
}
|
|
|
|
buffer[end] = b;
|
|
|
|
end = newEnd;
|
|
|
|
notifyAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public synchronized void close() {
|
|
|
|
closed = true;
|
|
|
|
notifyAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
public OutputStream createOutput() {
|
|
|
|
return new OutputStream() {
|
|
|
|
@Override public void write(int b) throws IOException {
|
|
|
|
PipeInputStream.this.write(b);
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public void write(byte[] b, int off, int len) throws IOException {
|
|
|
|
for (int i = 0 ; i < len ; i++) {
|
|
|
|
write(Byte.toUnsignedInt(b[off + i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@Override
|
|
|
|
public void close() throws IOException {
|
|
|
|
PipeInputStream.this.close();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|