net.morimekta.console.args.ArgumentParser Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of console-util Show documentation
Show all versions of console-util Show documentation
Utilities helping with various *nix console topics. Mostly geared
toward expressive and interactive command line applications.
/*
* Copyright (c) 2016, Stein Eldar Johnsen
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package net.morimekta.console.args;
import net.morimekta.console.chr.CharUtil;
import net.morimekta.util.Strings;
import net.morimekta.util.io.IndentedPrintWriter;
import net.morimekta.util.json.JsonException;
import net.morimekta.util.json.JsonTokenizer;
import org.apache.commons.lang3.text.WordUtils;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import static java.nio.charset.StandardCharsets.UTF_8;
/**
* Argument argumentParser class. This is the actual argumentParser that is initialized with
* a set of options, arguments and properties, and is then initialized with
* the appropriate fields.
*/
public class ArgumentParser {
/**
* Create an argument argumentParser instance.
*
* @param program The program name.
* @param version The program version.
* @param description The program description.
*/
public ArgumentParser(String program, String version, String description) {
this(program, version, description, ArgumentOptions.defaults());
}
/**
* Create an argument argumentParser instance.
*
* @param program The program name.
* @param version The program version.
* @param description The program description.
* @param argumentOptions The argument options.
*/
public ArgumentParser(String program, String version, String description, ArgumentOptions argumentOptions) {
this.program = program;
this.version = version;
this.description = description;
this.argumentOptions = argumentOptions;
this.options = new ArrayList<>();
this.arguments = new ArrayList<>();
this.longNameOptions = new HashMap<>();
this.shortOptions = new HashMap<>();
this.parent = null;
}
/**
* Create a sub-command argument parser for the given sub-command.
*
* @param parent Parent argument parser to defer with on unknown options.
* @param subCommand The name of the sub-command the parser is created for.
* @param description The sub-command description.
*/
public ArgumentParser(ArgumentParser parent, String subCommand, String description) {
this.program = parent.program + " " + subCommand;
this.version = parent.version;
this.description = description;
this.argumentOptions = parent.argumentOptions;
this.options = new ArrayList<>();
this.arguments = new ArrayList<>();
this.longNameOptions = new HashMap<>();
this.shortOptions = new HashMap<>();
this.parent = parent;
}
/**
* The name of the program. Should be essentially what the user types on
* the command line to invoke the program.
*
* @return The program name.
*/
public String getProgram() {
return program;
}
/**
* The program version string.
*
* @return The program version.
*/
public String getVersion() {
return version;
}
/**
* Short description of the program. Should be the string that is shown
* on the top of the program usage help, usually just a few words. Should
* be capitalized.
*
* @return The program description.
*/
public String getDescription() {
return description;
}
/**
* Add a command line option.
*
* @param option The option to add.
* @param The base option type.
* @return The argument argumentParser.
*/
public ArgumentParser add(O option) {
if (option.getName() != null) {
if (longNameOptions.containsKey(option.getName())) {
throw new IllegalArgumentException("Option " + option.getName() + " already exists");
}
if (parent != null && parent.longNameOptions.containsKey(option.getName())) {
throw new IllegalArgumentException("Option " + option.getName() + " already exists in parent");
}
longNameOptions.put(option.getName(), option);
}
if (option instanceof Flag) {
String negate = ((Flag) option).getNegateName();
if (negate != null) {
if (longNameOptions.containsKey(negate)) {
throw new IllegalArgumentException("Flag " + negate + " already exists");
}
if (parent != null && parent.longNameOptions.containsKey(negate)) {
throw new IllegalArgumentException("Flag " + negate + " already exists in parent");
}
longNameOptions.put(negate, option);
}
}
if (option.getShortNames()
.length() > 0) {
for (char s : option.getShortNames()
.toCharArray()) {
if (shortOptions.containsKey(s)) {
throw new IllegalArgumentException("Short option -" + s + " already exists");
}
if (parent != null && parent.shortOptions.containsKey(s)) {
throw new IllegalArgumentException("Short option -" + s + " already exists in parent");
}
shortOptions.put(s, option);
}
}
this.options.add(option);
return this;
}
/**
* Add a sub-command.
*
* @param arg The command to add.
* @param The base argument type.
* @return The argument argumentParser.
*/
public ArgumentParser add(A arg) {
if (arg instanceof BaseOption) {
return add((BaseOption) arg);
}
// No arguments can be added after a sub-command-set.
if (arguments.size() > 0 && arguments.get(arguments.size() - 1) instanceof SubCommandSet) {
throw new IllegalArgumentException("No arguments can be added after a sub-command set");
}
arguments.add(arg);
return this;
}
/**
* Parse arguments from the main method.
*
* @param args The argument array.
*/
public void parse(String... args) {
parse(new ArgumentList(args));
}
/**
* Parse arguments from the main method.
*
* @param args The argument list.
*/
public void parse(ArgumentList args) {
try {
parseInternal(args);
} catch (ArgumentException e) {
if (e.getParser() == null) {
e.setParser(this);
}
throw e;
}
}
private void parseInternal(ArgumentList args) {
while (args.remaining() > 0) {
String cur = args.get(0);
if (cur.equals("--")) {
// The remaining *must* be arguments / sub-commands.
args.consume(1);
break;
} else if (cur.startsWith("--")) {
// long opt.
if (cur.contains("=")) {
cur = cur.substring(0, cur.indexOf("="));
}
BaseOption opt = getLongNameOption(cur);
if (opt == null) {
throw new ArgumentException("No option for " + cur);
}
args.consume(opt.apply(args));
} else if (cur.startsWith("-")) {
// short opt.
String remaining = cur.substring(1);
int skip = 0;
while (remaining.length() > 0) {
BaseOption opt = getShortNameOption(remaining.charAt(0));
if (opt == null) {
throw new ArgumentException("No short opt for -" + remaining.charAt(0));
}
skip = opt.applyShort(remaining, args);
if (skip == 0) {
remaining = remaining.substring(1);
} else {
break;
}
}
args.consume(Math.max(1, skip));
} else {
if (cur.startsWith("@")) {
File f = new File(cur.substring(1));
if (f.exists() && f.isFile()) {
try (FileInputStream fis = new FileInputStream(f);
BufferedReader reader = new BufferedReader(new InputStreamReader(fis, UTF_8))) {
List lines = reader.lines()
.map(String::trim)
// strip empty lines and commented lines
.filter(l -> !(l.isEmpty() || l.startsWith("#")))
// be smart about splitting lines into single args.
.flatMap(this::argFileLineStream)
.collect(Collectors.toList());
if (lines.size() > 0) {
ArgumentList list = new ArgumentList(lines.toArray(new String[lines.size()]));
parse(list);
}
} catch (ArgumentException e) {
throw new ArgumentException(e, "Argument file " + f.getName() + ": " + e.getMessage())
.setParser(e.getParser());
} catch (IOException e) {
throw new ArgumentException(e, e.getMessage());
}
args.consume(1);
continue;
}
}
// Argument / sub-command.
int consumed = 0;
for (BaseArgument arg : arguments) {
consumed = arg.apply(args);
if (consumed > 0) {
break;
}
}
if (consumed == 0) {
throw new ArgumentException("No option found for " + args.get(0));
}
args.consume(consumed);
}
}
// Then consume the rest as arguments.
while (args.remaining() > 0) {
// Argument / sub-command.
int consumed = 0;
for (BaseArgument arg : arguments) {
consumed = arg.apply(args);
if (consumed > 0) {
break;
}
}
if (consumed == 0) {
throw new ArgumentException("No argument found for " + args.get(0));
}
args.consume(consumed);
}
}
private Stream argFileLineStream(String line) {
// If the line is double quoted (entirely), treat the whole
// as a json-escaped string argument.
if (line.startsWith("\"") && line.endsWith("\"")) {
try (StringReader lin = new StringReader(line)) {
JsonTokenizer tokenizer = new JsonTokenizer(lin);
String val = tokenizer.expectString("argument").decodeJsonLiteral();
if (lin.read() != -1) {
throw new ArgumentException("Garbage after quoted string argument: " + line);
}
return Arrays.stream(new String[]{val});
} catch (IOException | JsonException e) {
throw new ArgumentException(e, e.getMessage());
}
}
// otherwise split on the first space,
// and treat each part an arg entry, so that
// "--something and more"
// becomes:
// ["--something", "and more"]
return Arrays.stream(line.split("[ ]", 2))
.map(String::trim);
}
private BaseOption getLongNameOption(String name) {
BaseOption option = longNameOptions.get(name);
if (option == null && parent != null) {
option = parent.getLongNameOption(name);
}
return option;
}
private BaseOption getShortNameOption(char c) {
BaseOption option = shortOptions.get(c);
if (option == null && parent != null) {
option = parent.getShortNameOption(c);
}
return option;
}
/**
* Validate all options and arguments.
*/
public void validate() {
options.forEach(BaseArgument::validate);
arguments.forEach(BaseArgument::validate);
}
/**
* Print the option usage list. Essentially printed as a list of options
* with the description indented where it overflows the available line
* width.
*
* @param out The output stream.
*/
public void printUsage(OutputStream out) {
printUsage(out, false);
}
/**
* Print the option usage list. Essentially printed as a list of options
* with the description indented where it overflows the available line
* width.
*
* @param out The output stream.
* @param showHidden Whether to show hidden options.
*/
public void printUsage(OutputStream out, boolean showHidden) {
printUsage(new PrintWriter(new OutputStreamWriter(out, UTF_8)), showHidden);
}
/**
* Print the option usage list. Essentially printed as a list of options
* with the description indented where it overflows the available line
* width.
*
* @param writer The output printer.
* @param showHidden Whether to show hidden options.
*/
public void printUsage(PrintWriter writer, boolean showHidden) {
if (writer instanceof IndentedPrintWriter) {
printUsageInternal((IndentedPrintWriter) writer, showHidden);
} else {
printUsageInternal(new IndentedPrintWriter(writer), showHidden);
}
}
/**
* Get the program description line. Contains essentially the line
* "description - version".
*
* @return The program description.
*/
public String getProgramDescription() {
return description + " - " + version;
}
/**
* Get the single line usage string for the parser. Contains essentially
* the line "program options args".
*
* @return The single line usage.
*/
public String getSingleLineUsage() {
StringBuilder writer = new StringBuilder();
writer.append(program);
// first just list up all the unary short opts.
StringBuilder sh = new StringBuilder();
// Only include the first short name form the flag.
options.stream()
.filter(opt -> opt instanceof Flag)
.filter(opt -> opt.getShortNames().length() > 0)
.forEachOrdered(opt -> sh.append(opt.getShortNames().charAt(0)));
if (sh.length() > 0) {
writer.append(" [-").append(sh.toString()).append(']');
}
for (BaseOption opt : options) {
if (opt instanceof Flag && opt.getShortNames().length() > 0) {
// already included as short opt.
continue;
}
String usage = opt.getSingleLineUsage();
if (usage != null) {
writer.append(' ').append(usage);
}
}
for (BaseArgument arg : arguments) {
String usage = arg.getSingleLineUsage();
if (usage != null) {
writer.append(' ').append(usage);
}
}
return writer.toString();
}
private void printUsageInternal(IndentedPrintWriter writer, boolean showHidden) {
int usageWidth = argumentOptions.getUsageWidth();
int prefixLen = 0;
for (BaseOption opt : options) {
prefixLen = Math.max(prefixLen, opt.getPrefix().length());
}
for (BaseArgument arg : arguments) {
prefixLen = Math.max(prefixLen, arg.getPrefix().length());
}
prefixLen = Math.min(prefixLen, (usageWidth / 3) - USAGE_EXTRA_CHARS);
String indent = Strings.times(" ", prefixLen + USAGE_EXTRA_CHARS);
boolean first = true;
if (options.size() > 0) {
if (argumentOptions.getOptionComparator() != null) {
options.sort(argumentOptions.getOptionComparator());
}
for (BaseOption opt : options) {
if (opt.isHidden() && !showHidden) {
continue;
}
String prefix = opt.getPrefix();
String usage = opt.getUsage();
if (argumentOptions.isDefaultsShown() && opt.getDefaultValue() != null) {
usage = usage + " (default: " + opt.getDefaultValue() + ")";
}
if (first) {
first = false;
} else {
writer.appendln();
}
writer.begin(indent);
printSingleUsageEntry(writer, prefix, usage, prefixLen, usageWidth);
writer.end();
}
}
if (arguments.size() > 0) {
for (BaseArgument arg : arguments) {
if (arg.isHidden() && !showHidden) {
continue;
}
String prefix = arg.getPrefix();
String usage = arg.getUsage();
if (argumentOptions.isDefaultsShown() && arg.getDefaultValue() != null) {
usage = usage + " (default: " + arg.getDefaultValue() + ")";
}
if (first) {
first = false;
} else {
writer.appendln();
}
writer.begin(indent);
printSingleUsageEntry(writer, prefix, usage, prefixLen, usageWidth);
writer.end();
}
if (arguments.get(arguments.size() - 1) instanceof SubCommandSet &&
argumentOptions.isSubCommandsShown()) {
SubCommandSet scs = (SubCommandSet) arguments.get(arguments.size() - 1);
if (!first) {
writer.newline();
}
writer.appendln(argumentOptions.getSubCommandsString());
writer.newline();
writer.appendln();
scs.printUsage(writer, showHidden);
first = true;
}
}
// if nothing printed, complete nothing.
if (!first) {
writer.newline();
}
writer.flush();
}
static void printSingleUsageEntry(IndentedPrintWriter writer,
String prefix,
String usage,
int prefixLen,
int usageWidth) {
String[] descLines;
int printLinesFrom = 0;
writer.append(" ");
if (prefix.length() > prefixLen) {
writer.append(prefix);
if (prefix.length() > (prefixLen * 1.7)) {
descLines = WordUtils.wrap(usage, usageWidth - prefixLen - USAGE_EXTRA_CHARS).split("[\\r]?[\\n]");
} else {
writer.append(" : ");
String[] tmp = WordUtils.wrap(usage, usageWidth - prefix.length() - USAGE_EXTRA_CHARS).split("[\\r]?[\\n]", 2);
writer.append(tmp[0]);
if (tmp.length > 1) {
descLines = WordUtils.wrap(tmp[1].replaceAll("[\\r]?[\\n]", " "), usageWidth - prefixLen - USAGE_EXTRA_CHARS).split("[\\r]?[\\n]");
} else {
descLines = new String[0];
}
}
} else {
writer.append(CharUtil.rightPad(prefix, prefixLen));
writer.append(" : ");
descLines = WordUtils.wrap(usage, usageWidth - prefixLen - USAGE_EXTRA_CHARS).split("[\\r]?[\\n]");
writer.append(descLines[0]);
printLinesFrom = 1;
}
for (int i = printLinesFrom; i < descLines.length; ++i) {
writer.appendln(descLines[i]);
}
}
static final int USAGE_EXTRA_CHARS = 4;
private final ArgumentParser parent;
private final String program;
private final String version;
private final String description;
private final ArgumentOptions argumentOptions;
private final ArrayList options;
private final ArrayList arguments;
private final Map longNameOptions;
private final Map shortOptions;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy