文中涉及的示例代码,已同步更新到 HelloGitHub-Team 仓库python
在上一篇文章中,咱们介绍了 click
中的“参数”,本文将继续深刻了解 click
,着重讲解它的“选项”。linux
本系列文章默认使用 Python 3 做为解释器进行讲解。
若你仍在使用 Python 2,请注意二者之间语法和库的使用差别哦~
复制代码
经过 click.option
能够给命令增长选项,并经过配置函数的参数来配置不一样功能的选项。git
click.option
中的命令规则可参考参数名称。它接受的前两个参数为长、短选项(顺序随意),其中:github
第三个参数为选项参数的名称,若是不指定,将会使用长选项的下划线形式名称:编程
@click.command()
@click.option('-s', '--string-to-echo')
def echo(string_to_echo):
click.echo(string_to_echo)
复制代码
显示指定为 stringapi
@click.command()
@click.option('-s', '--string-to-echo', 'string')
def echo(string):
click.echo(string)
复制代码
值选项是很是经常使用的选项,它接受一个值。若是在命令行中提供了值选项,则须要提供对应的值;反之则使用默认值。若没在 click.option
中指定默认值,则默认值为 None
,且该选项的类型为 STRING;反之,则选项类型为默认值的类型。bash
好比,提供默认值为 1,则选项类型为 INT:ide
@click.command()
@click.option('--n', default=1)
def dots(n):
click.echo('.' * n)
复制代码
若是要求选项为必填,则可指定 click.option
的 required=True
:函数
@click.command()
@click.option('--n', required=True, type=int)
def dots(n):
click.echo('.' * n)
复制代码
若是选项名称和 Python 中的关键字冲突,则能够显式的指定选项名称。好比将 --from
的名称设置为 from_
:学习
@click.command()
@click.option('--from', '-f', 'from_')
@click.option('--to', '-t')
def reserved_param_name(from_, to):
click.echo(f'from {from_} to {to}')
复制代码
若是要在帮助中显式默认值,则可指定 click.option
的 show_default=True
:
@click.command()
@click.option('--n', default=1, show_default=True)
def dots(n):
click.echo('.' * n)
复制代码
在命令行中调用则有:
$ dots --help
Usage: dots [OPTIONS]
Options:
--n INTEGER [default: 1]
--help Show this message and exit.
复制代码
有时,咱们会但愿命令行中一个选项能接收多个值,经过指定 click.option
中的 nargs
参数(必须是大于等于 0)。这样,接收的多值选项就会变成一个元组。
好比,在下面的示例中,当经过 --pos
指定多个值时,pos
变量就是一个元组,里面的每一个元素是一个 float
:
@click.command()
@click.option('--pos', nargs=2, type=float)
def findme(pos):
click.echo(pos)
复制代码
在命令行中调用则有:
$ findme --pos 2.0 3.0
(1.0, 2.0)
复制代码
有时,经过同一选项指定的多个值得类型可能不一样,这个时候能够指定 click.option
中的 type=(类型1, 类型2, ...)
来实现。而因为元组的长度同时表示了值的数量,因此就无须指定 nargs
参数。
@click.command()
@click.option('--item', type=(str, int))
def putitem(item):
click.echo('name=%s id=%d' % item)
复制代码
在命令行中调用则有:
$ putitem --item peter 1338
name=peter id=1338
复制代码
不一样于多值选项是经过一个选项指定多个值,多选项则是使用多个相同选项分别指定值,经过 click.option
中的 multiple=True
来实现。
当咱们定义以下多选项:
@click.command()
@click.option('--message', '-m', multiple=True)
def commit(message):
click.echo('\n'.join(message))
复制代码
即可以指定任意数量个选项来指定值,获取到的 message
是一个元组:
$ commit -m foo -m bar --message baz
foo
bar
baz
复制代码
有时咱们可能须要得到选项的数量,那么能够指定 click.option
中的 count=True
来实现。
最多见的使用场景就是指定多个 --verbose
或 -v
选项来表示输出内容的详细程度。
@click.command()
@click.option('-v', '--verbose', count=True)
def log(verbose):
click.echo(f'Verbosity: {verbose}')
复制代码
在命令行中调用则有:
$ log -vvv
Verbosity: 3
复制代码
经过上面的例子,verbose
就是数字,表示 -v
选项的数量,由此能够进一步使用该值来控制日志的详细程度。
布尔选项用来表示真或假,它有多种实现方式:
click.option
的 is_flag=True
参数来实现:import sys
@click.command()
@click.option('--shout', is_flag=True)
def info(shout):
rv = sys.platform
if shout:
rv = rv.upper() + '!!!!111'
click.echo(rv)
复制代码
在命令行中调用则有:
$ info --shout
LINUX!!!!111
复制代码
click.option
的选项定义中使用 /
分隔表示真假两个选项来实现:import sys
@click.command()
@click.option('--shout/--no-shout', default=False)
def info(shout):
rv = sys.platform
if shout:
rv = rv.upper() + '!!!!111'
click.echo(rv)
复制代码
在命令行中调用则有:
$ info --shout
LINUX!!!!111
$ info --no-shout
linux
复制代码
在 Windows 中,一个选项能够以 /
开头,这样就会真假选项的分隔符冲突了,这个时候可使用 ;
进行分隔:
@click.command()
@click.option('/debug;/no-debug')
def log(debug):
click.echo(f'debug={debug}')
if __name__ == '__main__':
log()
复制代码
在 cmd 中调用则有:
> log /debug
debug=True
复制代码
所谓特性切换就是切换同一个操做对象的不一样特性,好比指定 --upper
就让输出大写,指定 --lower
就让输出小写。这么来看,布尔值实际上是特性切换的一个特例。
要实现特性切换选项,须要让多个选项都有相同的参数名称,而且定义它们的标记值 flag_value
:
import sys
@click.command()
@click.option('--upper', 'transformation', flag_value='upper',
default=True)
@click.option('--lower', 'transformation', flag_value='lower')
def info(transformation):
click.echo(getattr(sys.platform, transformation)())
复制代码
在命令行中调用则有:
$ info --upper
LINUX
$ info --lower
linux
$ info
LINUX
复制代码
在上面的示例中,--upper
和 --lower
都有相同的参数值 transformation
:
--upper
时,transformation
就是 --upper
选项的标记值 upper
--lower
时,transformation
就是 --lower
选项的标记值 lower
进而就能够作进一步的业务逻辑处理。
选择项选项
和 上篇文章中介绍的 选择项参数
相似,只不过是限定选项内容,依旧是经过 type=click.Choice
实现。此外,case_sensitive=False
还能够忽略选项内容的大小写。
@click.command()
@click.option('--hash-type',
type=click.Choice(['MD5', 'SHA1'], case_sensitive=False))
def digest(hash_type):
click.echo(hash_type)
复制代码
在命令行中调用则有:
$ digest --hash-type=MD5
MD5
$ digest --hash-type=md5
MD5
$ digest --hash-type=foo
Usage: digest [OPTIONS]
Try "digest --help" for help.
Error: Invalid value for "--hash-type": invalid choice: foo. (choose from MD5, SHA1)
$ digest --help
Usage: digest [OPTIONS]
Options:
--hash-type [MD5|SHA1]
--help Show this message and exit.
复制代码
顾名思义,当提供了选项却没有提供对应的值时,会提示用户输入值。这种交互式的方式会让命令行变得更加友好。经过指定 click.option
中的 prompt
能够实现。
prompt=True
时,提示内容为选项的参数名称@click.command()
@click.option('--name', prompt=True)
def hello(name):
click.echo(f'Hello {name}!')
复制代码
在命令行调用则有:
$ hello --name=John
Hello John!
$ hello
Name: John
Hello John!
复制代码
prompt='Your name please'
时,提示内容为指定内容@click.command()
@click.option('--name', prompt='Your name please')
def hello(name):
click.echo(f'Hello {name}!')
复制代码
在命令行中调用则有:
$ hello
Your name please: John
Hello John!
复制代码
基于提示选项,咱们还能够指定 hide_input=True
来隐藏输入,confirmation_prompt=True
来让用户进行二次输入,这很是适合输入密码的场景。
@click.command()
@click.option('--password', prompt=True, hide_input=True,
confirmation_prompt=True)
def encrypt(password):
click.echo(f'Encrypting password to {password.encode("rot13")}')
复制代码
固然,也能够直接使用 click.password_option
:
@click.command()
@click.password_option()
def encrypt(password):
click.echo(f'Encrypting password to {password.encode("rot13")}')
复制代码
咱们还能够给提示选项设置默认值,经过 default
参数进行设置,若是被设置为函数,则能够实现动态默认值。
@click.command()
@click.option('--username', prompt=True,
default=lambda: os.environ.get('USER', ''))
def hello(username):
print("Hello,", username)
复制代码
详情请阅读 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
表示没有限制@click.command()
@click.option('--count', type=click.IntRange(0, None, clamp=True))
@click.option('--digit', type=click.IntRange(0, 10))
def repeat(count, digit):
click.echo(str(digit) * count)
if __name__ == '__main__':
repeat()
复制代码
在命令行中调用则有:
$ repeat --count=1000 --digit=5
55555555555555555555
$ repeat --count=1000 --digit=12
Usage: repeat [OPTIONS]
Error: Invalid value for "--digit": 12 is not in the valid range of 0 to 10.
复制代码
回调 经过 click.option
中的 callback
能够指定选项的回调,它会在该选项被解析后调用。回调函数的签名以下:
def callback(ctx, param, value):
pass
复制代码
其中:
使用回调函数能够完成额外的参数校验逻辑。好比,经过 --rolls 的选项来指定摇骰子的方式,内容为“{N}d{M}”,表示 M 面的骰子摇 N 次,N 和 M 都是数字。在真正的处理 rolls 前,咱们须要经过回调函数来校验它的格式:
def validate_rolls(ctx, param, value):
try:
rolls, dice = map(int, value.split('d', 2))
return (dice, rolls)
except ValueError:
raise click.BadParameter('rolls need to be in format NdM')
@click.command()
@click.option('--rolls', callback=validate_rolls, default='1d6')
def roll(rolls):
click.echo('Rolling a %d-sided dice %d time(s)' % rolls)
复制代码
这样,当咱们输入错误格式时,变会校验不经过:
$ roll --rolls=42
Usage: roll [OPTIONS]
Error: Invalid value for "--rolls": rolls need to be in format NdM
复制代码
输入正确格式时,则正常输出信息:
$ roll --rolls=2d12
Rolling a 12-sided dice 2 time(s)
复制代码
优先 经过 click.option
中的 is_eager
可让该选项成为优先选项,这意味着它会先于全部选项处理。
利用回调和优先选项,咱们就能够很好地实现 --version
选项。不论命令行中写了多少选项和参数,只要包含了 --version
,咱们就但愿它打印版本就退出,而不执行其余选项的逻辑,那么就须要让它成为优先选项,而且在回调函数中打印版本。
此外,在 click
中每一个选项都对应到命令处理函数的同名参数,若是不想把该选项传递处处理函数中,则须要指定 expose_value=True
,因而有:
def print_version(ctx, param, value):
if not value or ctx.resilient_parsing:
return
click.echo('Version 1.0')
ctx.exit()
@click.command()
@click.option('--version', is_flag=True, callback=print_version,
expose_value=False, is_eager=True)
def hello():
click.echo('Hello World!')
复制代码
固然 click
提供了便捷的 click.version_option
来实现 --version
:
@click.command()
@click.version_option(version='0.1.0')
def hello():
pass
复制代码
基于前面的学习,咱们能够实现 Yes 选项,也就是对于某些操做,不提供 --yes
则进行二次确认,提供了则直接操做:
def abort_if_false(ctx, param, value):
if not value:
ctx.abort()
@click.command()
@click.option('--yes', is_flag=True, callback=abort_if_false,
expose_value=False,
prompt='Are you sure you want to drop the db?')
def dropdb():
click.echo('Dropped all tables!')
复制代码
固然 click
提供了便捷的 click.confirmation_option
来实现 Yes 选项:
@click.command()
@click.confirmation_option(prompt='Are you sure you want to drop the db?')
def dropdb():
click.echo('Dropped all tables!')
复制代码
在命令行中调用则有:
$ dropdb
Are you sure you want to drop the db? [y/N]: n
Aborted!
$ dropdb --yes
Dropped all tables!
复制代码
click
支持从环境中读取选项的值,这是 argparse
所不支持的,可参阅官方文档的 Values from Environment Variables 和 Multiple Values from Environment Values。
click
支持指定选项前缀,你能够不使用 -
做为选项前缀,还可以使用 +
或 /
,固然在通常状况下并不建议这么作。详情参阅官方文档的 Other Prefix Characters
能够看出,click
对命令行选项的支持很是丰富和强大,除了支持 argarse
所支持的全部选项类型外,还提供了诸如 计值选项
、特性切换选项
、提示选项
等更丰富的选项类型。此外,还提供了从环境中读变量等方便易用的加强功能。简直就是开发命令行程序的利器。
在下篇文章中,咱们着重介绍下 click
的命令和组,这但是实现它的重要特性(任意嵌套命令)的方式。
『讲解开源项目系列』——让对开源项目感兴趣的人再也不畏惧、让开源项目的发起者再也不孤单。跟着咱们的文章,你会发现编程的乐趣、使用和发现参与开源项目如此简单。欢迎留言联系咱们、加入咱们,让更多人爱上开源、贡献开源~