В Python, используя argparse, допускают только положительные целые числа

Название в значительной степени суммирует то, что я хотел бы иметь.

Вот что у меня есть, и пока программа не взорвется на неположительное целое число, я хочу, чтобы пользователь был проинформирован о том, что неположительное целое является в основном абсурдом.

import argparse
parser = argparse.ArgumentParser()
parser.add_argument("-g", "--games", type=int, default=162,
                    help="The number of games to simulate")
args = parser.parse_args()

И вывод:

python simulate_many.py -g 20
Setting up...
Playing games...
....................

Выход с отрицательным значением:

python simulate_many.py -g -2
Setting up...
Playing games...

Теперь, очевидно, я мог бы просто добавить if, если определить if args.games отрицательно, но мне было любопытно, есть ли способ уловить его на уровне argparse, чтобы воспользоваться автоматической печатью использования.

В идеале, он напечатает что-то похожее на это:

python simulate_many.py -g a
usage: simulate_many.py [-h] [-g GAMES] [-d] [-l LEAGUE]
simulate_many.py: error: argument -g/--games: invalid int value: 'a'

Так же:

python simulate_many.py -g -2
usage: simulate_many.py [-h] [-g GAMES] [-d] [-l LEAGUE]
simulate_many.py: error: argument -g/--games: invalid positive int value: '-2'

Пока я делаю это, и я доволен, что я счастлив:

if args.games <= 0:
    parser.print_help()
    print "-g/--games: must be positive."
    sys.exit(1)

Ответ 1

Это должно быть возможно с использованием type. Вам все еще нужно определить фактический метод, который решит это за вас:

def check_positive(value):
    ivalue = int(value)
    if ivalue <= 0:
        raise argparse.ArgumentTypeError("%s is an invalid positive int value" % value)
    return ivalue

parser = argparse.ArgumentParser(...)
parser.add_argument('foo', type=check_positive)

Это просто адаптированный пример из функции perfect_square в документах на argparse.

Ответ 2

type будет рекомендуемым вариантом для обработки условий/проверок, как в ответе Yuushi.

В вашем конкретном случае вы также можете использовать параметр choices, если ваш верхний предел также известен:

parser.add_argument('foo', type=int, choices=xrange(5, 10))

Ответ 3

Быстрый и грязный способ, если у вас есть предсказуемый максимум, а также минимальный для вашего аргумента, используется choices с диапазоном

parser.add_argument('foo', type=int, choices=xrange(0, 1000))

Ответ 4

Более простая альтернатива, особенно если используется подкласс argparse.ArgumentParser, состоит в том, чтобы инициировать проверку изнутри метода parse_args.

Внутри такого подкласса:

def parse_args(self, args=None, namespace=None):
    """Parse and validate args."""
    namespace = super().parse_args(args, namespace)
    if namespace.games <= 0:
         raise self.error('The number of games must be a positive integer.')
    return namespace

Эта техника может быть не такой крутой, как пользовательская функция вызова, но она выполняет свою работу.


О ArgumentParser.error(message):

Этот метод печатает сообщение об использовании, включая сообщение о стандартной ошибке, и завершает программу с кодом состояния 2.


Кредит: ответ джонатана