Comma separated inputs instead of space separated inputs for argparse

前端 未结 5 970
灰色年华
灰色年华 2021-01-19 02:30

I\'m using argparse to receive inputs from the command line to run my script.

My current input string looks like this:

path> python &         


        
5条回答
  •  感情败类
    2021-01-19 03:18

    There are some useful answers here already, but I wanted a bit more: splitting on commas, validating values with choices, and getting useful error messages, so below I offer a solution.

    Simple Version

    As a first pass, we can just pass in an appropriate function to the type parameter:

    >>> import argparse
    >>> parser = argparse.ArgumentParser(prog='cmd')
    >>> parser.add_argument('--foo', type=lambda arg: arg.split(','))
    >>> parser.parse_args(['--foo', 'a,b,c'])
    Namespace(foo=['a', 'b', 'c'])
    

    But this doesn't work with choices because it checks if the whole list is in choices and not each value:

    >>> parser = argparse.ArgumentParser(prog='cmd')
    >>> parser.add_argument('--foo', type=lambda arg: arg.split(','), choices=('a', 'b', 'c'))
    >>> parser.parse_args(['--foo', 'a,b,c'])
    usage: cmd [-h] [--foo {a,b,c}]
    cmd: error: argument --foo: invalid choice: ['a', 'b', 'c'] (choose from 'a', 'b', 'c')
    

    Setting nargs to something like * or + checks each value, but only for space-separated arguments (e.g., --foo a b) and not the comma-separated ones. It seems like there is no supported way to check if each value is in the choices if we produce the list ourselves. Therefore we need to raise errors ourselves via the type parameter (as Shiplu Mokaddim partially implemented). Creating a custom Action class sounds promising as actions have access to the choices, but the action happens after the type function applies and values are checked, so we still couldn't use the choices parameter on add_argument() for this purpose.

    Better Version

    Here is a solution using a custom type function. We this function to take a list of valid choices, but since the function for type conversion can only take an argument string, we need to wrap it in a class (and define the special __call__() method) or a function closure. This solution uses the latter.

    >>> def csvtype(choices):
    ...     """Return a function that splits and checks comma-separated values."""
    ...     def splitarg(arg):
    ...         values = arg.split(',')
    ...         for value in values:
    ...             if value not in choices:
    ...                 raise argparse.ArgumentTypeError(
    ...                     'invalid choice: {!r} (choose from {})'
    ...                     .format(value, ', '.join(map(repr, choices))))
    ...         return values
    ...     return splitarg
    >>> parser = argparse.ArgumentParser(prog='cmd')
    >>> parser.add_argument('--foo', type=csvtype(('a', 'b', 'c')))
    >>> parser.parse_args(['--foo', 'a,b,c'])
    Namespace(foo=['a', 'b', 'c'])
    >>> parser.parse_args(['--foo', 'a,b,d'])
    usage: cmd [-h] [-f F]
    cmd: error: argument -f: invalid choice: 'd' (choose from 'a', 'b', 'c')
    

    Notice that we get an appropriate error as well. For this, be sure to use argparse.ArgumentTypeError and not argparse.ArgumentError inside the function.

    Other Options

    User wim suggested some other options not discussed above. I don't find those attractive for the following reasons:

    • Post-processing the argument after parsing means you have to do more work to get the error messages to be consistent with those from argparse. Just raising argparse.ArgumentError will lead to a stacktrace. Also, argparse catches errors raised during parsing and alters them to specify the option that was used, which you'd otherwise need to do manually.

    • Subclassing ArgumentParser is more work, and convert_arg_line_to_args() is for reading arguments from a file, not the command line.

提交回复
热议问题