argparse: ArgumentDefaultsHelpFormatter with MetavarTypeHelpFormatter

1.2k Views Asked by At

Is it possible to combine the functionality of argparse.ArgumentDefaultsHelpFormatter with argparse.MetavarTypeHelpFormatter?

There are examples on how to use any one of these in the docs, but it's not clear how one would create help text that combines the best of both?

For example, I would like something like the following:

parser = argparse.ArgumentParser(prog='PROG')
parser.add_argument('--foo', type=int, default=42, help='FOO!')
parser.add_argument('bar', nargs='*', default=[1, 2, 3], help='BAR!')

to produce something like the following:

usage: PROG [-h] [--foo int] [float [float ...]]

positional arguments:
 float         BAR! (default: [1, 2, 3])

optional arguments:
 -h, --help  show this help message and exit
 --foo int   FOO! (default: 42)
1

There are 1 best solutions below

3
On BEST ANSWER

In theory you can define a new formatter class that is a subclass of both of the other classes. (I could find a bug/issue where that is suggested by the developer.) Whether that will work with this pair is needs to be tested or checked in the code. Basically we need to see whether the methods that these 2 classes change are compatible.

import argparse
class MyFormatter(argparse.ArgumentDefaultsHelpFormatter, argparse.MetavarTypeHelpFormatter):
    pass
parser = argparse.ArgumentParser(prog='PROG', formatter_class=MyFormatter)
parser.add_argument('--foo', type=float, default=42, help='FOO!')
parser.add_argument('bar', nargs='*', default=[1, 2, 3], help='BAR!', type=int)
parser.print_help()

produces

2035:~/mypy$ python3.5 stack41195224.py 
usage: PROG [-h] [--foo float] [int [int ...]]

positional arguments:
  int          BAR! (default: [1, 2, 3])

optional arguments:
  -h, --help   show this help message and exit
  --foo float  FOO! (default: 42)

Note that I had to specify a type for the positional as well. Without that I got an error:

AttributeError: 'NoneType' object has no attribute '__name__'

With this requirement, I suspect the MetavarTypeHelpFormatter does not get much use. The default 'string' type (None) will produce an error. This is a convenience formatter class, and likely to create more problems than it solves.

The defaults help formatter just adds a ' (default: %(default)s)' string to the help string. You can do that yourself. So it is just a convenience class, not something that anyone requires.

The alternative to subclassing these 2 classes, is to subclass their parent, and include all the methods that the 2 subclasses change.


adding:

parser.add_argument('foobar', type=str)

displays

1039:~/mypy$ python3 stack41195224_1.py 
usage: PROG [-h] [--foo float] [int [int ...]] str

positional arguments:
  int          BAR! (default: [1, 2, 3])
  str

optional arguments:
  -h, --help   show this help message and exit
  --foo float  FOO! (default: 42)

Using just

parser.add_argument('foobar')

produces

AttributeError: 'NoneType' object has no attribute '__name__'