【问题标题】:How to print the help (usage) section for the main parser of Argparse and not the usage of a subparser?如何打印 Argparse 的主解析器的帮助(使用)部分而不是子解析器的使用?
【发布时间】:2018-03-27 01:29:28
【问题描述】:

我用 Argparse 编写命令行程序已经有一段时间了,我正在尝试以这样一种方式编写它,即当用户向命令行提供以下内容时:

$python my_script.py -h

将打印出一个帮助部分(用法),打印出主解析器的帮助部分,以及子解析器的简要概述。

但是现在,每当我在终端中输入上一行时,我都没有收到任何使用信息,而是收到大量回溯和以下错误:

TypeError: expected string or buffer

在使用基于 argparse 的命令行程序之前,我从未遇到过此错误。此外,如果我提供其中一个子解析器的名称,

$python my_script.py subparserA -h

我得到了子解析器使用的打印输出。其他子解析器也是如此。

那么为什么我无法获得主解析器的用法呢?这以前对我有用,所以我不知道为什么它现在不起作用。我真的希望用户能够查看可用的不同子解析器的概述。

我的基本代码目前是这样设置的:

import argparse
import sys

if __name__ == "__main__":
    Parser = argparse.ArgumentParser(prog= "My_program")

    Parser.description= "This program does A and B things."
    subparsers= Parser.add_subparsers(help= "SubparserA does A things and SubparserB does B things", dest='mode')

    subparserA= subparsers.add_parser("subparserA", help= "Additional explanation of what A things entail")

    subparserA.add_arguments("-foo", required=True, help= "foo is needed for SubparserA to work")

    subparserB= subparsers.add_parser("subparserB", help="Additional explanation of what B things entail")

    subparserB.add_argument("-bar", required=True, help= "bar is needed for SubparserB to work")

    args= Parser.parse_args()

    if args.mode == "subparserA":
        ###do things pertinent to subparserA
    elif args.mode== "subparserB":
        ###do things pertinent to subparserB
    else:
        argparse.print_help()
        argparse.ArgumentError("too few arguments")

更新

这是错误的完整回溯:

Traceback (most recent call last):
  File "my_program.py", line 164, in <module>
    args= Parser.parse_args()
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 1701, in parse_args
    args, argv = self.parse_known_args(args, namespace)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 1733, in parse_known_args
    namespace, args = self._parse_known_args(args, namespace)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 1939, in _parse_known_args
    start_index = consume_optional(start_index)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 1879, in consume_optional
    take_action(action, args, option_string)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 1807, in take_action
    action(self, namespace, argument_values, option_string)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 996, in __call__
    parser.print_help()
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 2340, in print_help
    self._print_message(self.format_help(), file)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 2314, in format_help
    return formatter.format_help()
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 281, in format_help
    help = self._root_section.format_help()
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 211, in format_help
    func(*args)
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 485, in _format_text
    return self._fill_text(text, text_width, indent) + '\n\n'
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/argparse.py", line 621, in _fill_text
    text = self._whitespace_matcher.sub(' ', text).strip()
TypeError: expected string or buffer

【问题讨论】:

  • 格式化帮助时出现问题。确保所有文本行,例如描述、帮助是正确的字符串。另外argparse.py 源文件已损坏。
  • @hpaulj 是的,你一针见血!在我试图让我的代码遵守 PEP8 的过程中,我通过输入一个逗号将一个长字符串与 Parser.description 分开,然后将其余字符串放在下一行。显然它以某种方式通过我的代码。非常感谢!

标签: python-2.7 argparse


【解决方案1】:

你应该使用

Parser.print_help()
Parser.error('too few arguments')

即使用现有Parser对象的方法。


当我运行你的脚本时

1019:~/mypy$ python stack46754855.py 
Traceback (most recent call last):
  File "stack46754855.py", line 10, in <module>
    subparserA= subparsers.add_parser("subparserA", help= "Additional explanation of what A things entail", dest= 'mode')
  File "/usr/lib/python2.7/argparse.py", line 1066, in add_parser
    parser = self._parser_class(**kwargs)
TypeError: __init__() got an unexpected keyword argument 'dest'

dest 不是 add_parser 方法的有效参数。这是add_subparsers 的有效且有用的参数。

subparsers= Parser.add_subparsers(dest='mode')

它还反对add_arguments 方法。

修正后我得到的:

1022:~/mypy$ python stack46754855.py 
usage: My_program [-h] {subparserA,subparserB} ...
My_program: error: too few arguments

在 Py2 中,subparsers 是必需的参数。它在 Py3 中是可选的(一个 bug),允许脚本运行到无效的 argparse.print_help 调用:

1022:~/mypy$ python3 stack46754855.py 
Traceback (most recent call last):
  File "stack46754855.py", line 27, in <module>
    argparse.print_help()
AttributeError: module 'argparse' has no attribute 'print_help'

根据我上面建议的更改:

1025:~/mypy$ python3 stack46754855.py 
usage: My_program [-h] {subparserA,subparserB} ...

This program does A and B things.

positional arguments:
  {subparserA,subparserB}
                        SubparserA does A things and SubparserB does B things
    subparserA          Additional explanation of what A things entail
    subparserB          Additional explanation of what B things entail

optional arguments:
  -h, --help            show this help message and exit
usage: My_program [-h] {subparserA,subparserB} ...
My_program: error: too few arguments

第二个usage 来自Parser.error 调用。


我无法复制你的

大量回溯和以下错误: TypeError: 预期的字符串或缓冲区

我需要查看该回溯(或其中的一部分)以了解究竟是什么引发了错误。这不是一个正常的argparse 错误;当然它不是argparse 陷阱和重新路由。


How to Set a Default Subparser using Argparse Module with Python 2.7 上有关必需/非必需子解析器行为的更多信息

【讨论】:

  • 我也遇到了试图找出可能导致这种行为的问题。在任何情况下,您是否会建议我制作另一个子解析器,当没有指定子解析器时,它的唯一工作就是打印出其他子解析器的用法/帮助部分?
  • @BobMcBobson,如果您想要 Py2 中的可选子解析器行为,那么我在另一个问题中建议的 2 阶段解析器可能是最好的。但这会让您的用户的生活更轻松吗?我正在探索conda。它需要 12+ 个命令,但没有任何迹象表明它们是 optional。这是大型程序接口中的正常行为。
  • @BobMcBobson。想想usage。你如何告诉你的用户 subparser 命令是可选的?当前用法类似于prog [-h] {cmd1,cmd2} ...。应该是prog [-h] [{cmd1,cmd2}] ...,添加[] 表示提供cmd 是可选的吗?这是否增加了清晰度?
  • 抱歉回复晚了,最近工作很乱,忙于其他项目。无论如何,我不想告诉我的用户 subparser 命令是可选的,而是希望用户两个 subparser 之一是程序运行所必需的。另外,我不明白为什么在命令行中添加[...] 方括号表明某个cmd 是可选的。
  • argparse(可能还有其他解析器)使用用括号标记可选参数的约定。我刚刚意识到,虽然 Py3 argparse 已经将 subparsers 保留了一段时间,但它并没有将其标记为这样。这算不算缺陷?
【解决方案2】:

使用 + 代替 , 用于 parser.add_argument 中的多行帮助字符串。如果您使用 ',' 将参数帮助拆分为多行,那么您将看到此问题

parser.add_argument("xml",help=("long help here",
                                " long help second line"))

这将导致上述异常

改为

parser.add_argument("xml",help=("long help here" +
                                " long help second line"))

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2015-12-11
    • 2013-12-04
    • 2012-04-25
    • 2015-11-04
    • 2013-02-01
    • 2023-03-23
    • 2013-06-09
    相关资源
    最近更新 更多