3

这是一个两部分的问题,请参见下文:

  • 我需要创建某种控制台供测试人员手动运行一些命令!模块是cmd一个好方法吗?

下面是我到目前为止使用cmd模块的代码,我只是想学习,到目前为止我有两个问题:

  • 为什么自动完成功能不起作用?如果我加倍<TAB>,我什么也得不到,光标只会向前移动。不是默认提供的自动完成功能吗?

  • 我是否必须为每种方法处理错误数量的参数?如果使用错误数量的参数调用方法,或者何时应该使用参数调用它们而没有调用,我希望方法-“帮助”文本自动显示。

.

class InteractiveConsole(cmd.Cmd):
    """ Interactive command line """

    def __init__(self):
        cmd.Cmd.__init__(self)
        self.prompt = "=>> "
        self.intro = "Welcome to IRT console!"

    def do_hist(self, args):
        """Print a list of commands that have been entered"""
        print self._hist

    def do_exit(self, args):
        """Exits from the console"""
        return -1

    def do_help(self, args):
        """Get help on commands
           'help' or '?' with no arguments prints a list of commands for which help is available
           'help <command>' or '? <command>' gives help on <command>
        """
        # # The only reason to define this method is for the help text in the doc string
        cmd.Cmd.do_help(self, args)

    # # Override methods in Cmd object ##
    def preloop(self):
        """Initialization before prompting user for commands.
           Despite the claims in the Cmd documentaion, Cmd.preloop() is not a stub.
        """
        cmd.Cmd.preloop(self)  # # sets up command completion
        self._hist = []  # # No history yet
        self._locals = {}  # # Initialize execution namespace for user
        self._globals = {}

    def postloop(self):
        """Take care of any unfinished business.
           Despite the claims in the Cmd documentaion, Cmd.postloop() is not a stub.
        """
        cmd.Cmd.postloop(self)  # # Clean up command completion
        print "Exiting..."

    def precmd(self, line):
        """ This method is called after the line has been input but before
            it has been interpreted. If you want to modify the input line
            before execution (for example, variable substitution) do it here.
        """
        if line != '':
            self._hist += [ line.strip() ]
        return line

    def postcmd(self, stop, line):
        """If you want to stop the console, return something that evaluates to true.
           If you want to do some post command processing, do it here.
        """
        return stop

    def default(self, line):
        """Called on an input line when the command prefix is not recognized.
           In that case we execute the line as Python code.
        """
        try:
            exec(line) in self._locals, self._globals
        except Exception, e:
            print e.__class__, ":", e

    def emptyline(self):
        """Do nothing on empty input line"""
        pass





    def do_install(self, pathToBuild):
        """install [pathToBuild]
        install using the specified file"""
        if pathToBuild:
            print "installing %s" % pathToBuild
        else:
            print "<ERROR> You must specify the absolute path to a file which should be used!"


    def do_configure(self, pathToConfiguration):
        """configure [pathToConfiguration]
        configure using the specified file"""
        if pathToConfiguration:
            print "configuring %s" % pathToConfiguration
        else:
            print "<ERROR> You must specify the absolute path to a file which should be used!"
4

2 回答 2

4

cmd文档中:

可选参数completekeyreadline完成键的名称;它默认为Tab. 如果completekey不可用Nonereadline可用,则会自动完成命令完成。

您需要有readline可用的选项卡完成才能工作。

命令方法只接受一个参数,您需要在命令方法本身中解析参数。当然,如果需要,您可以调用self.do_help()self.help_<cmd>()方法。

于 2013-07-19T12:10:29.720 回答
1

对于第一部分,是的,我发现 cmd 模块易于使用且功能强大,足以实现类似于 Python 内置命令提示符的 CLI。

对于第二部分的第一个问题,您需要告诉模块如何完成命令行,方法是实现一个类似于 complete_install(self, word, line, begindex, endindex) 的方法,该方法获取当前的单词、行、开始和结束索引在该行中,并返回表示有效完成的字符串列表或元组。您应该通常根据当前单词(第一个参数)计算和过滤列表。

例如,我使用命令 'll' 来设置日志记录级别,实现如下:

def complete_ll(self, a, ln, bi, ei):
    return tuple(
        k for k in logging._nameToLevel.keys()
        if k.casefold().find(a.casefold()) >= 0)

def do_ll(self, a):
    "Set or get debug level: DL [10 .. 50 | levelName]"
    def ll():
        n = log.getEffectiveLevel()
        return f"{logging.getLevelName(n)} ({n})"
    print(ll())
    if a:
        try:
            log.setLevel(eval(a.upper(), logging._nameToLevel))
            print("Logging level changed to", ll())
        except Exception as e:
            log.exception(f"{e}, value {a}", exc_info=1)

对于第二个问题,是的,您应该检查“do_...”方法中参数的数量、类型和有效性,这在您的示例中已经在某种程度上完成了。当然,你也可以在那个时候调用 'help_..." 方法,如果它真的有帮助的话。

于 2020-04-18T14:24:59.307 回答