-
-
Notifications
You must be signed in to change notification settings - Fork 31.7k
argparse allows nargs>1 for positional arguments but doesn't allow metavar to be a tuple #58282
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Comments
Here's four commands to demonstrate my problem: My sample code: This output could be friendlier:
$ python3 prog.py --help
Traceback (most recent call last):
File "prog.py", line 4, in <module>
args = parse.parse_args()
File "/home/wena/cpython/Lib/argparse.py", line 1722, in parse_args
args, argv = self.parse_known_args(args, namespace)
File "/home/wena/cpython/Lib/argparse.py", line 1754, in parse_known_args
namespace, args = self._parse_known_args(args, namespace)
File "/home/wena/cpython/Lib/argparse.py", line 1960, in _parse_known_args
start_index = consume_optional(start_index)
File "/home/wena/cpython/Lib/argparse.py", line 1900, in consume_optional
take_action(action, args, option_string)
File "/home/wena/cpython/Lib/argparse.py", line 1828, in take_action
action(self, namespace, argument_values, option_string)
File "/home/wena/cpython/Lib/argparse.py", line 1015, in __call__
parser.print_help()
File "/home/wena/cpython/Lib/argparse.py", line 2347, in print_help
self._print_message(self.format_help(), file)
File "/home/wena/cpython/Lib/argparse.py", line 2314, in format_help
formatter.add_arguments(action_group._group_actions)
File "/home/wena/cpython/Lib/argparse.py", line 270, in add_arguments
self.add_argument(action)
File "/home/wena/cpython/Lib/argparse.py", line 255, in add_argument
invocations = [get_invocation(action)]
File "/home/wena/cpython/Lib/argparse.py", line 533, in _format_action_invocation
metavar, = self._metavar_formatter(action, default)(1)
ValueError: too many values to unpack (expected 1)
On Python 3.2 (from Debian), I get "error: too few arguments". But with latest mercurial I get:
$ python3 prog.py
Traceback (most recent call last):
File "prog.py", line 4, in <module>
args = parse.parse_args()
File "/home/wena/cpython/Lib/argparse.py", line 1722, in parse_args
args, argv = self.parse_known_args(args, namespace)
File "/home/wena/cpython/Lib/argparse.py", line 1754, in parse_known_args
namespace, args = self._parse_known_args(args, namespace)
File "/home/wena/cpython/Lib/argparse.py", line 1973, in _parse_known_args
', '.join(required_actions))
TypeError: sequence item 0: expected str instance, tuple found But this looks okay: |
Looks like the problem is that "_format_action_invocation" is not being as careful with the different possibilities for metavar as "_format_args" is. Patches welcome! |
This patch fixes the problem by joining the metavar terms with '|'. So the help for the test case (adapted from an existing tuple test) looks like:
Alternatives include:
The last alternative would use: #metavar = '|'.join(metavar)
if len(metavar)>1:
metavar = default
else:
metavar = metavar[0] |
oops - to fix the error message that OP complained about, I need to patch '_get_action_name' as well: def _get_action_name(argument):
...
elif argument.metavar not in (None, SUPPRESS):
metavar = argument.metavar
if isinstance(metavar, tuple):
metavar = '|'.join(metavar)
return metavar |
This patch fixes both help and error formatting. A module level '_format_metavars' does the formatting for both. I have tried several alternatives, including using the 'usage' style. There is similarity between this fix and that for bpo-16468 (custom choices), though I don't think they conflict. In both cases, code needed to format the usage or help is also needed to help format error messages. bpo-9849 (better nargs warning) is another case where error checking in the parser depends on the formatter. In the long run we may want to refactor these issues. |
An alternative fix is to disallow tuple metavars for positionals. A tuple metavar may look nice in the usage. But a well selected dest seems better in the help line, and error messages. Using dest in all 3 roles, as we do now, is the simplest compromise. |
Can confirm this bug still exists on master branch, python3.7, python3.6, and very likely other versions since it's reported. It seems only |
-0 on going forward with this. AFAICT, no one has requested tuple support (the OP just wanted better error handling). And more fined grained control can already be had just by specifying separate positional arguments. Paul suggested that it may be a better plan to just disallow tuple metavars and to give a better error message as the OP suggested. That makes sense to me. |
I am not sure disallowing tuple metavars would break backup compatibility or not~ |
Given that tuple metavars currently doesn't work, there is no backward compatibility issue. What is being proposed is to give more friendly error messages. |
Got it, thanks.
+1 |
Tuple support is documented: 98047eb#diff-9c4a053d29149ba40370fbdddd3e34faR1059 https://docs.python.org/3/library/argparse.html#metavar
From his word here I feel like our professor meant to support tuple metavar but the implementation was not careful. He called for a patch eight years ago. Let's summon him and see if he has changed his mind? |
Reproduced on 3.11, with a different error: % ./python.exe prog.py
Traceback (most recent call last):
File "/Users/iritkatriel/src/cpython-1/prog.py", line 5, in <module>
args = parse.parse_args()
^^^^^^^^^^^^^^^^^^
File "/Users/iritkatriel/src/cpython-1/Lib/argparse.py", line 1822, in parse_args
args, argv = self.parse_known_args(args, namespace)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/Users/iritkatriel/src/cpython-1/Lib/argparse.py", line 1855, in parse_known_args
namespace, args = self._parse_known_args(args, namespace)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/Users/iritkatriel/src/cpython-1/Lib/argparse.py", line 2093, in _parse_known_args
', '.join(required_actions))
^^^^^^^^^^^^^^^^^^^^^^^^^^^
TypeError: sequence item 0: expected str instance, tuple found |
Hi All. I have stumbled upon this issue almost three years ago and wrote a question on StackOverflow. My sample program was import argparse
parser = argparse.ArgumentParser()
parser.add_argument('size', type=int, nargs=2, help='size', metavar=('w', 'h'))
args = parser.parse_args()
print(args) When the program is called as For optional arguments import argparse
parser = argparse.ArgumentParser()
parser.add_argument('--size', type=int, nargs=2, help='size', metavar=('w', 'h'))
args = parser.parse_args()
print(args) all is fine
The official documentation mentions using tuples in metavar, but only provides an example for optional arguments. I would argue that positional arguments should behave the same and would rather see this fixed than banned. |
… in argparse Previously, formatting help output or error message for positional argument with a tuple metavar raised exception. Co-authered-by: Cyker Way <cykerway@gmail.com>
… in argparse Previously, formatting help output or error message for positional argument with a tuple metavar raised exception. Co-authored-b: Cyker Way <cykerway@gmail.com>
…gparse (GH-124782) Previously, formatting help output or error message for positional argument with a tuple metavar raised exception. Co-authored-by: Cyker Way <cykerway@gmail.com>
… in argparse (pythonGH-124782) Previously, formatting help output or error message for positional argument with a tuple metavar raised exception. (cherry picked from commit 9b31a2d) Co-authored-by: Serhiy Storchaka <storchaka@gmail.com> Co-authored-by: Cyker Way <cykerway@gmail.com>
…guments in argparse (pythonGH-124782) Previously, formatting help output or error message for positional argument with a tuple metavar raised exception. (cherry picked from commit 9b31a2d) Co-authored-by: Serhiy Storchaka <storchaka@gmail.com> Co-authored-by: Cyker Way <cykerway@gmail.com>
Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.
Show more details
GitHub fields:
bugs.python.org fields:
Linked PRs
The text was updated successfully, but these errors were encountered: