package net.knarcraft.ffmpegconverter.utility;
import net.knarcraft.ffmpegconverter.parser.ConverterArgument;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* A class to help with command parsing
*/
public final class Parser {
private Parser() {
}
/**
* This function parses the input to understandable converter instructions
*
* @param input
The input string to parse.
* @param validArguments All arguments which are considered valid.
* @return A map with all parsed arguments.
*/
static Map parse(String input, List validArguments) {
return parse(tokenize(input), validArguments);
}
/**
* This function parses command inputs into understandable converter instructions
*
* @param tokens A list of tokens containing all arguments
* @param validArguments A list of arguments which are considered valid.
* @return A map with all parsed arguments.
*/
private static Map parse(List tokens, List validArguments) {
Map parsedArguments = new HashMap<>();
while (!tokens.isEmpty()) {
parseArgument(tokens, validArguments, parsedArguments);
}
return parsedArguments;
}
/**
* Parses the first found token as an argument
*
* @param tokens The tokens to parse.
* @param converterArguments A list of all the valid arguments in existence.
* @param parsedArguments The map to store the parsed argument to.
*/
private static void parseArgument(List tokens, List converterArguments, Map parsedArguments) {
String currentToken = tokens.remove(0);
List foundArguments;
if (currentToken.startsWith("--")) {
String argumentName = currentToken.substring(2);
foundArguments = ListUtil.getMatching(converterArguments, (item) -> item.getName().equals(argumentName));
} else if (currentToken.startsWith("-")) {
char argumentShorthand = currentToken.substring(1).charAt(0);
foundArguments = ListUtil.getMatching(converterArguments, (item) -> item.getShorthand() == argumentShorthand);
} else {
throw new IllegalArgumentException("Unexpected value when not given an argument.");
}
if (foundArguments.isEmpty()) {
throw new IllegalArgumentException(String.format("Invalid argument %s encountered.", currentToken));
}
ConverterArgument foundArgument = foundArguments.get(0);
storeArgumentValue(tokens, foundArgument, parsedArguments);
}
/**
* Stores the value of a found argument to parsed arguments
*
* @param tokens The token list to parse.
* @param foundArgument The found argument to store.
* @param parsedArguments The map to store parsed arguments to.
*/
private static void storeArgumentValue(List tokens, ConverterArgument foundArgument, Map parsedArguments) {
String argumentValue;
if (tokens.isEmpty()) {
argumentValue = "";
} else {
argumentValue = tokens.get(0);
}
boolean valueIsArgument = argumentValue.startsWith("-");
if (valueIsArgument) {
if (foundArgument.isValueRequired()) {
throw new IllegalArgumentException(String.format("Argument %s requires a value, but no value was given.", foundArgument.getName()));
} else {
parsedArguments.put(foundArgument.getName(), "true");
}
} else {
String value;
if (tokens.isEmpty()) {
value = "";
} else {
value = tokens.remove(0);
}
if (!foundArgument.testArgumentValue(value)) {
throw new IllegalArgumentException(String.format("Invalid value %s for argument %s.", value, foundArgument.getName()));
}
parsedArguments.put(foundArgument.getName(), value);
}
}
/**
* Tokenizes a string
*
* @param input A string.
* @return A list of tokens.
*/
public static List tokenize(String input) {
List tokens = new ArrayList<>();
boolean startedQuote = false;
StringBuilder currentToken = new StringBuilder();
for (int index = 0; index < input.length(); index++) {
char character = input.charAt(index);
switch (character) {
case ' ':
if (tokenizeSpace(startedQuote, currentToken, tokens)) {
currentToken = new StringBuilder();
}
break;
case '"':
if (startedQuote) {
//This quote signifies the end of the argument
if (isNotEmpty(currentToken)) {
tokens.add(currentToken.toString());
currentToken = new StringBuilder();
}
startedQuote = false;
} else {
//This quote signifies the start of the argument
startedQuote = true;
currentToken = new StringBuilder();
}
break;
default:
tokenizeNormalCharacter(currentToken, character, input.length(), index, tokens);
break;
}
}
return tokens;
}
/**
* Adds a normal character to the token. Adds the current token to tokens if at the end of the input
*
* @param currentToken The string builder containing the current token.
* @param character The character found in the input.
* @param inputLength The length of the given input
* @param index The index of the read character.
* @param tokens The list of processed tokens.
*/
private static void tokenizeNormalCharacter(StringBuilder currentToken, char character, int inputLength, int index,
List tokens) {
currentToken.append(character);
if (index == inputLength - 1) {
tokens.add(currentToken.toString());
}
}
/**
* Tokenizes a space character
*
* @param startedQuote Whether this space is inside a pair of quotes.
* @param currentToken The string builder containing the current token.
* @param tokens The list of processed tokens.
* @return True if the token is finished.
*/
private static boolean tokenizeSpace(boolean startedQuote, StringBuilder currentToken, List tokens) {
if (!startedQuote) {
//If not inside "", a space marks the end of a parameter
if (isNotEmpty(currentToken)) {
tokens.add(currentToken.toString());
}
return true;
} else {
currentToken.append(' ');
return false;
}
}
/**
* Checks whether a string builder is empty
*
* @param builder The string builder to check.
* @return True if the string builder is non empty.
*/
private static boolean isNotEmpty(StringBuilder builder) {
return !builder.toString().trim().isEmpty();
}
}