将argparse alias解析回原始命令

2024-06-06 05:17:07 发布

您现在位置:Python中文网/ 问答频道 /正文

我用的是subparser/subcommand that has an alias。你知道吗

我正在使用子parser的dest选项来存储子命令的名称,以便稍后获得它。你知道吗

当前,如果子命令的名称是reallyLongName,别名是r(比方说),那么dest选项会存储reallyLongNamer,我输入的内容都会被存储。这很烦人,因为我现在必须检查命令名或它的任何别名,以便识别命令。你知道吗

有没有办法让argparse将子命令的名称存储在dest字段中的某种单一规范文本字符串中?你知道吗

例如,给定以下代码:

import argparse

parser = argparse.ArgumentParser()

subparsers = parser.add_subparsers(dest='command', help='sub-command help')

parser_ag = subparsers.add_parser(  'mySubcommand',
                                    aliases=['m'],
                                    help='Subcommand help')

print(parser.parse_args('mySubcommand'.split()))

print(parser.parse_args('m'.split()))

产生以下输出:

Namespace(command='mySubcommand')
Namespace(command='m')

期望的结果:command对这两个都有一个单一的规范值,例如:

Namespace(command='mySubcommand')
Namespace(command='mySubcommand')

Tags: 命令规范名称addparser选项argparsehelp
2条回答

这是出人意料的难以挖掘。添加子parser时,它将存储在parents._actions属性中。从那里它只是通过挖掘属性来得到你需要的。下面,我创建字典,通过dest名称引用子parser参数,然后添加一个函数,让我们将输入的参数重新映射到主参数名称。你知道吗

from collections import defaultdict

def get_subparser_aliases(parser, dest):
    out = defaultdict(list)
    prog_str = parser.prog
    dest_dict = {a.dest: a for a in parser._actions}
    try:
        choices = dest_dict.get(dest).choices
    except AttributeError:
        raise AttributeError(f'The parser "{parser}" has no subparser with a `dest` of "{dest}"')

    for k, v in choices.items():
        clean_v = v.prog.replace(prog_str, '', 1).strip()
        out[clean_v].append(k)
    return dict(out)

def remap_args(args, mapping, dest):
    setattr(args, dest, mapping.get(getattr(args, dest)))
    return args

使用您的示例,我们可以使用以下方法重新映射解析参数:

import argparse

parser = argparse.ArgumentParser()
subparsers = parser.add_subparsers(dest='command', help='sub-command help')
parser_ag = subparsers.add_parser('mySubcommand',
                                  aliases=['m'],
                                  help='Subcommand help')

args = parser.parse_args('m'.split())
mapping = get_subparser_aliases(parser, 'command')
remap_args(args, mapping, 'command')
print(args)
# prints:
Namespace(command='mySubcommand')

这是一个例子,它在工作中与多个子帕瑟水平。。我们有一个带有可选参数和子参数的解析器。subparser有3个可能的参数,最后一个参数调用另一个subparser(sub subparser),其中有2个可能的参数。你知道吗

您可以检查顶级解析器或第一级子解析器来查看别名映射。你知道吗

import argparse

parser = argparse.ArgumentParser()
parser.add_argument(' someoption', '-s', action='store_true')

subparser1 = parser.add_subparsers(help='sub-command help', dest='sub1')
parser_r = subparser1.add_parser('reallyLongName', aliases=['r'])
parser_r.add_argument('foo', type=int, help='foo help')
parser_s = subparser1.add_parser('otherReallyLong', aliases=['L'])
parser_s.add_argument('bar', choices='abc', help='bar help')
parser_z = subparser1.add_parser('otherOptions', aliases=['oo'])

subparser2 = parser_z.add_subparsers(help='sub-sub-command help', dest='sub2')
parser_x = subparser2.add_parser('xxx', aliases=['x'])
parser_x.add_argument('fizz', type=float, help='fizz help')
parser_y = subparser2.add_parser('yyy', aliases=['y'])
parser_y.add_argument('blip', help='blip help')

get_subparser_aliases(parser, 'sub1')
# returns:
{'reallyLongName': ['reallyLongName', 'r'],
 'otherReallyLong': ['otherReallyLong', 'L'],
 'otherOptions': ['otherOptions', 'oo']}

get_subparser_aliases(parser_z, 'sub2')
# returns:
{'xxx': ['xxx', 'x'], 'yyy': ['yyy', 'y']}

通过使用上面的函数,我们可以将收集到的参数重新映射到它们较长的名称。你知道吗

args = parser.parse_args('-s oo x 1.23'.split())
print(args)
# prints:
Namespace(fizz=1.23, someoption=True, sub1='oo', sub2='x')

for p, dest in zip((parser, parser_z), ('sub1', 'sub2')):
    mapping = get_subparser_aliases(p, dest)
    remap_args(args, mapping, dest)

print(args)
# prints:
Namespace(fizz=1.23, someoption=True, sub1='otherOptions', sub2='xxx')

有一个Python错误/问题要求这样做-保存“base”名称,而不是别名。如果不更改argparse.py代码,就无法更改它。我认为这个变化将局限于处理子parser的Action子类。https://bugs.python.org/issue36664

但我指出有一种更简单的方法来处理这个问题。只需使用set_defaults,如https://docs.python.org/3/library/argparse.html#sub-commands部分末尾所述。在那里

parser_foo.set_defaults(func=foo)

用于设置特定于子Parser的函数,但也可以用于设置“base”名称。你知道吗

parser_foo.set_defaults(name='theIncrediblyLongAlias')

相关问题 更多 >