gpt4 book ai didi

python - Python click 库中急切选项的非交互式确认

转载 作者:行者123 更新时间:2023-12-04 03:34:01 25 4
gpt4 key购买 nike

我有一个用于重置配置文件的点击子命令,myprog config-reset:

import click

@click.command()
@click.confirmation_option(prompt="Are you sure?")
def config_reset():
"""Reset the user configuration."""
click.echo("Success")

click.confirmation_option的使用允许我通过 --yes 绕过“你确定吗?”提示,这对测试很有用:

$ myprog config-reset --yes  # No interactive confirmation required.

出于其他原因,我现在将 config-reset 子命令合并到一个通用的 config 子命令中,这样用户就可以调用 myprog config --重置。我通过使用一个急切的选项来实现这一点,如果 --reset 存在,它会执行函数 config_reset,而不是执行主函数体:

def config_reset(ctx, option, value):
if not value or ctx.resilient_parsing: # Copied from click docs.
return

click.confirm("Are you sure?", abort=True)
# If we got this far, the config should be reset...

@click.command()
@click.option(
"--reset",
is_flag=True,
callback=config_reset,
expose_value=False,
is_eager=True,
help="Reset the user configuration to the default.",
)
def config():
"""Show the configuration."""
# print config here...

现在的问题是,我没有办法通过 --yes 选项来绕过交互式“Are you sure?”确认。有没有办法结合 is_eager=True 来实现 click.confirmation_option?本质上我希望能够写...

$ myprog config --reset --yes

...并让配置在没有确认的情况下自行重置。

最佳答案

我们可以使用 --yes 选项通过创建自定义 click.Option 类来跳过急切回调中的确认。

自定义类

def confirm_eager_option(*args, confirm_skip_flag_name='yes'):

class ConfirmEagerOption(click.Option):
def __init__(self, param_decls, **kwargs):
assert not param_decls, 'pass options names into confirm_eager_option()'
self.original_callback = kwargs['callback']
kwargs['callback'] = self.callback_handler
kwargs['is_eager'] = True
kwargs['is_flag'] = True
kwargs['expose_value'] = False
super().__init__(args, **kwargs)

def handle_parse_result(self, ctx, opts, args):
# grab the parsed options for later reference
self.parse_opts = opts
return super().handle_parse_result(ctx, opts, args)

def callback_handler(self, ctx, option, value):
if value and not self.parse_opts.get('help'):
# If we have the confirm flag, then set value to None
if self.parse_opts.get(confirm_skip_flag_name):
value = None
self.original_callback(ctx, option, value)

# build a decorator with our skip flag
confirm_skip_option = click.option(
f'--{confirm_skip_flag_name}', is_flag=True, expose_value=False, is_eager=True,
help=f"Skip confirmation for {args[0]}"
)

return ConfirmEagerOption

使用自定义类:

要使用自定义类,首先通过调用 confirm_eager_option() 构建类并通过它是选项名称和跳过标志名称,如:

confirm_eager_option_cls = confirm_eager_option(
'--reset', confirm_skip_flag_name='yes')

然后将该类作为 click.option 装饰器的 cls 参数传递,并添加另一个跳过选项的装饰器,如::

@click.command()
@click.option(cls=confirm_eager_option_cls, callback=config_reset,
help="Reset the user configuration to the default.")
@confirm_eager_option_cls.confirm_skip_option
def cli():
...

这是如何运作的?

之所以可行,是因为 click 是一个设计良好的 OO 框架。 @click.option() 装饰器通常实例化一个 click.Option 对象,但允许使用 cls 覆盖此行为范围。所以在我们自己的类中继承click.Option是一件相对容易的事情乘坐所需的方法。

在这种情况下,我们重写了 handle_parse_result() 方法。重写的方法允许我们捕获其他已解析的选项,然后在急切的回调 Hook 中查看这些选项以决定是否允许跳过确认。

测试代码:

confirm_eager_option_cls = confirm_eager_option(
'--reset', confirm_skip_flag_name='yes')

def config_reset(ctx, option, value):
if value is None or value:
if value:
click.echo('Confirming reset...')
# click.confirm("Are you sure?", abort=True)
click.echo('Resetting...')

@click.command()
@click.option(cls=confirm_eager_option_cls, callback=config_reset,
help="Reset the user configuration to the default.")
@confirm_eager_option_cls.confirm_skip_option
def cli():
click.echo('running command')

if __name__ == "__main__":
commands = (
'',
'--reset',
'--reset --yes',
'--help',
)

import sys, time
time.sleep(1)
print('Click Version: {}'.format(click.__version__))
print('Python Version: {}'.format(sys.version))
for cmd in commands:
try:
time.sleep(0.1)
print('-----------')
print('> ' + cmd)
time.sleep(0.1)
cli(cmd.split())

except BaseException as exc:
if str(exc) != '0' and \
not isinstance(exc, (click.ClickException, SystemExit)):
raise

测试结果:

Click Version: 7.1.2
Python Version: 3.8.5 (tags/v3.8.5:580fbb0, Jul 20 2020, 15:57:54) [MSC v.1924 64 bit (AMD64)]
-----------
>
running command
-----------
> --reset
Confirming reset...
Resetting...
running command
-----------
> --reset --yes
Resetting...
running command
-----------
> --help
Usage: test_code.py [OPTIONS]

Options:
--reset Reset the user configuration to the default.
--yes Skip confirmation for --reset
--help Show this message and exit.

关于python - Python click 库中急切选项的非交互式确认,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67297248/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com