前言
在上一篇文章中,我们介绍了 click 中的“参数”,本文将继续深入了解 click,着重讲解它的“选项”。
1 | 本系列文章默认使用 Python 3 作为解释器进行讲解。 |
选项
通过 click.option 可以给命令增加选项,并通过配置函数的参数来配置不同功能的选项。
给选项命名
click.option 中的命令规则可参考参数名称。它接受的前两个参数为长、短选项(顺序随意),其中:
- 长选项以 “–” 开头,比如 “–string-to-echo”
- 短选项以 “-” 开头,比如 “-s”
第三个参数为选项参数的名称,如果不指定,将会使用长选项的下划线形式名称:
1 |
|
显示指定为 string
1 |
|
基本值选项
值选项是非常常用的选项,它接受一个值。如果在命令行中提供了值选项,则需要提供对应的值;反之则使用默认值。若没在 click.option 中指定默认值,则默认值为 None,且该选项的类型为 STRING;反之,则选项类型为默认值的类型。
比如,提供默认值为 1,则选项类型为 INT:
1 |
|
如果要求选项为必填,则可指定 click.option 的 required=True:
1 |
|
如果选项名称和 Python 中的关键字冲突,则可以显式的指定选项名称。比如将 --from 的名称设置为 from_:
1 |
|
如果要在帮助中显式默认值,则可指定 click.option 的 show_default=True:
1 |
|
在命令行中调用则有:
1 | $ dots --help |
多值选项
有时,我们会希望命令行中一个选项能接收多个值,通过指定 click.option 中的 nargs 参数(必须是大于等于 0)。这样,接收的多值选项就会变成一个元组。
比如,在下面的示例中,当通过 --pos 指定多个值时,pos 变量就是一个元组,里面的每个元素是一个 float:
1 |
|
在命令行中调用则有:
1 | $ findme --pos 2.0 3.0 |
有时,通过同一选项指定的多个值得类型可能不同,这个时候可以指定 click.option 中的 type=(类型1, 类型2, ...) 来实现。而由于元组的长度同时表示了值的数量,所以就无须指定 nargs 参数。
1 |
|
在命令行中调用则有:
1 | $ putitem --item peter 1338 |
多选项
不同于多值选项是通过一个选项指定多个值,多选项则是使用多个相同选项分别指定值,通过 click.option 中的 multiple=True 来实现。
当我们定义如下多选项:
1 |
|
便可以指定任意数量个选项来指定值,获取到的 message 是一个元组:
1 | $ commit -m foo -m bar --message baz |
计值选项
有时我们可能需要获得选项的数量,那么可以指定 click.option 中的 count=True 来实现。
最常见的使用场景就是指定多个 --verbose 或 -v 选项来表示输出内容的详细程度。
1 |
|
在命令行中调用则有:
1 | $ log -vvv |
通过上面的例子,verbose 就是数字,表示 -v 选项的数量,由此可以进一步使用该值来控制日志的详细程度。
布尔选项
布尔选项用来表示真或假,它有多种实现方式:
- 通过
click.option的is_flag=True参数来实现:
1 | import sys |
在命令行中调用则有:
1 | $ info --shout |
- 通过在
click.option的选项定义中使用/分隔表示真假两个选项来实现:
1 | import sys |
在命令行中调用则有:
1 | $ info --shout |
在 Windows 中,一个选项可以以 / 开头,这样就会真假选项的分隔符冲突了,这个时候可以使用 ; 进行分隔:
1 |
|
在 cmd 中调用则有:
1 | > log /debug |
特性切换选项
所谓特性切换就是切换同一个操作对象的不同特性,比如指定 --upper 就让输出大写,指定 --lower 就让输出小写。这么来看,布尔值其实是特性切换的一个特例。
要实现特性切换选项,需要让多个选项都有相同的参数名称,并且定义它们的标记值 flag_value:
1 | import sys |
在命令行中调用则有:
1 | $ info --upper |
在上面的示例中,--upper 和 --lower 都有相同的参数值 transformation:
- 当指定
--upper时,transformation就是--upper选项的标记值upper - 当指定
--lower时,transformation就是--lower选项的标记值lower
进而就可以做进一步的业务逻辑处理。
选择项选项
选择项选项 和 上篇文章中介绍的 选择项参数 类似,只不过是限定选项内容,依旧是通过 type=click.Choice 实现。此外,case_sensitive=False 还可以忽略选项内容的大小写。
1 |
|
在命令行中调用则有:
1 | $ digest --hash-type=MD5 |
提示选项
顾名思义,当提供了选项却没有提供对应的值时,会提示用户输入值。这种交互式的方式会让命令行变得更加友好。通过指定 click.option 中的 prompt 可以实现。
- 当
prompt=True时,提示内容为选项的参数名称
1 |
|
在命令行调用则有:
1 | $ hello --name=John |
- 当
prompt='Your name please'时,提示内容为指定内容
1 |
|
在命令行中调用则有:
1 | $ hello |
基于提示选项,我们还可以指定 hide_input=True 来隐藏输入,confirmation_prompt=True 来让用户进行二次输入,这非常适合输入密码的场景。
1 |
|
当然,也可以直接使用 click.password_option:
1 |
|
我们还可以给提示选项设置默认值,通过 default 参数进行设置,如果被设置为函数,则可以实现动态默认值。
1 |
|
详情请阅读 Dynamic Defaults for Prompts。
范围选项
如果希望选项的值在某个范围内,就可以使用范围选项,通过指定 type=click.IntRange 来实现。它有两种模式:
- 默认模式(非强制模式),如果值不在区间范围内将会引发一个错误。如
type=click.IntRange(0, 10)表示范围是 [0, 10],超过该范围报错 - 强制模式,如果值不在区间范围内,将会强制选取一个区间临近值。如
click.IntRange(0, None, clamp=True)表示范围是 [0, +∞),小于 0 则取 0,大于 20 则取 20。其中None表示没有限制
1 |
|
在命令行中调用则有:
1 | $ repeat --count=1000 --digit=5 |
回调和优先
回调
通过 click.option 中的 callback 可以指定选项的回调,它会在该选项被解析后调用。回调函数的签名如下:
1 | def callback(ctx, param, value): |
其中:
- ctx 是命令的上下文 click.Context
- param 为选项变量 click.Option
- value 为选项的值
使用回调函数可以完成额外的参数校验逻辑。比如,通过 –rolls 的选项来指定摇骰子的方式,内容为“{N}d{M}”,表示 M 面的骰子摇 N 次,N 和 M 都是数字。在真正的处理 rolls 前,我们需要通过回调函数来校验它的格式:
1 | def validate_rolls(ctx, param, value): |
这样,当我们输入错误格式时,变会校验不通过:
1 | $ roll --rolls=42 |
输入正确格式时,则正常输出信息:
1 | $ roll --rolls=2d12 |
优先
通过 click.option 中的 is_eager 可以让该选项成为优先选项,这意味着它会先于所有选项处理。
利用回调和优先选项,我们就可以很好地实现 --version 选项。不论命令行中写了多少选项和参数,只要包含了 --version,我们就希望它打印版本就退出,而不执行其他选项的逻辑,那么就需要让它成为优先选项,并且在回调函数中打印版本。
此外,在 click 中每个选项都对应到命令处理函数的同名参数,如果不想把该选项传递到处理函数中,则需要指定 expose_value=True,于是有:
1 | def print_version(ctx, param, value): |
当然 click 提供了便捷的 click.version_option 来实现 --version:
1 |
|
Yes 选项
基于前面的学习,我们可以实现 Yes 选项,也就是对于某些操作,不提供 --yes 则进行二次确认,提供了则直接操作:
1 | def abort_if_false(ctx, param, value): |
当然 click 提供了便捷的 click.confirmation_option 来实现 Yes 选项:
1 |
|
在命令行中调用则有:
1 | $ dropdb |
其他增强功能
click 支持从环境中读取选项的值,这是 argparse 所不支持的,可参阅官方文档的 Values from Environment Variables 和 Multiple Values from Environment Values。
click 支持指定选项前缀,你可以不使用 - 作为选项前缀,还可使用 + 或 /,当然在一般情况下并不建议这么做。详情参阅官方文档的 Other Prefix Characters
总结
可以看出,click 对命令行选项的支持非常丰富和强大,除了支持 argarse 所支持的所有选项类型外,还提供了诸如 计值选项、特性切换选项、提示选项 等更丰富的选项类型。此外,还提供了从环境中读变量等方便易用的增强功能。简直就是开发命令行程序的利器。
在下篇文章中,我们着重介绍下 click 的命令和组,这可是实现它的重要特性(任意嵌套命令)的方式。
相关文章
- Python 命令行之旅:初探 argparse
- Python 命令行之旅:深入 argparse(一)
- Python 命令行之旅:深入 argparse(二)
- Python 命令行之旅:使用 argparse 实现 git 命令
- Python 命令行之旅:初探 docopt
- Python 命令行之旅:深入 docopt
- Python 命令行之旅:使用 docopt 实现 git 命令
- Python 命令行之旅:初探 click
- Python 命令行之旅:深入 click(一)
- Python 命令行之旅:深入 click(二)
- Python 命令行之旅:深入 click(三)
- Python 命令行之旅:深入 click(四)
- Python 命令行之旅:使用 click 实现 git 命令
- Python 命令行之旅:初探 fire
- Python 命令行之旅:深入 fire(一)
- Python 命令行之旅:深入 fire(二)
- Python 命令行之旅:使用 fire 实现 git 命令
- Python 命令行之旅:argparse、docopt、click 和 fire 总结篇
- Python 命令行大乱斗
