10

我正在使用 click ( http://click.pocoo.org/3/ ) 创建命令行应用程序,但我不知道如何为该应用程序创建 shell。
假设我正在编写一个名为test的程序,并且我有名为subtest1subtest2的命令

我能够使其从终端工作,如:

$ test subtest1
$ test subtest2

但我在想的是一个shell,所以我可以这样做:

$ test  
>> subtest1  
>> subtest2

这可以通过点击实现吗?

4

4 回答 4

15

点击这并非不可能,但也没有内置支持。首先你必须做的是通过传递invoke_without_command=True到组装饰器(如here所述)使你的组回调可以在没有子命令的情况下调用。然后你的组回调必须实现一个 REPL。Python 在标准库中有执行此操作的cmd框架。使 click 子命令在那里可用涉及覆盖cmd.Cmd.default,如下面的代码片段所示。help只需几行就可以正确获取所有细节,例如。

import click
import cmd

class REPL(cmd.Cmd):
    def __init__(self, ctx):
        cmd.Cmd.__init__(self)
        self.ctx = ctx

    def default(self, line):
        subcommand = cli.commands.get(line)
        if subcommand:
            self.ctx.invoke(subcommand)
        else:
            return cmd.Cmd.default(self, line)

@click.group(invoke_without_command=True)
@click.pass_context
def cli(ctx):
    if ctx.invoked_subcommand is None:
        repl = REPL(ctx)
        repl.cmdloop()

@cli.command()
def a():
    """The `a` command prints an 'a'."""
    print "a"

@cli.command()
def b():
    """The `b` command prints a 'b'."""
    print "b"

if __name__ == "__main__":
    cli()
于 2015-03-11T19:22:20.753 回答
3

现在有一个名为click_repl的库可以为您完成大部分工作。我想我会分享我的努力来让它发挥作用。

一个困难是您必须将特定命令设为repl命令,但如果未提供另一个命令,我们可以重新调整@fpbhb 的方法以允许默认调用该命令。

这是一个完整的工作示例,它支持所有单击选项,具有命令历史记录,并且能够直接调用命令而无需输入 REPL:

import click
import click_repl
import os
from prompt_toolkit.history import FileHistory

@click.group(invoke_without_command=True)
@click.pass_context
def cli(ctx):
    """Pleasantries CLI"""
    if ctx.invoked_subcommand is None:
        ctx.invoke(repl)

@cli.command()
@click.option('--name', default='world')
def hello(name):
    """Say hello"""
    click.echo('Hello, {}!'.format(name))

@cli.command()
@click.option('--name', default='moon')
def goodnight(name):
    """Say goodnight"""
    click.echo('Goodnight, {}.'.format(name))

@cli.command()
def repl():
    """Start an interactive session"""
    prompt_kwargs = {
        'history': FileHistory(os.path.expanduser('~/.repl_history'))
    }
    click_repl.repl(click.get_current_context(), prompt_kwargs=prompt_kwargs)

if __name__ == '__main__':
    cli(obj={})

下面是使用 REPL 的样子:

$ python pleasantries.py
> hello
Hello, world!
> goodnight --name fpbhb
Goodnight, fpbhb.

并直接使用命令行子命令:

$ python pleasntries.py goodnight
Goodnight, moon.
于 2018-05-16T16:51:38.037 回答
2

我知道这已经很老了,但我一直在研究 fpbhb 的解决方案来支持选项。我确信这可能需要更多的工作,但这里有一个如何完成的基本示例:

import click
import cmd
import sys

from click import BaseCommand, UsageError


class REPL(cmd.Cmd):
    def __init__(self, ctx):
        cmd.Cmd.__init__(self)
        self.ctx = ctx

    def default(self, line):
        subcommand = line.split()[0]
        args = line.split()[1:]

        subcommand = cli.commands.get(subcommand)
        if subcommand:
            try:
                subcommand.parse_args(self.ctx, args)
                self.ctx.forward(subcommand)
            except UsageError as e:
                print(e.format_message())
        else:
            return cmd.Cmd.default(self, line)


@click.group(invoke_without_command=True)
@click.pass_context
def cli(ctx):
    if ctx.invoked_subcommand is None:
        repl = REPL(ctx)
        repl.cmdloop()


@cli.command()
@click.option('--foo', required=True)
def a(foo):
    print("a")
    print(foo)
    return 'banana'


@cli.command()
@click.option('--foo', required=True)
def b(foo):
    print("b")
    print(foo)

if __name__ == "__main__":
    cli()
于 2017-07-07T00:31:08.600 回答
1

我试图做一些类似于 OP 的事情,但有额外的选项/嵌套的子子命令。使用内置 cmd 模块的第一个答案在我的情况下不起作用;也许还有更多的摆弄..但我确实遇到了click-shell。还没有机会对其进行广泛测试,但到目前为止,它似乎完全按预期工作。

于 2017-02-01T07:27:33.653 回答