5

我希望能够使用 argparse.ArgumentParser 对象中的全局选项来覆盖/增加子命令的默认值。

一个示例是显示的帮助反映了全局更新,即,对于以下玩具示例:

import argparse
import os
import sys

# Global parser and options.
parser = argparse.ArgumentParser(
    formatter_class=argparse.ArgumentDefaultsHelpFormatter)

parser.add_argument("--user", 
                    dest="user", 
                    default=os.environ.get("USER"),
                    help="Override the setting of the $USER variable.")

# Sub-command parser and options.
subparsers = parser.add_subparsers()

command = subparsers.add_parser(
    "command",
    formatter_class=argparse.ArgumentDefaultsHelpFormatter)

command.add_argument("--config",
                     dest="config",
                     default="~%s/config" % os.environ.get("USER"),
                     help="The config file.")

options = parser.parse_args()

理想情况下,当我在帮助模式下运行它时,我会得到,

> python example.py --user thing command --help
usage: example.py command [-h] [--config CONFIG]

optional arguments:
  -h, --help       show this help message and exit
  --config CONFIG  The config file. (default: ~thing/config)

即,配置文件路径是用户特定的(事物)。我意识到我可以将默认配置更改为"~%(user)s/config",然后在运行时使用 options 命名空间解决这个问题,但是我希望帮助更明确。

我收集另一种解决方案是尝试解析参数一次以获得全局选项,即

if "--help" in sys.argv:

    # Parse the command minus the help to obtain the global options. 
    args = sys.argv[1:]
    args.remove("--help")

    # Update the defaults with the global options.
    options = parser.parse_args(args) 
    command.set_defaults(config="~%s/config" % options.user)

    # Re-parse the options.
    parser.parse_args()

虽然这看起来有点 hacky。有更好的方法/解决方案吗?

4

2 回答 2

4

在定义全局选项之后,但在定义子命令之前,调用parse_known_args以找出值--user是什么。然后,在调用解析命令行上的所有选项之前,完成子解析器命令的定义,使用 的值--user定义默认值。--configparse_args

这与您的替代方案略有不同,但它将所有命令行处理保留在argparse对象内。

(我稍微缩减​​了你的代码只是为了缩短这个答案。)

import os
import argparse

# Global preparser and options.
preparser = argparse.ArgumentParser(add_help=False)
preparser.add_argument("--user", dest="user", default=os.environ.get("USER"))

# ****** NEW *******
options, _ = preparser.parse_known_args() # Ignore what we haven't defined yet
user = options.user                       # Use this to define the default of --config

parser = argparse.ArgumentParser(parents=[ preparser ], add_help=True,
                   formatter_class=argparse.ArgumentDefaultsHelpFormatter)

# Sub-command parser and options.
subparsers = parser.add_subparsers()

command = subparsers.add_parser("command")

# ****** MODIFIED *******
command.add_argument("--config", dest="config", default="~%s/config" % (user,))

options = parser.parse_args()
于 2012-08-28T21:20:31.553 回答
0

这是一个在运行时直接修改帮助行的解决方案。它也可以修改os.environ或其他一些全局的,但我会保持简单。关键是将创建的操作分配给add_argument('--config'...)变量。 help只是该变量的一个属性。您可以自由修改。

class FooAction(argparse.Action):
    def __call__(self, parser, namespace, values, option_string=None):
        config.help = 'new user (%s)'% values
        setattr(namespace, self.dest, values)

parser = argparse.ArgumentParser()
parser.add_argument('--user',action=FooAction)
sub = parser.add_subparsers()
cmd = sub.add_parser('cmd')
config = cmd.add_argument('--config',help='initial help')
config.help = 'default help' # alt way of setting 'help'
# print config  # to see other attributes of the config action
args = parser.parse_args()
print args

调用cmd -h我们得到default help

$ python stack12167228.py cmd -h
usage: stack12167228.py cmd [-h] [--config CONFIG]

optional arguments:
  -h, --help       show this help message and exit
  --config CONFIG  default help

调用时--user xxx cmd -h,帮助是自定义的

$ python stack12167228.py --user xxx cmd -h
usage: stack12167228.py cmd [-h] [--config CONFIG]

optional arguments:
  -h, --help       show this help message and exit
  --config CONFIG  new user (xxx)
于 2013-10-10T03:18:43.410 回答