作者:HelloGitHub-Prodesire

HelloGitHub 的《讲解开源项目》系列,项目地址:https://github.com/HelloGitHub-Team/Article

一、前言

在前面三篇文章中,我们介绍了 click 中的参数、选项和命令,本文将介绍 click 锦上添花的功能,以帮助我们更加轻松地打造一个更加强大的命令行程序。

  1. 本系列文章默认使用 Python 3 作为解释器进行讲解。
  2. 若你仍在使用 Python 2,请注意两者之间语法和库的使用差异哦~

二、增强功能

2.1 Bash 补全

Bash 补全是 click 提供的一个非常便捷和强大的功能,这是它比 argpasedocopt 强大的一个表现。

在命令行程序正确安装后,Bash 补全才可以使用。而如何安装可以参考 setup 集成。Click 目前仅支持 Bash 和 Zsh 的补全。

2.1.1 补全能力

通常来说,Bash 补全支持对子命令、选项、以及选项或参数值得补全。比如:

  1. $ repo <TAB><TAB>
  2. clone commit copy delete setuser
  3. $ repo clone -<TAB><TAB>
  4. --deep --help --rev --shallow -r

此外,click 还支持自定义补全,这在动态生成补全场景中很有用,使用 autocompletion 参数。autocompletion 需要指定为一个回调函数,并且返回字符串的列表。此函数接受三个参数:

  • ctx —— 当前的 click 上下文
  • args 传入的参数列表
  • incomplete 正在补全的词

这里有一个根据环境变量动态生成补全的示例:

  1. import os
  2. def get_env_vars(ctx, args, incomplete):
  3. return [k for k in os.environ.keys() if incomplete in k]
  4. @click.command()
  5. @click.argument("envvar", type=click.STRING, autocompletion=get_env_vars)
  6. def cmd1(envvar):
  7. click.echo('Environment variable: %s' % envvar)
  8. click.echo('Value: %s' % os.environ[envvar])

ZSH 中,还支持补全帮助信息。只需将 autocompletion 回调函数中返回的字符串列表中的字符串改为二元元组,第一个元素是补全内容,第二个元素是帮助信息。

这里有一个颜色补全的示例:

  1. import os
  2. def get_colors(ctx, args, incomplete):
  3. colors = [('red', 'help string for the color red'),
  4. ('blue', 'help string for the color blue'),
  5. ('green', 'help string for the color green')]
  6. return [c for c in colors if incomplete in c[0]]
  7. @click.command()
  8. @click.argument("color", type=click.STRING, autocompletion=get_colors)
  9. def cmd1(color):
  10. click.echo('Chosen color is %s' % color)

2.1.2 激活补全

要激活 Bash 的补全功能,就需要告诉它你的命令行程序有补全的能力。通常通过一个神奇的环境变量 _<PROG_NAME>_COMPLETE 来告知,其中 <PROG_NAME> 是大写下划线形式的程序名称。

比如有一个命令行程序叫做 foo-bar,那么对应的环境变量名称为 _FOO_BAR_COMPLETE,然后在 .bashrc 中使用 source 导出即可:

  1. eval "$(_FOO_BAR_COMPLETE=source foo-bar)"

或者在 .zshrc 中使用:

  1. eval "$(_FOO_BAR_COMPLETE=source_zsh foo-bar)"

不过上面的方式总是在命令行程序启动时调用,这可能在有多个程序时减慢 shell 激活的速度。另一种方式是把命令放在文件中,就像这样:

  1. # 针对 Bash
  2. _FOO_BAR_COMPLETE=source foo-bar > foo-bar-complete.sh
  3. # 针对 ZSH
  4. _FOO_BAR_COMPLETE=source_zsh foo-bar > foo-bar-complete.sh

然后把脚本文件路径加到 .bashrc.zshrc 中:

  1. . /path/to/foo-bar-complete.sh

2.2 实用工具

2.2.1 打印到标准输出

echo() 函数可以说是最有用的实用工具了。它和 Python 的 print 类似,主要的区别在于它同时在 Python 2 和 3 中生效,能够智能地检测未配置正确的输出流,且几乎不会失败(除了 Python 3 中的少数限制。)

echo 即支持 unicode,也支持二级制数据,如:

  1. import click
  2. click.echo('Hello World!')
  3. click.echo(b'\xe2\x98\x83', nl=False) # nl=False 表示不输出换行符

2.2.2 ANSI 颜色

有些时候你可能希望输出是有颜色的,这尤其在输出错误信息时有用,而 click 在这方面支持的很好。

首先,你需要安装 colorama

  1. pip install colorama

然后,就可以使用 style() 函数来指定颜色:

  1. import click
  2. click.echo(click.style('Hello World!', fg='green'))
  3. click.echo(click.style('Some more text', bg='blue', fg='white'))
  4. click.echo(click.style('ATTENTION', blink=True, bold=True))

click 还提供了更加简便的函数 secho,它就是 echostyle 的组合:

  1. click.secho('Hello World!', fg='green')
  2. click.secho('Some more text', bg='blue', fg='white')
  3. click.secho('ATTENTION', blink=True, bold=True)

2.2.3 分页支持

有些时候,命令行程序会输出长文本,但你希望能让用户盘也浏览。使用 echo_via_pager() 函数就可以轻松做到。

例如:

  1. def less():
  2. click.echo_via_pager('\n'.join('Line %d' % idx
  3. for idx in range(200)))

如果输出的文本特别大,处于性能的考虑,希望翻页时生成对应内容,那么就可以使用生成器:

  1. def _generate_output():
  2. for idx in range(50000):
  3. yield "Line %d\n" % idx
  4. @click.command()
  5. def less():
  6. click.echo_via_pager(_generate_output())

2.2.4 清除屏幕

使用 clear() 可以轻松清除屏幕内容:

  1. import click
  2. click.clear()

2.2.5 从终端获取字符

通常情况下,使用内建函数 inputraw_input 获得的输入是用户输出一段字符然后回车得到的。但在有些场景下,你可能想在用户输入单个字符时就能获取到并且做一定的处理,这个时候 getchar() 就派上了用场。

比如,根据输入的 yn 做特定处理:

  1. import click
  2. click.echo('Continue? [yn] ', nl=False)
  3. c = click.getchar()
  4. click.echo()
  5. if c == 'y':
  6. click.echo('We will go on')
  7. elif c == 'n':
  8. click.echo('Abort!')
  9. else:
  10. click.echo('Invalid input :(')

2.2.6 等待按键

在 Windows 的 cmd 中我们经常看到当执行完一个命令后,提示按下任意键退出。通过使用 pause() 可以实现暂停直至用户按下任意键:

  1. import click
  2. click.pause()

2.2.7 启动编辑器

通过 edit() 可以自动启动编辑器。这在需要用户输入多行内容时十分有用。

在下面的示例中,会启动默认的文本编辑器,并在里面输入一段话:

  1. import click
  2. def get_commit_message():
  3. MARKER = '# Everything below is ignored\n'
  4. message = click.edit('\n\n' + MARKER)
  5. if message is not None:
  6. return message.split(MARKER, 1)[0].rstrip('\n')

edit() 函数还支持打开特定文件,比如:

  1. import click
  2. click.edit(filename='/etc/passwd')

2.2.8 启动应用程序

通过 launch 可以打开 URL 或文件类型所关联的默认应用程序。如果设置 locate=True,则可以启动文件管理器并自动选中特定文件。

示例:

  1. # 打开浏览器,访问 URL
  2. click.launch("https://click.palletsprojects.com/")
  3. # 使用默认应用程序打开 txt 文件
  4. click.launch("/my/downloaded/file.txt")
  5. # 打开文件管理器,并自动选中 file.txt
  6. click.launch("/my/downloaded/file.txt", locate=True)

2.2.9 显示进度条

click 内置了 progressbar() 函数来方便地显示进度条。

它的用法也很简单,假定你有一个要处理的可迭代对象,处理完每一项就要输出一下进度,那么就有两种用法。

用法一:使用 progressbar 构造出 bar 对象,迭代 bar 对象来自动告知进度:

  1. import time
  2. import click
  3. all_the_users_to_process = ['a', 'b', 'c']
  4. def modify_the_user(user):
  5. time.sleep(0.5)
  6. with click.progressbar(all_the_users_to_process) as bar:
  7. for user in bar:
  8. modify_the_user(user)

用法二:使用 progressbar 构造出 bar 对象,迭代原始可迭代对象,并不断向 bar 更新进度:

  1. import time
  2. import click
  3. all_the_users_to_process = ['a', 'b', 'c']
  4. def modify_the_user(user):
  5. time.sleep(0.5)
  6. with click.progressbar(all_the_users_to_process) as bar:
  7. for user in enumerate(all_the_users_to_process):
  8. modify_the_user(user)
  9. bar.update(1)

2.2.10 更多实用工具

三、总结

click 提供了非常多的增强型功能,本文着重介绍了它的 Bash 补全和十多个实用工具,这会让你在实现命令行的过程中如虎添翼。此外,click 还提供了诸如命令别名、参数修改、标准化令牌、调用其他命令、回调顺序等诸多高级模式 以应对更加复杂或特定的场景,我们就不再深入介绍。

click 的介绍就告一段落,它将会是你编写命令行程序的一大利器。在下一篇文章中,我们依然会通过实现一个简单的 git 程序来进行 click 的实战。


『讲解开源项目系列』——让对开源项目感兴趣的人不再畏惧、让开源项目的发起者不再孤单。跟着我们的文章,你会发现编程的乐趣、使用和发现参与开源项目如此简单。欢迎留言联系我们、加入我们,让更多人爱上开源、贡献开源~

Python 命令行之旅:深入 click 之增强功能的更多相关文章

  1. Python 命令行之旅 —— 初探 argparse

    『讲解开源项目系列』启动--让对开源项目感兴趣的人不再畏惧.让开源项目的发起者不再孤单.跟着我们的文章,你会发现编程的乐趣.使用和发现参与开源项目如此简单.欢迎联系我们给我们投稿,让更多人爱上开源.贡 ...

  2. Python 命令行之旅:深入 click 之参数篇

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  3. Python 命令行之旅:深入 click 之选项篇

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  4. Python 命令行之旅:深入 click 之子命令篇

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  5. Python 命令行之旅:使用 click 实现 git 命令

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  6. Python 命令行之旅:使用 docopt 实现 git 命令

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  7. Python 命令行之旅 —— 深入 argparse (一)

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  8. Python 命令行之旅:深入 argparse(二)

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

  9. Python 命令行之旅:使用 argparse 实现 git 命令

    作者:HelloGitHub-Prodesire HelloGitHub 的<讲解开源项目>系列,项目地址:https://github.com/HelloGitHub-Team/Arti ...

随机推荐

  1. LeetCode刷题总结-数组篇(下)

    本期讲O(n)类型问题,共14题.3道简单题,9道中等题,2道困难题.数组篇共归纳总结了50题,本篇是数组篇的最后一篇.其他三个篇章可参考: LeetCode刷题总结-数组篇(上),子数组问题(共17 ...

  2. 线段树合并学习笔记(P4556)

    直入主题: 学习线段树合并..... 从名字就能看出,这个东西要合并线段树..... 线段树怎么能合并呢...... 暴力合就行了啊...... 一次从上往下的遍历,把所有的节点信息暴力合并,然后就没 ...

  3. JavaScript-原型对象与原型链

    原型对象 1.每个对象一定会有一个原型对象 2.原型对象实际是构造实例对象的构造器中的一个属性,只不过这个属性是个对象 3.这个原型对象中的属性与方法,都会被对象实例所共享(类似python中的类方法 ...

  4. Elasticsearch系列---Elasticsearch的基本概念及工作原理

    基本概念 Elasticsearch有几个核心的概念,花几分钟时间了解一下,有助于后面章节的学习. NRT Near Realtime,近实时,有两个层面的含义,一是从写入一条数据到这条数据可以被搜索 ...

  5. 2-了解DBMS

    1.DB,DBS,DBMS的区别是什么?     1.1 DB 就是数据库,数据库是存储数据的集合,可理解为多个数据表     1.2 DBS 数据库系统,包括数据库,数据库管理系统和数据库管理人员D ...

  6. 深入理解java继承从“我爸是李刚”讲起

    目录 1.继承的概述 2.关于继承之后的成员变量 3.关于继承之后的成员方法 4.关于继承之后的构造方法 5.关于继承的多态性支持的例子 6.super与this的用法 前言 本文主要多方面讲解jav ...

  7. Servlet——用户登录案例

    案例:用户登录 * 用户登录案例需求: 1.编写login.html登录页面 username & password 两个输入框 2.使用Druid数据库连接池技术,操作mysql,day14 ...

  8. Github PageHelper 原理解析

    任何服务对数据库的日常操作,都离不开增删改查.如果一次查询的纪录很多,那我们必须采用分页的方式.对于一个Springboot项目,访问和查询MySQL数据库,持久化框架可以使用MyBatis,分页工具 ...

  9. nyoj 62-笨小熊(以对应数组中的ASC位 + 1)

    62-笨小熊 内存限制:64MB 时间限制:2000ms Special Judge: No accepted:15 submit:43 题目描述: 笨小熊的词汇量很小,所以每次做英语选择题的时候都很 ...

  10. ZeroC ICE的远程调用框架 Slice如何帮助我们进行Ice异步编程(AMI,AMD)

    Slice最大的用处就是为我们使用Ice进行编程,代劳绝大部分的重复性代码,并提供一些帮助性的框架代码,如用于AMI和AMD方式进行异步编程的回调框架. 当Slice不为我们生成代码时,我们仍然可以按 ...