Чтобы суммировать отношения, эти параметры должны иметь:
-B
, -A1
, -A2
и -A3
все требуют параметр -A
-B
запрещает любой из параметров -A1
, -A2
и -A3
-A1
, -A2
и -A3
позволяют друг другу - опция
-A
позволяет (но не требует) -B
, -A1
, -A2
и -A3
options
Одних аннотаций picocli будет недостаточно для декларативного express всех этих отношений, потребуется некоторая пользовательская проверка.
Так что мы могли бы также упростить и создать группа с одним аргументом, так как мы не можем express требование 2 (-B является эксклюзивным с -A1, -A2, -A3) одновременно с требованием 1 и 3 (-B, -A1, -A2 и -A3 все требуют -A и -A1, -A2, -A3 разрешают друг другу).
Одна группа, такая как [-A [-B] [-A1] [-A2] [-A3]]
, позаботится о некоторых проверках: все, кроме требования 2 (-B является ex кластеризуется с -A1, -A2, -A3). Для требования 2 нам нужно кодировать некоторую пользовательскую проверку в приложении (пример ниже).
Для вашего случая использования может быть полезно иметь пользовательский краткий обзор, который точно отражает отношения между опциями. Примерно так:
Usage: app [-hV] [-A [-B]]
app [-hV] [-A [-A1] [-A2] [-A3]]
Пример кода для достижения этой цели:
import picocli.CommandLine;
import picocli.CommandLine.*;
import picocli.CommandLine.Model.CommandSpec;
@Command(name = "app", mixinStandardHelpOptions = true,
synopsisHeading = "",
customSynopsis = {
"Usage: app [-hV] [-A [-B]]",
" app [-hV] [-A [-A1] [-A2] [-A3]]",
})
public class App implements Runnable {
static class MyGroup {
@Option(names = "-A", required = true) boolean a;
@Option(names = "-B") boolean b;
@Option(names = "-A1") boolean a1;
@Option(names = "-A2") boolean a2;
@Option(names = "-A3") boolean a3;
boolean isInvalid() {
return b && (a1 || a2 || a3);
}
}
@ArgGroup(exclusive = false)
MyGroup myGroup;
@Spec CommandSpec spec;
public void run() {
if (myGroup != null && myGroup.isInvalid()) {
String msg = "Option -B is mutually exclusive with -A1, -A2 and -A3";
throw new ParameterException(spec.commandLine(), msg);
}
System.out.printf("OK: %s%n", spec.commandLine().getParseResult().originalArgs());
}
public static void main(String[] args) {
//new CommandLine(new App()).usage(System.out);
//test: these are all valid
new CommandLine(new App()).execute();
new CommandLine(new App()).execute("-A -B".split(" "));
// requires validation in the application to disallow
new CommandLine(new App()).execute("-A -B -A1".split(" "));
// picocli validates this, gives: "Error: Missing required argument(s): -A"
new CommandLine(new App()).execute("-B -A1".split(" "));
}
}