Java commons-cli, opções com lista de valores possíveis
Pergunta
Como posso fazer uma opção aceitar apenas alguns valores especificados como no exemplo a seguir:
$ java -jar Mumu.jar -a foo
OK
$ java -jar Mumu.jar -a bar
OK
$ java -jar Mumu.jar -a foobar
foobar is not a valid value for -a
Solução
Desde commons-cli não suporta isso diretamente, a solução mais simples é, provavelmente, para verificar o valor de uma opção quando você obtê-lo.
Outras dicas
A outra forma pode ser estender a classe Option. No trabalho que fizemos que:
public static class ChoiceOption extends Option {
private final String[] choices;
public ChoiceOption(
final String opt,
final String longOpt,
final boolean hasArg,
final String description,
final String... choices) throws IllegalArgumentException {
super(opt, longOpt, hasArg, description + ' ' + Arrays.toString(choices));
this.choices = choices;
}
public String getChoiceValue() throws RuntimeException {
final String value = super.getValue();
if (value == null) {
return value;
}
if (ArrayUtils.contains(choices, value)) {
return value;
}
throw new RuntimeException( value " + describe(this) + " should be one of " + Arrays.toString(choices));
}
@Override
public boolean equals(final Object o) {
if (this == o) {
return true;
} else if (o == null || getClass() != o.getClass()) {
return false;
}
return new EqualsBuilder().appendSuper(super.equals(o))
.append(choices, ((ChoiceOption) o).choices)
.isEquals();
}
@Override
public int hashCode() {
return new ashCodeBuilder().appendSuper(super.hashCode()).append(choices).toHashCode();
}
}
Eu queria esse tipo de comportamento antes, e nunca me deparei com uma maneira de fazer isso com um método já fornecidas. Isso não quer dizer que não existe. Um tipo de forma coxo, é adicionar o código de si mesmo, como:
private void checkSuitableValue(CommandLine line) {
if(line.hasOption("a")) {
String value = line.getOptionValue("a");
if("foo".equals(value)) {
println("OK");
} else if("bar".equals(value)) {
println("OK");
} else {
println(value + "is not a valid value for -a");
System.exit(1);
}
}
}
Obviamente não haveria maneiras mais agradáveis ??de fazer isso do que a longa if / else, possivelmente com uma enum
, mas que deve ser tudo o que você precisa. Também não compilamos isso, mas eu acho que deve funcionar.
Este exemplo também não faz o "-a" mudar obrigatória, uma vez que não foi especificado na questão.