a45c9af124
Reviewed-by: jlahoda
445 lines
20 KiB
Java
445 lines
20 KiB
Java
/*
|
|
* 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
|
|
* 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 8144095 8164825 8169818 8153402 8165405 8177079 8178013 8167554 8166232 8277328
|
|
* @summary Test Command Completion
|
|
* @modules jdk.compiler/com.sun.tools.javac.api
|
|
* jdk.compiler/com.sun.tools.javac.main
|
|
* jdk.jdeps/com.sun.tools.javap
|
|
* jdk.jshell/jdk.internal.jshell.tool
|
|
* @library /tools/lib
|
|
* @build toolbox.ToolBox toolbox.JarTask toolbox.JavacTask
|
|
* @build ReplToolTesting TestingInputStream Compiler
|
|
* @run testng CommandCompletionTest
|
|
*/
|
|
|
|
import java.io.IOException;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.nio.file.Paths;
|
|
import java.util.Arrays;
|
|
import java.util.List;
|
|
import java.util.Locale;
|
|
import java.util.function.Predicate;
|
|
import java.util.stream.Collectors;
|
|
import java.util.stream.Stream;
|
|
|
|
import org.testng.SkipException;
|
|
import org.testng.annotations.Test;
|
|
|
|
import jdk.internal.jshell.tool.JShellTool;
|
|
import jdk.internal.jshell.tool.JShellToolBuilder;
|
|
import jdk.jshell.SourceCodeAnalysis.Suggestion;
|
|
|
|
import static org.testng.Assert.assertEquals;
|
|
import static org.testng.Assert.assertTrue;
|
|
import static org.testng.Assert.fail;
|
|
|
|
public class CommandCompletionTest extends ReplToolTesting {
|
|
|
|
|
|
private JShellTool repl;
|
|
|
|
@Override
|
|
protected void testRawRun(Locale locale, String[] args) {
|
|
repl = ((JShellToolBuilder) builder(locale))
|
|
.rawTool();
|
|
try {
|
|
repl.start(args);
|
|
} catch (Exception ex) {
|
|
fail("Repl tool died with exception", ex);
|
|
}
|
|
}
|
|
|
|
public void assertCompletion(boolean after, String code, int minElements) {
|
|
if (!after) {
|
|
setCommandInput("\n");
|
|
} else {
|
|
List<String> completions = computeCompletions(code, false);
|
|
assertTrue(completions.size() >= minElements, "Command: " + code + ", output: " +
|
|
completions.toString() + ", expected output with at least " + minElements + " elements");
|
|
}
|
|
}
|
|
|
|
public void assertCompletion(boolean after, String code, boolean isSmart, String... expected) {
|
|
if (!after) {
|
|
setCommandInput("\n");
|
|
} else {
|
|
assertCompletion(code, isSmart, expected);
|
|
}
|
|
}
|
|
|
|
public void assertCompletion(String code, boolean isSmart, String... expected) {
|
|
List<String> completions = computeCompletions(code, isSmart);
|
|
List<String> expectedL = Arrays.asList(expected);
|
|
assertEquals(completions, expectedL, "Command: " + code + ", output: " +
|
|
completions.toString() + ", expected: " + expectedL.toString());
|
|
}
|
|
|
|
private List<String> computeCompletions(String code, boolean isSmart) {
|
|
int cursor = code.indexOf('|');
|
|
code = code.replace("|", "");
|
|
assertTrue(cursor > -1, "'|' not found: " + code);
|
|
List<Suggestion> completions =
|
|
repl.commandCompletionSuggestions(code, cursor, new int[] {-1}); //XXX: ignoring anchor for now
|
|
return completions.stream()
|
|
.filter(s -> isSmart == s.matchesType())
|
|
.map(s -> s.continuation())
|
|
.distinct()
|
|
.collect(Collectors.toList());
|
|
}
|
|
|
|
@Test
|
|
public void testCommand() {
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/deb|", false),
|
|
a -> assertCompletion(a, "/re|", false, "/reload ", "/reset "),
|
|
a -> assertCompletion(a, "/h|", false, "/help ", "/history ")
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testList() {
|
|
test(false, new String[] {"--no-startup"},
|
|
a -> assertCompletion(a, "/l|", false, "/list "),
|
|
a -> assertCompletion(a, "/list |", false, "-all", "-history", "-start "),
|
|
a -> assertCompletion(a, "/list -h|", false, "-history"),
|
|
a -> assertCompletion(a, "/list q|", false),
|
|
a -> assertVariable(a, "int", "xray"),
|
|
a -> assertCompletion(a, "/list |", false, "-all", "-history", "-start ", "1 ", "xray "),
|
|
a -> assertCompletion(a, "/list x|", false, "xray "),
|
|
a -> assertCompletion(a, "/list xray |", false)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testHistory() {
|
|
test(false, new String[] {"--no-startup"},
|
|
a -> assertCompletion(a, "/hi|", false, "/history "),
|
|
a -> assertCompletion(a, "/history |", false, "-all")
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testDrop() {
|
|
test(false, new String[] {"--no-startup"},
|
|
a -> assertCompletion(a, "/d|", false, "/drop "),
|
|
a -> assertClass(a, "class cTest {}", "class", "cTest"),
|
|
a -> assertMethod(a, "int mTest() { return 0; }", "()I", "mTest"),
|
|
a -> assertVariable(a, "int", "fTest"),
|
|
a -> assertCompletion(a, "/drop |", false, "1 ", "2 ", "3 ", "cTest ", "fTest ", "mTest "),
|
|
a -> assertCompletion(a, "/drop f|", false, "fTest ")
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testEdit() {
|
|
test(false, new String[]{"--no-startup"},
|
|
a -> assertCompletion(a, "/e|", false, "/edit ", "/env ", "/exit "),
|
|
a -> assertCompletion(a, "/ed|", false, "/edit "),
|
|
a -> assertClass(a, "class cTest {}", "class", "cTest"),
|
|
a -> assertMethod(a, "int mTest() { return 0; }", "()I", "mTest"),
|
|
a -> assertVariable(a, "int", "fTest"),
|
|
a -> assertCompletion(a, "/edit |", false,
|
|
"-all" , "-start " , "1 ", "2 ", "3 ", "cTest ", "fTest ", "mTest "),
|
|
a -> assertCompletion(a, "/edit cTest |", false,
|
|
"2 ", "3 ", "fTest ", "mTest "),
|
|
a -> assertCompletion(a, "/edit 1 fTest |", false,
|
|
"2 ", "mTest "),
|
|
a -> assertCompletion(a, "/edit f|", false, "fTest "),
|
|
a -> assertCompletion(a, "/edit mTest f|", false, "fTest ")
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testHelp() {
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/help |", false,
|
|
"/! ", "/-<n> ", "/<id> ", "/? ", "/drop ",
|
|
"/edit ", "/env ", "/exit ",
|
|
"/help ", "/history ", "/imports ",
|
|
"/list ", "/methods ", "/open ", "/reload ", "/reset ",
|
|
"/save ", "/set ", "/types ", "/vars ", "context ",
|
|
"id ", "intro ", "keys ", "rerun ", "shortcuts "),
|
|
a -> assertCompletion(a, "/? |", false,
|
|
"/! ", "/-<n> ", "/<id> ", "/? ", "/drop ",
|
|
"/edit ", "/env ", "/exit ",
|
|
"/help ", "/history ", "/imports ",
|
|
"/list ", "/methods ", "/open ", "/reload ", "/reset ",
|
|
"/save ", "/set ", "/types ", "/vars ", "context ",
|
|
"id ", "intro ", "keys ", "rerun ", "shortcuts "),
|
|
a -> assertCompletion(a, "/help /s|", false,
|
|
"/save ", "/set "),
|
|
a -> assertCompletion(a, "/help /set |", false,
|
|
"editor", "feedback", "format", "indent", "mode", "prompt", "start", "truncation"),
|
|
a -> assertCompletion(a, "/help set |", false,
|
|
"editor", "feedback", "format", "indent", "mode", "prompt", "start", "truncation"),
|
|
a -> assertCompletion(a, "/help /edit |", false),
|
|
a -> assertCompletion(a, "/help dr|", false,
|
|
"drop ")
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testReload() {
|
|
String[] ropts = new String[] { "-add-exports ", "-add-modules ",
|
|
"-class-path ", "-module-path ", "-quiet ", "-restore " };
|
|
String[] dropts = new String[] { "--add-exports ", "--add-modules ",
|
|
"--class-path ", "--module-path ", "--quiet ", "--restore " };
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/reloa |", false, ropts),
|
|
a -> assertCompletion(a, "/relo |", false, ropts),
|
|
a -> assertCompletion(a, "/reload -|", false, ropts),
|
|
a -> assertCompletion(a, "/reload --|", false, dropts),
|
|
a -> assertCompletion(a, "/reload -restore |", false, ropts),
|
|
a -> assertCompletion(a, "/reload -restore --|", false, dropts),
|
|
a -> assertCompletion(a, "/reload -rest|", false, "-restore "),
|
|
a -> assertCompletion(a, "/reload --r|", false, "--restore "),
|
|
a -> assertCompletion(a, "/reload -q|", false, "-quiet "),
|
|
a -> assertCompletion(a, "/reload -add|", false, "-add-exports ", "-add-modules "),
|
|
a -> assertCompletion(a, "/reload -class-path . -quiet |", false, ropts)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testEnv() {
|
|
String[] ropts = new String[] { "-add-exports ", "-add-modules ",
|
|
"-class-path ", "-module-path " };
|
|
String[] dropts = new String[] { "--add-exports ", "--add-modules ",
|
|
"--class-path ", "--module-path " };
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/env |", false, ropts),
|
|
a -> assertCompletion(a, "/env -|", false, ropts),
|
|
a -> assertCompletion(a, "/env --|", false, dropts),
|
|
a -> assertCompletion(a, "/env --a|", false, "--add-exports ", "--add-modules "),
|
|
a -> assertCompletion(a, "/env -add-|", false, "-add-exports ", "-add-modules "),
|
|
a -> assertCompletion(a, "/env -class-path . |", false, ropts),
|
|
a -> assertCompletion(a, "/env -class-path . --|", false, dropts)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testReset() {
|
|
String[] ropts = new String[] { "-add-exports ", "-add-modules ",
|
|
"-class-path ", "-module-path " };
|
|
String[] dropts = new String[] { "--add-exports ", "--add-modules ",
|
|
"--class-path ", "--module-path " };
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/reset |", false, ropts),
|
|
a -> assertCompletion(a, "/res -m|", false, "-module-path "),
|
|
a -> assertCompletion(a, "/res -module-|", false, "-module-path "),
|
|
a -> assertCompletion(a, "/res --m|", false, "--module-path "),
|
|
a -> assertCompletion(a, "/res --module-|", false, "--module-path "),
|
|
a -> assertCompletion(a, "/reset -add|", false, "-add-exports ", "-add-modules "),
|
|
a -> assertCompletion(a, "/rese -class-path . |", false, ropts),
|
|
a -> assertCompletion(a, "/rese -class-path . --|", false, dropts)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testVarsMethodsTypes() {
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/v|", false, "/vars "),
|
|
a -> assertCompletion(a, "/m|", false, "/methods "),
|
|
a -> assertCompletion(a, "/t|", false, "/types "),
|
|
a -> assertClass(a, "class cTest {}", "class", "cTest"),
|
|
a -> assertMethod(a, "int mTest() { return 0; }", "()I", "mTest"),
|
|
a -> assertVariable(a, "int", "fTest"),
|
|
a -> assertCompletion(a, "/vars |", false, "-all", "-start ", "3 ", "fTest "),
|
|
a -> assertCompletion(a, "/meth |", false, "-all", "-start ", "2 ", "mTest "),
|
|
a -> assertCompletion(a, "/typ |", false, "-all", "-start ", "1 ", "cTest "),
|
|
a -> assertCompletion(a, "/var f|", false, "fTest ")
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testOpen() throws IOException {
|
|
Compiler compiler = new Compiler();
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/o|", false, "/open ")
|
|
);
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/open |", 1)
|
|
);
|
|
Path classDir = compiler.getClassDir();
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/open " + classDir + "/|", 1)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testSave() throws IOException {
|
|
Compiler compiler = new Compiler();
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/s|", false, "/save ", "/set ")
|
|
);
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/save |", 4)
|
|
);
|
|
Path classDir = compiler.getClassDir();
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/save " + classDir + "/|", 1),
|
|
a -> assertCompletion(a, "/save -all " + classDir + "/|", 1)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testClassPath() throws IOException {
|
|
Compiler compiler = new Compiler();
|
|
Path outDir = compiler.getPath("testClasspathCompletion");
|
|
Files.createDirectories(outDir);
|
|
Files.createDirectories(outDir.resolve("dir"));
|
|
createIfNeeded(outDir.resolve("test.jar"));
|
|
createIfNeeded(outDir.resolve("test.zip"));
|
|
compiler.compile(outDir, "package pkg; public class A { public String toString() { return \"A\"; } }");
|
|
String jarName = "test.jar";
|
|
compiler.jar(outDir, jarName, "pkg/A.class");
|
|
compiler.getPath(outDir).resolve(jarName);
|
|
List<String> paths = listFiles(outDir, CLASSPATH_FILTER);
|
|
String[] pathArray = paths.toArray(new String[paths.size()]);
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/env -class-path " + outDir + "/|", false, pathArray),
|
|
a -> assertCompletion(a, "/env --class-path " + outDir + "/|", false, pathArray),
|
|
a -> assertCompletion(a, "/env -clas " + outDir + "/|", false, pathArray),
|
|
a -> assertCompletion(a, "/env --class-p " + outDir + "/|", false, pathArray),
|
|
a -> assertCompletion(a, "/env --module-path . --class-p " + outDir + "/|", false, pathArray)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testClassPathWithSpace() throws IOException {
|
|
Compiler compiler = new Compiler();
|
|
Path outDir = compiler.getPath("testClassPathWithSpace");
|
|
Path dirWithSpace = Files.createDirectories(outDir.resolve("dir with space"));
|
|
Files.createDirectories(dirWithSpace.resolve("nested with space"));
|
|
String[] pathArray = new String[] {"dir\\ with\\ space/"};
|
|
String[] pathArray2 = new String[] {"nested\\ with\\ space/"};
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/env -class-path " + outDir + "/|", false, pathArray),
|
|
a -> assertCompletion(a, "/env -class-path " + outDir + "/dir|", false, pathArray),
|
|
a -> assertCompletion(a, "/env -class-path " + outDir + "/dir\\ with|", false, pathArray),
|
|
a -> assertCompletion(a, "/env -class-path " + outDir + "/dir\\ with\\ space/|", false, pathArray2)
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testUserHome() throws IOException {
|
|
List<String> completions;
|
|
Path home = Paths.get(System.getProperty("user.home"));
|
|
String selectedFile;
|
|
try (Stream<Path> content = Files.list(home)) {
|
|
selectedFile = content.filter(CLASSPATH_FILTER)
|
|
.filter(file -> file.getFileName().toString().contains(" "))
|
|
.findAny()
|
|
.map(file -> file.getFileName().toString().replace(" ", "\\ "))
|
|
.orElse(null);
|
|
}
|
|
if (selectedFile == null) {
|
|
throw new SkipException("No suitable file(s) found for this test in " + home);
|
|
}
|
|
try (Stream<Path> content = Files.list(home)) {
|
|
completions = content.filter(CLASSPATH_FILTER)
|
|
.filter(file -> file.getFileName().toString().startsWith(selectedFile.replace("\\ ", " ")))
|
|
.map(file -> file.getFileName().toString().replace(" ", "\\ ") + (Files.isDirectory(file) ? "/" : ""))
|
|
.sorted()
|
|
.collect(Collectors.toList());
|
|
}
|
|
testNoStartUp(
|
|
a -> assertCompletion(a, "/env --class-path ~/" + selectedFile + "|", false, completions.toArray(new String[completions.size()]))
|
|
);
|
|
}
|
|
|
|
@Test
|
|
public void testSet() throws IOException {
|
|
|
|
String[] modes = {"concise ", "normal ", "silent ", "verbose "};
|
|
String[] options = {"-command", "-delete", "-quiet"};
|
|
String[] modesWithOptions = Stream.concat(Arrays.stream(options), Arrays.stream(modes)).sorted().toArray(String[]::new);
|
|
test(false, new String[] {"--no-startup"},
|
|
a -> assertCompletion(a, "/se|", false, "/set "),
|
|
a -> assertCompletion(a, "/set |", false, "editor ", "feedback ", "format ", "indent ", "mode ", "prompt ", "start ", "truncation "),
|
|
|
|
// /set editor
|
|
a -> assertCompletion(a, "/set e|", false, "editor "),
|
|
a -> assertCompletion(a, "/set editor |", 1),
|
|
|
|
// /set feedback
|
|
a -> assertCompletion(a, "/set fe|", false, "feedback "),
|
|
a -> assertCompletion(a, "/set fe |", false, modes),
|
|
|
|
// /set format
|
|
a -> assertCompletion(a, "/set fo|", false, "format "),
|
|
a -> assertCompletion(a, "/set fo |", false, modes),
|
|
|
|
// /set mode
|
|
a -> assertCompletion(a, "/set mo|", false, "mode "),
|
|
a -> assertCompletion(a, "/set mo |", false),
|
|
a -> assertCompletion(a, "/set mo newmode |", false, modesWithOptions),
|
|
a -> assertCompletion(a, "/set mo newmode -|", false, options),
|
|
a -> assertCompletion(a, "/set mo newmode -command |", false),
|
|
a -> assertCompletion(a, "/set mo newmode normal |", false, options),
|
|
|
|
// /set prompt
|
|
a -> assertCompletion(a, "/set pro|", false, "prompt "),
|
|
a -> assertCompletion(a, "/set pro |", false, modes),
|
|
|
|
// /set start
|
|
a -> assertCompletion(a, "/set st|", false, "start "),
|
|
a -> assertCompletion(a, "/set st |", 1),
|
|
|
|
// /set truncation
|
|
a -> assertCompletion(a, "/set tr|", false, "truncation "),
|
|
a -> assertCompletion(a, "/set tr |", false, modes)
|
|
);
|
|
}
|
|
|
|
private void createIfNeeded(Path file) throws IOException {
|
|
if (!Files.exists(file))
|
|
Files.createFile(file);
|
|
}
|
|
private List<String> listFiles(Path path) throws IOException {
|
|
return listFiles(path, ACCEPT_ALL);
|
|
}
|
|
|
|
private List<String> listFiles(Path path, Predicate<? super Path> filter) throws IOException {
|
|
try (Stream<Path> stream = Files.list(path)) {
|
|
return stream.filter(filter)
|
|
.map(p -> p.getFileName().toString() + (Files.isDirectory(p) ? "/" : ""))
|
|
.sorted()
|
|
.collect(Collectors.toList());
|
|
}
|
|
}
|
|
|
|
private static final Predicate<? super Path> ACCEPT_ALL =
|
|
(file) -> !file.endsWith(".") && !file.endsWith("..");
|
|
|
|
private static final Predicate<? super Path> CLASSPATH_FILTER =
|
|
(file) -> ACCEPT_ALL.test(file) &&
|
|
(Files.isDirectory(file) ||
|
|
file.getFileName().toString().endsWith(".jar") ||
|
|
file.getFileName().toString().endsWith(".zip"));
|
|
|
|
}
|