All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jline.builtins.Completers Maven / Gradle / Ivy

/*
 * Copyright (c) 2002-2021, the original author or authors.
 *
 * This software is distributable under the BSD license. See the terms of the
 * BSD license in the documentation provided with this software.
 *
 * https://opensource.org/licenses/BSD-3-Clause
 */
package org.jline.builtins;

import java.io.File;
import java.io.IOException;
import java.lang.reflect.Array;
import java.nio.file.DirectoryStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
import java.util.function.Function;
import java.util.function.Supplier;

import org.jline.reader.Candidate;
import org.jline.reader.LineReader;
import org.jline.reader.LineReader.Option;
import org.jline.reader.impl.completer.NullCompleter;
import org.jline.reader.impl.completer.StringsCompleter;
import org.jline.reader.ParsedLine;
import org.jline.terminal.Terminal;
import org.jline.utils.AttributedString;
import org.jline.utils.AttributedStringBuilder;
import org.jline.utils.OSUtils;
import org.jline.utils.StyleResolver;

public class Completers {

    public interface CompletionEnvironment {
        Map> getCompletions();
        Set getCommands();
        String resolveCommand(String command);
        String commandName(String command);
        Object evaluate(LineReader reader, ParsedLine line, String func) throws Exception;
    }

    public static class CompletionData {
        public final List options;
        public final String description;
        public final String argument;
        public final String condition;

        public CompletionData(List options, String description, String argument, String condition) {
            this.options = options;
            this.description = description;
            this.argument = argument;
            this.condition = condition;
        }
    }

    public static class Completer implements org.jline.reader.Completer {

        private final CompletionEnvironment environment;

        public Completer(CompletionEnvironment environment) {
            this.environment = environment;
        }

        public void complete(LineReader reader, ParsedLine line, List candidates) {
            if (line.wordIndex() == 0) {
                completeCommand(candidates);
            } else {
                tryCompleteArguments(reader, line, candidates);
            }
        }

        protected void tryCompleteArguments(LineReader reader, ParsedLine line, List candidates) {
            String command = line.words().get(0);
            String resolved = environment.resolveCommand(command);
            Map> comp = environment.getCompletions();
            if (comp != null) {
                List cmd = comp.get(resolved);
                if (cmd != null) {
                    completeCommandArguments(reader, line, candidates, cmd);
                }
            }
        }

        protected void completeCommandArguments(LineReader reader, ParsedLine line, List candidates, List completions) {
            for (CompletionData completion : completions) {
                boolean isOption = line.word().startsWith("-");
                String prevOption = line.wordIndex() >= 2 && line.words().get(line.wordIndex() - 1).startsWith("-")
                        ? line.words().get(line.wordIndex() - 1) : null;
                String key = UUID.randomUUID().toString();
                boolean conditionValue = true;
                if (completion.condition != null) {
                    Object res = Boolean.FALSE;
                    try {
                        res = environment.evaluate(reader, line, completion.condition);
                    } catch (Throwable t) {
                        // Ignore
                    }
                    conditionValue = isTrue(res);
                }
                if (conditionValue && isOption && completion.options != null) {
                    for (String opt : completion.options) {
                        candidates.add(new Candidate(opt, opt, "options", completion.description, null, key, true));
                    }
                } else if (!isOption && prevOption != null && completion.argument != null
                        && (completion.options != null && completion.options.contains(prevOption))) {
                    Object res = null;
                    try {
                        res = environment.evaluate(reader, line, completion.argument);
                    } catch (Throwable t) {
                        // Ignore
                    }
                    if (res instanceof Candidate) {
                        candidates.add((Candidate) res);
                    } else if (res instanceof String) {
                        candidates.add(new Candidate((String) res, (String) res, null, null, null, null, true));
                    } else if (res instanceof Collection) {
                        for (Object s : (Collection) res) {
                            if (s instanceof Candidate) {
                                candidates.add((Candidate) s);
                            } else if (s instanceof String) {
                                candidates.add(new Candidate((String) s, (String) s, null, null, null, null, true));
                            }
                        }
                    } else if (res != null && res.getClass().isArray()) {
                        for (int i = 0, l = Array.getLength(res); i < l; i++) {
                            Object s = Array.get(res, i);
                            if (s instanceof Candidate) {
                                candidates.add((Candidate) s);
                            } else if (s instanceof String) {
                                candidates.add(new Candidate((String) s, (String) s, null, null, null, null, true));
                            }
                        }
                    }
                } else if (!isOption && completion.argument != null) {
                    Object res = null;
                    try {
                        res = environment.evaluate(reader, line, completion.argument);
                    } catch (Throwable t) {
                        // Ignore
                    }
                    if (res instanceof Candidate) {
                        candidates.add((Candidate) res);
                    } else if (res instanceof String) {
                        candidates.add(new Candidate((String) res, (String) res, null, completion.description, null, null, true));
                    } else if (res instanceof Collection) {
                        for (Object s : (Collection) res) {
                            if (s instanceof Candidate) {
                                candidates.add((Candidate) s);
                            } else if (s instanceof String) {
                                candidates.add(new Candidate((String) s, (String) s, null, completion.description, null, null, true));
                            }
                        }
                    }
                }
            }
        }

        protected void completeCommand(List candidates) {
            Set commands = environment.getCommands();
            for (String command : commands) {
                String name = environment.commandName(command);
                boolean resolved = command.equals(environment.resolveCommand(name));
                if (!name.startsWith("_")) {
                    String desc = null;
                    Map> comp = environment.getCompletions();
                    if (comp != null) {
                        List completions = comp.get(command);
                        if (completions != null) {
                            for (CompletionData completion : completions) {
                                if (completion.description != null
                                        && completion.options == null
                                        && completion.argument == null
                                        && completion.condition == null) {
                                    desc = completion.description;
                                }
                            }
                        }
                    }
                    String key = UUID.randomUUID().toString();
                    if (desc != null) {
                        candidates.add(new Candidate(command, command, null, desc, null, key, true));
                        if (resolved) {
                            candidates.add(new Candidate(name, name, null, desc, null, key, true));
                        }
                    } else {
                        candidates.add(new Candidate(command, command, null, null, null, key, true));
                        if (resolved) {
                            candidates.add(new Candidate(name, name, null, null, null, key, true));
                        }
                    }
                }
            }
        }

        private boolean isTrue(Object result) {
            if (result == null)
                return false;
            if (result instanceof Boolean)
                return (Boolean) result;
            if (result instanceof Number && 0 == ((Number) result).intValue()) {
                return false;
            }
            return !("".equals(result) || "0".equals(result));

        }

    }

    public static class DirectoriesCompleter extends FileNameCompleter {

        private final Supplier currentDir;

        public DirectoriesCompleter(File currentDir) {
            this(currentDir.toPath());
        }

        public DirectoriesCompleter(Path currentDir) {
            this.currentDir = () -> currentDir;
        }

        public DirectoriesCompleter(Supplier currentDir) {
            this.currentDir = currentDir;
        }

        @Override
        protected Path getUserDir() {
            return currentDir.get();
        }

        @Override
        protected boolean accept(Path path) {
            return Files.isDirectory(path) && super.accept(path);
        }
    }

    public static class FilesCompleter extends FileNameCompleter {

        private final Supplier currentDir;
        private final String namePattern;

        public FilesCompleter(File currentDir) {
            this(currentDir.toPath(), null);
        }

        public FilesCompleter(File currentDir, String namePattern) {
            this(currentDir.toPath(), namePattern);
        }

        public FilesCompleter(Path currentDir) {
            this(currentDir, null);
        }

        public FilesCompleter(Path currentDir, String namePattern) {
            this.currentDir = () -> currentDir;
            this.namePattern = compilePattern(namePattern);
        }

        public FilesCompleter(Supplier currentDir) {
            this(currentDir, null);
        }

        public FilesCompleter(Supplier currentDir, String namePattern) {
            this.currentDir = currentDir;
            this.namePattern = compilePattern(namePattern);
        }

        private String compilePattern(String pattern) {
            if (pattern == null) {
                return null;
            }
            StringBuilder sb = new StringBuilder();
            for (int i = 0; i < pattern.length(); i++) {
                char ch = pattern.charAt(i);
                if (ch == '\\') {
                    ch = pattern.charAt(++i);
                    sb.append(ch);
                } else if (ch == '.') {
                    sb.append('\\').append('.');
                } else if (ch == '*') {
                    sb.append('.').append('*');
                } else {
                    sb.append(ch);
                }
            }
            return sb.toString();
        }

        @Override
        protected Path getUserDir() {
            return currentDir.get();
        }

        @Override
        protected boolean accept(Path path) {
            if (namePattern == null || Files.isDirectory(path)) {
                return super.accept(path);
            }
            return path.getFileName().toString().matches(namePattern) && super.accept(path);
        }
    }

    /**
     * A file name completer takes the buffer and issues a list of
     * potential completions.
     * 

* This completer tries to behave as similar as possible to * bash's file name completion (using GNU readline) * with the following exceptions: *

    *
  • Candidates that are directories will end with "/"
  • *
  • Wildcard regular expressions are not evaluated or replaced
  • *
  • The "~" character can be used to represent the user's home, * but it cannot complete to other users' homes, since java does * not provide any way of determining that easily
  • *
* * @author Marc Prud'hommeaux * @author Jason Dillon * @since 2.3 */ public static class FileNameCompleter implements org.jline.reader.Completer { protected static StyleResolver resolver = Styles.lsStyle(); public void complete(LineReader reader, ParsedLine commandLine, final List candidates) { assert commandLine != null; assert candidates != null; String buffer = commandLine.word().substring(0, commandLine.wordCursor()); Path current; String curBuf; String sep = getSeparator(reader.isSet(LineReader.Option.USE_FORWARD_SLASH)); int lastSep = buffer.lastIndexOf(sep); try { if (lastSep >= 0) { curBuf = buffer.substring(0, lastSep + 1); if (curBuf.startsWith("~")) { if (curBuf.startsWith("~" + sep)) { current = getUserHome().resolve(curBuf.substring(2)); } else { current = getUserHome().getParent().resolve(curBuf.substring(1)); } } else { current = getUserDir().resolve(curBuf); } } else { curBuf = ""; current = getUserDir(); } try (DirectoryStream directory = Files.newDirectoryStream(current, this::accept)) { directory.forEach(p -> { String value = curBuf + p.getFileName().toString(); if (Files.isDirectory(p)) { candidates.add( new Candidate(value + (reader.isSet(LineReader.Option.AUTO_PARAM_SLASH) ? sep : ""), getDisplay(reader.getTerminal(), p, resolver, sep), null, null, reader.isSet(LineReader.Option.AUTO_REMOVE_SLASH) ? sep : null, null, false)); } else { candidates.add(new Candidate(value, getDisplay(reader.getTerminal(), p, resolver, sep), null, null, null, null, true)); } }); } catch (IOException e) { // Ignore } } catch (Exception e) { // Ignore } } protected boolean accept(Path path) { try { return !Files.isHidden(path); } catch (IOException e) { return false; } } protected Path getUserDir() { return Paths.get(System.getProperty("user.dir")); } protected Path getUserHome() { return Paths.get(System.getProperty("user.home")); } protected String getSeparator(boolean useForwardSlash) { return useForwardSlash ? "/" : getUserDir().getFileSystem().getSeparator(); } protected String getDisplay(Terminal terminal, Path p, StyleResolver resolver, String separator) { AttributedStringBuilder sb = new AttributedStringBuilder(); String name = p.getFileName().toString(); int idx = name.lastIndexOf("."); String type = idx != -1 ? ".*" + name.substring(idx): null; if (Files.isSymbolicLink(p)) { sb.styled(resolver.resolve(".ln"), name).append("@"); } else if (Files.isDirectory(p)) { sb.styled(resolver.resolve(".di"), name).append(separator); } else if (Files.isExecutable(p) && !OSUtils.IS_WINDOWS) { sb.styled(resolver.resolve(".ex"), name).append("*"); } else if (type != null && resolver.resolve(type).getStyle() != 0) { sb.styled(resolver.resolve(type), name); } else if (Files.isRegularFile(p)) { sb.styled(resolver.resolve(".fi"), name); } else { sb.append(name); } return sb.toAnsi(terminal); } } public static class TreeCompleter implements org.jline.reader.Completer { final Map completers = new HashMap<>(); final RegexCompleter completer; public TreeCompleter(Node... nodes) { this(Arrays.asList(nodes)); } public TreeCompleter(List nodes) { StringBuilder sb = new StringBuilder(); addRoots(sb, nodes); completer = new RegexCompleter(sb.toString(), completers::get); } public static Node node(Object... objs) { org.jline.reader.Completer comp = null; List cands = new ArrayList<>(); List nodes = new ArrayList<>(); for (Object obj : objs) { if (obj instanceof String) { cands.add(new Candidate((String) obj)); } else if (obj instanceof Candidate) { cands.add((Candidate) obj); } else if (obj instanceof Node) { nodes.add((Node) obj); } else if (obj instanceof org.jline.reader.Completer) { comp = (org.jline.reader.Completer) obj; } else { throw new IllegalArgumentException(); } } if (comp != null) { if (!cands.isEmpty()) { throw new IllegalArgumentException(); } return new Node(comp, nodes); } else if (!cands.isEmpty()) { return new Node((r, l, c) -> c.addAll(cands), nodes); } else { throw new IllegalArgumentException(); } } void addRoots(StringBuilder sb, List nodes) { if (!nodes.isEmpty()) { sb.append(" ( "); boolean first = true; for (Node n : nodes) { if (first) { first = false; } else { sb.append(" | "); } String name = "c" + completers.size(); completers.put(name, n.completer); sb.append(name); addRoots(sb, n.nodes); } sb.append(" ) "); } } @Override public void complete(LineReader reader, ParsedLine line, List candidates) { completer.complete(reader, line, candidates); } public static class Node { final org.jline.reader.Completer completer; final List nodes; public Node(org.jline.reader.Completer completer, List nodes) { this.completer = completer; this.nodes = nodes; } } } public static class RegexCompleter implements org.jline.reader.Completer { private final NfaMatcher matcher; private final Function completers; private final ThreadLocal reader = new ThreadLocal<>(); public RegexCompleter(String syntax, Function completers) { this.matcher = new NfaMatcher<>(syntax, this::doMatch); this.completers = completers; } @Override public synchronized void complete(LineReader reader, ParsedLine line, List candidates) { List words = line.words().subList(0, line.wordIndex()); this.reader.set(reader); Set next = matcher.matchPartial(words); for (String n : next) { completers.apply(n).complete(reader, new ArgumentLine(line.word(), line.wordCursor()), candidates); } this.reader.set(null); } private boolean doMatch(String arg, String name) { List candidates = new ArrayList<>(); LineReader r = reader.get(); boolean caseInsensitive = r != null && r.isSet(Option.CASE_INSENSITIVE); completers.apply(name).complete(r, new ArgumentLine(arg, arg.length()), candidates); return candidates.stream().anyMatch(c -> caseInsensitive ? c.value().equalsIgnoreCase(arg) : c.value().equals(arg)); } public static class ArgumentLine implements ParsedLine { private final String word; private final int cursor; public ArgumentLine(String word, int cursor) { this.word = word; this.cursor = cursor; } @Override public String word() { return word; } @Override public int wordCursor() { return cursor; } @Override public int wordIndex() { return 0; } @Override public List words() { return Collections.singletonList(word); } @Override public String line() { return word; } @Override public int cursor() { return cursor; } } } public static class OptDesc { private String shortOption; private String longOption; private String description; private org.jline.reader.Completer valueCompleter; protected static List compile(Map> optionValues, Collection options) { List out = new ArrayList<>(); for (Map.Entry> entry: optionValues.entrySet()) { if (entry.getKey().startsWith("--")) { out.add(new OptDesc(null, entry.getKey(), new StringsCompleter(entry.getValue()))); } else if (entry.getKey().matches("-[a-zA-Z]")) { out.add(new OptDesc(entry.getKey(), null, new StringsCompleter(entry.getValue()))); } } for (String o: options) { if (o.startsWith("--")) { out.add(new OptDesc(null, o)); } else if (o.matches("-[a-zA-Z]")) { out.add(new OptDesc(o, null)); } } return out; } /** * Command option description. If option does not have short/long option assign to it null value. * If option does not have value set valueCompleter = NullCompleter.INSTANCE * @param shortOption short option * @param longOption long option * @param description short option description * @param valueCompleter option value completer */ public OptDesc(String shortOption, String longOption, String description, org.jline.reader.Completer valueCompleter) { this.shortOption = shortOption; this.longOption = longOption; this.description = description; this.valueCompleter = valueCompleter; } /** * Command option description. If option does not have short/long option assign to it null value. * If option does not have value set valueCompleter = NullCompleter.INSTANCE * @param shortOption short option * @param longOption long option * @param valueCompleter option value completer */ public OptDesc(String shortOption, String longOption, org.jline.reader.Completer valueCompleter) { this(shortOption, longOption, null, valueCompleter); } /** * Command option description. If option does not have short/long option assign to it null value. * @param shortOption short option * @param longOption long option * @param description short option description */ public OptDesc(String shortOption, String longOption, String description) { this(shortOption, longOption, description, null); } /** * Command option description. If option does not have short/long option assign to it null value. * @param shortOption short option * @param longOption long option */ public OptDesc(String shortOption, String longOption) { this(shortOption, longOption, null, null); } protected OptDesc() { } public void setValueCompleter(org.jline.reader.Completer valueCompleter) { this.valueCompleter = valueCompleter; } public String longOption() { return longOption; } public String shortOption() { return shortOption; } public String description() { return description; } protected boolean hasValue() { return valueCompleter != null && valueCompleter != NullCompleter.INSTANCE; } protected org.jline.reader.Completer valueCompleter() { return valueCompleter; } protected void completeOption(LineReader reader, final ParsedLine commandLine, List candidates, boolean longOpt) { if (!longOpt) { if (shortOption != null) { candidates.add(new Candidate(shortOption, shortOption, null, description, null, null, false)); } } else if (longOption != null) { if (hasValue()) { candidates.add(new Candidate(longOption + "=", longOption, null, description, null, null, false)); } else { candidates.add(new Candidate(longOption, longOption, null, description, null, null, true)); } } } protected boolean completeValue(LineReader reader, final ParsedLine commandLine, List candidates, String curBuf, String partialValue) { boolean out = false; List temp = new ArrayList<>(); ParsedLine pl = reader.getParser().parse(partialValue, partialValue.length()); valueCompleter.complete(reader, pl, temp); for (Candidate c : temp) { String v = c.value(); if (v.startsWith(partialValue)) { out = true; String val = c.value(); if (valueCompleter instanceof FileNameCompleter) { FileNameCompleter cc = (FileNameCompleter)valueCompleter; String sep = cc.getSeparator(reader.isSet(LineReader.Option.USE_FORWARD_SLASH)); val = cc.getDisplay(reader.getTerminal(), Paths.get(c.value()), FileNameCompleter.resolver, sep); } candidates.add(new Candidate(curBuf + v, val, null, null, null, null, c.complete())); } } return out; } protected boolean match(String option) { return (shortOption != null && shortOption.equals(option)) || (longOption != null && longOption.equals(option)); } protected boolean startsWith(String option) { return (shortOption != null && shortOption.startsWith(option)) || (longOption != null && longOption.startsWith(option)); } } public static class OptionCompleter implements org.jline.reader.Completer { private Function> commandOptions; private Collection options; private List argsCompleters = new ArrayList<>(); private int startPos; /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param completer command parameter completer * @param commandOptions command options descriptions * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(org.jline.reader.Completer completer, Function> commandOptions, int startPos) { this.startPos = startPos; this.commandOptions = commandOptions; this.argsCompleters.add(completer); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param completers command parameters completers * @param commandOptions command options descriptions * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(List completers, Function> commandOptions, int startPos) { this.startPos = startPos; this.commandOptions = commandOptions; this.argsCompleters = new ArrayList<>(completers); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param completers command parameters completers * @param optionValues command value options as map key and its possible values as map value * @param options command options that do not have value * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(List completers, Map> optionValues, Collection options, int startPos) { this(optionValues, options, startPos); this.argsCompleters = new ArrayList<>(completers); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param completer command parameter completer * @param optionValues command value options as map key and its possible values as map value * @param options command options that do not have value * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(org.jline.reader.Completer completer, Map> optionValues, Collection options, int startPos) { this(optionValues, options, startPos); this.argsCompleters.add(completer); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param optionValues command value options as map key and its possible values as map value * @param options command options that do not have value * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(Map> optionValues, Collection options, int startPos) { this(OptDesc.compile(optionValues, options), startPos); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param completer command parameter completer * @param options command options that do not have value * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(org.jline.reader.Completer completer, Collection options, int startPos) { this(options, startPos); this.argsCompleters.add(completer); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param completers command parameters completers * @param options command options that do not have value * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(List completers, Collection options, int startPos) { this(options, startPos); this.argsCompleters = new ArrayList<>(completers); } /** * OptionCompleter completes command options and parameters. OptionCompleter should be used as an argument of ArgumentCompleter * @param options command options that do not have value * @param startPos OptionCompleter position in ArgumentCompleter parameters */ public OptionCompleter(Collection options, int startPos) { this.options = options; this.startPos = startPos; } public void setStartPos(int startPos) { this.startPos = startPos; } @Override public void complete(LineReader reader, final ParsedLine commandLine, List candidates) { assert commandLine != null; assert candidates != null; List words = commandLine.words(); String buffer = commandLine.word().substring(0, commandLine.wordCursor()); if (startPos >= words.size()) { candidates.add(new Candidate(buffer, buffer, null, null, null, null, true)); return; } String command = reader.getParser().getCommand(words.get(startPos - 1)); if (buffer.startsWith("-")) { boolean addbuff = true; boolean valueCandidates = false; boolean longOption = buffer.startsWith("--"); int eq = buffer.matches("-[a-zA-Z][a-zA-Z0-9]+") ? 2 : buffer.indexOf('='); if (eq < 0) { List usedOptions = new ArrayList<>(); for (int i = startPos; i < words.size(); i++) { if (words.get(i).startsWith("-")) { String w = words.get(i); int ind = w.indexOf('='); if (ind < 0) { usedOptions.add(w); } else { usedOptions.add(w.substring(0,ind)); } } } for (OptDesc o : commandOptions == null ? options : commandOptions.apply(command)) { if (usedOptions.contains(o.shortOption()) || usedOptions.contains(o.longOption())) { continue; } if (o.startsWith(buffer)) { addbuff = false; } o.completeOption(reader, commandLine, candidates, longOption); } } else { addbuff = false; int nb = buffer.contains("=") ? 1 : 0; String value = buffer.substring(eq + nb); String curBuf = buffer.substring(0, eq + nb); String opt = buffer.substring(0, eq); OptDesc option = findOptDesc(command, opt); if (option.hasValue()) { valueCandidates = option.completeValue(reader, commandLine, candidates, curBuf, value); } } if ((buffer.contains("=") && !buffer.endsWith("=") && !valueCandidates) || addbuff) { candidates.add(new Candidate(buffer, buffer, null, null, null, null, true)); } } else if (words.size() > 1 && shortOptionValueCompleter(command, words.get(words.size() - 2)) != null) { shortOptionValueCompleter(command, words.get(words.size() - 2)).complete(reader, commandLine, candidates); } else if (words.size() > 1 && longOptionValueCompleter(command, words.get(words.size() - 2)) != null) { longOptionValueCompleter(command, words.get(words.size() - 2)).complete(reader, commandLine, candidates); } else if (!argsCompleters.isEmpty()) { int args = -1; for (int i = startPos; i < words.size(); i++) { if (!words.get(i).startsWith("-")) { if (i > 0 && shortOptionValueCompleter(command, words.get(i - 1)) == null && longOptionValueCompleter(command, words.get(i - 1)) == null) { args++; } } } if (args == -1) { candidates.add(new Candidate(buffer, buffer, null, null, null, null, true)); } else if (args < argsCompleters.size()) { argsCompleters.get(args).complete(reader, commandLine, candidates); } else { argsCompleters.get(argsCompleters.size() - 1).complete(reader, commandLine, candidates); } } } private org.jline.reader.Completer longOptionValueCompleter(String command, String opt) { if (!opt.matches("--[a-zA-Z]+")) { return null; } Collection optDescs = commandOptions == null ? options : commandOptions.apply(command); OptDesc option = findOptDesc(optDescs, opt); return option.hasValue() ? option.valueCompleter() : null; } private org.jline.reader.Completer shortOptionValueCompleter(String command, String opt) { if (!opt.matches("-[a-zA-Z]+")) { return null; } org.jline.reader.Completer out = null; Collection optDescs = commandOptions == null ? options : commandOptions.apply(command); if (opt.length() == 2) { out = findOptDesc(optDescs, opt).valueCompleter(); } else if (opt.length() > 2) { for (int i = 1; i < opt.length(); i++) { OptDesc o = findOptDesc(optDescs, "-" + opt.charAt(i)); if (o.shortOption() == null) { return null; } else if (out == null) { out = o.valueCompleter(); } } } return out; } private OptDesc findOptDesc(String command, String opt) { return findOptDesc(commandOptions == null ? options : commandOptions.apply(command), opt); } private OptDesc findOptDesc(Collection optDescs, String opt) { for (OptDesc o : optDescs) { if (o.match(opt)) { return o; } } return new OptDesc(); } } public static class AnyCompleter implements org.jline.reader.Completer { public static final AnyCompleter INSTANCE = new AnyCompleter(); @Override public void complete(LineReader reader, ParsedLine commandLine, List candidates) { assert commandLine != null; assert candidates != null; String buffer = commandLine.word().substring(0, commandLine.wordCursor()); candidates.add(new Candidate(AttributedString.stripAnsi(buffer) , buffer, null, null, null, null, true)); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy