Subparsers.add_parser TypeError:__ init __()得到了一个意外的关键字参数'prog'

时间:2017-12-15 13:55:43

标签: python argparse python-decorators

我需要实现一个ArgParse构建器来生成解析器以及子分析器和参数。我创建了一个装饰器来将subparsers声明为action。这是包含装饰器的类:

class Controller(object):
    endpoints = None
    parser = None
    subparsers = None

    def __init__(self, endpoint=None):
        from src.app.commons.ArgParser import ArgParser
        if not self.parser:
            self.parser = ArgParser()
        # Get all 
        self.endpoints = utilities.conf_wrapper()
        self.mod = endpoint
        if not self.subparsers:
            self.subparsers = self.parser.add_subparsers(help=gettext('%s_SUBPARSER_HELP' % str(self.mod).upper()))
        self.parser.add_argument_group(self.mod, gettext('%s_GROUP_DESC' % str(self.mod).upper()))


    def endpoint(self, endpoint, **kwargs):
        """ Create an endpoint to define a method on behalf of subparser"""
        ref = self
        def decorator(f):
            """
            Create subparser for controller object
            @param f as a function to call
            @return decorator method
            """
            # TODO Set given function f as a subparser destination
            new_sub_parser = ref.subparsers.add_parser(endpoint, help=gettext('%s_%s_HELP' % (ref.mod, endpoint)))
            [new_sub_parser.add_argument("--%s" % arg, action='store') for arg in ref.endpoints[ref.mod][endpoint]["params"]]
            setattr(ConsoleInterface, 'do_%s' % endpoint, f)

        return decorator

以下是我在课堂上称呼它的方式:

from src.app.controller import Controller

network = Controller("network")


@network.endpoint('create')
def create(*args):
    try:
        print "I am here"
    except Exception as err:
        print err

我在这里期待的是创建一些解析器作为这样的命令:

$~ network create [arguments]

根据ArgParse docs我做的一切都是正确的,但我得到了例外:

  File "/projectpath/src/app/controller/__init__.py", line 48, in decorator
    new_sub_parser = ref.subparsers.add_parser(endpoint, help=gettext('%s_%s_HELP' % (ref.mod, endpoint)))
  File "/usr/lib/python2.7/argparse.py", line 1066, in add_parser
    parser = self._parser_class(**kwargs)
  TypeError: __init__() got an unexpected keyword argument 'prog'

当我查看ArgParse._SubParsersAction.__init()__时,它涵盖了kwargs内部的'prog'。

2 个答案:

答案 0 :(得分:0)

问题可能出在ArgParser()级。我不知道那是什么,虽然它可能是一个自定义的argparse.ArgumentParser(作为子类)。

add_parser确保prog中有kwargs,然后传递给parser = self._parser_class(**kwargs)_parser_class是主解析器的类,除非在add_subparsers行中指定了替代。

所以我猜测ArgParser不接受prog关键字,即使ArgumentParser也是如此。但我不认为我们可以在不了解其来源的情况下帮助您。

答案 1 :(得分:0)

我遇到了同样的问题!

我创建了我的类ArgParser,该类扩展了argparse.ArgumentParser并引起了self.add_subparsers()的调用,这导致了这个问题。

class ArgParser(argparse.ArgumentParser):
    """A class to group all the arg parse stuff.
    You dont need to pay attention here unless you want to edit CLI args spec"""

    def __init__(self):
        p_args = dict(formatter_class=argparse.ArgumentDefaultsHelpFormatter)
        super().__init__(**p_args)
        self.add_argument('-v', '--version', action='version', version=f'%(prog)s {__version__}')
        sub_ps = self.add_subparsers(dest='cmd')
        sub_ps.required = True
        # ...  and lot other stuff

所以,这会导致TypeError: __init__() got an unexpected keyword argument 'prog'

如何解决?

明确设置parser_class=argparse.ArgumentParser

sub_ps = self.add_subparsers(dest='cmd', parser_class=argparse.ArgumentParser)