Подтвердить что ты не робот

Совместная эксклюзивная группа Python argparse

Что мне нужно:

pro [-a xxx | [-b yyy -c zzz]]

Я пробовал это, но не работал. Может ли кто-нибудь помочь мне?

group= parser.add_argument_group('Model 2')
group_ex = group.add_mutually_exclusive_group()
group_ex.add_argument("-a", type=str, action = "store", default = "", help="test")
group_ex_2 = group_ex.add_argument_group("option 2")
group_ex_2.add_argument("-b", type=str, action = "store", default = "", help="test")
group_ex_2.add_argument("-c", type=str, action = "store", default = "", help="test")

Спасибо!

4b9b3361

Ответ 1

add_mutually_exclusive_group не делает целую группу взаимоисключающей. Это делает опции внутри группы взаимоисключающими.

То, что вы ищете, это подкоманды. Вместо prog [-a xxxx | [-b yyy -c zzz]], у вас будет:

prog 
  command 1 
    -a: ...
  command 2
    -b: ...
    -c: ...

Чтобы вызвать с помощью первого набора аргументов:

prog command_1 -a xxxx

Чтобы вызвать второй набор аргументов:

prog command_2 -b yyyy -c zzzz

Вы также можете установить аргументы вспомогательной команды как позиционные.

prog command_1 xxxx

Вид вроде git или svn:

git commit -am
git merge develop

Рабочий пример

# create the top-level parser
parser = argparse.ArgumentParser(prog='PROG')
parser.add_argument('--foo', action='store_true', help='help for foo arg.')
subparsers = parser.add_subparsers(help='help for subcommand')

# create the parser for the "command_1" command
parser_a = subparsers.add_parser('command_1', help='command_1 help')
parser_a.add_argument('a', type=str, help='help for bar, positional')

# create the parser for the "command_2" command
parser_b = subparsers.add_parser('command_2', help='help for command_2')
parser_b.add_argument('-b', type=str, help='help for b')
parser_b.add_argument('-c', type=str, action='store', default='', help='test')

Проверить его

>>> parser.print_help()
usage: PROG [-h] [--foo] {command_1,command_2} ...

positional arguments:
  {command_1,command_2}
                        help for subcommand
    command_1           command_1 help
    command_2           help for command_2

optional arguments:
  -h, --help            show this help message and exit
  --foo                 help for foo arg.
>>>

>>> parser.parse_args(['command_1', 'working'])
Namespace(a='working', foo=False)
>>> parser.parse_args(['command_1', 'wellness', '-b x'])
usage: PROG [-h] [--foo] {command_1,command_2} ...
PROG: error: unrecognized arguments: -b x

Удачи.

Ответ 2

В то время как ответ Джонатана отлично подходит для сложных вариантов, есть очень простое решение, которое будет работать для простых случаев, например. 1 исключает 2 других варианта, таких как

command [- a xxx | [ -b yyy | -c zzz ]] 

или даже как в исходном вопросе:

pro [-a xxx | [-b yyy -c zzz]]

Вот как я это сделаю:

parser = argparse.ArgumentParser()

# group 1 
parser.add_argument("-q", "--query", help="query", required=False)
parser.add_argument("-f", "--fields", help="field names", required=False)

# group 2 
parser.add_argument("-a", "--aggregation", help="aggregation",
                    required=False)

Я использую здесь опции, предоставленные оболочке командной строки для запроса mongodb. Экземпляр collection может либо вызвать метод aggregate, либо метод find с дополнительными аргументами query и fields, поэтому вы видите, почему первые два аргумента совместимы, а последний - нет.

Итак, теперь я запускаю parser.parse_args() и проверяю его содержимое:

args = parser().parse_args()

print args.aggregation
if args.aggregation and (args.query or args.fields):
    print "-a and -q|-f are mutually exclusive ..."
    sys.exit(2)

Конечно, этот маленький взлом работает только для простых случаев, и для кошмаров можно было бы проверить все возможные варианты, если у вас есть много взаимоисключающих опций и групп. В этом случае вы должны разбить свои параметры в командных группах, как предположил Джонатан.

Ответ 3

Существует патч python (в разработке), который позволит вам сделать это.
http://bugs.python.org/issue10984

Идея состоит в том, чтобы позволить перекрывать взаимоисключающие группы. Поэтому usage может выглядеть так:

pro [-a xxx | -b yyy] [-a xxx | -c zzz]

Изменение кода argparse, чтобы вы могли создать две группы, такие как это была легкая часть. Изменение кода форматирования usage требует записи пользовательского HelpFormatter.

В argparse, группы действий не влияют на синтаксический анализ. Это всего лишь инструмент форматирования help. В help взаимоисключающие группы влияют только на строку usage. При анализе parser использует взаимоисключающие группы для создания словаря потенциальных конфликтов (a не может встречаться с b или c, b не может произойти с a и т.д.)., а затем вызывает ошибку, если возникает конфликт.

Без этого патча argparse я считаю, что ваш лучший выбор - проверить пространство имен, созданное parse_args самостоятельно (например, если оба a и b имеют значения небезопасности) и поднимите собственную ошибку. Вы даже можете использовать собственный механизм анализа парсера.

parser.error('custom error message')