mirror of
https://gitlab.com/pulsechaincom/staking-deposit-cli.git
synced 2025-01-11 05:20:06 +00:00
100 lines
3.1 KiB
Python
100 lines
3.1 KiB
Python
import click
|
|
from typing import (
|
|
Any,
|
|
Callable,
|
|
Sequence,
|
|
Tuple,
|
|
Union,
|
|
)
|
|
|
|
from eth2deposit.exceptions import ValidationError
|
|
|
|
|
|
def _value_of(f: Union[Callable[[], Any], Any]) -> Any:
|
|
'''
|
|
If the input, f, is a function, return f(), else return f.
|
|
'''
|
|
return(f() if callable(f) else f)
|
|
|
|
|
|
class JITOption(click.Option):
|
|
'''
|
|
A click.Option, except certain values are recomputed before they are used.
|
|
'''
|
|
def __init__(
|
|
self,
|
|
param_decls: str,
|
|
default: Union[Callable[[], Any], None, Any] = None,
|
|
help: Union[Callable[[], str], str, None] = None,
|
|
prompt: Union[Callable[[], str], str, None] = None,
|
|
**kwargs: Any,
|
|
):
|
|
|
|
self.callable_default = default
|
|
self.callable_help = help
|
|
self.callable_prompt = prompt
|
|
|
|
return super().__init__(
|
|
param_decls=[_value_of(param_decls)],
|
|
default=_value_of(default),
|
|
help=_value_of(help),
|
|
prompt=_value_of(prompt),
|
|
**kwargs,
|
|
)
|
|
|
|
def prompt_for_value(self, ctx: click.Context) -> Any:
|
|
self.prompt = _value_of(self.callable_prompt)
|
|
return super().prompt_for_value(ctx)
|
|
|
|
def get_help_record(self, ctx: click.Context) -> Tuple[str, str]:
|
|
self.help = _value_of(self.callable_help)
|
|
return super().get_help_record(ctx)
|
|
|
|
def get_default(self, ctx: click.Context) -> Any:
|
|
self.default = _value_of(self.callable_default)
|
|
return super().get_default(ctx)
|
|
|
|
|
|
def jit_option(*args: Any, **kwargs: Any) -> Callable[[Any], Any]:
|
|
"""Attaches an option to the command. All positional arguments are
|
|
passed as parameter declarations to :class:`Option`; all keyword
|
|
arguments are forwarded unchanged (except ``cls``).
|
|
This is equivalent to creating an :class:`Option` instance manually
|
|
and attaching it to the :attr:`Command.params` list.
|
|
|
|
:param cls: the option class to instantiate. This defaults to
|
|
:class:`Option`.
|
|
"""
|
|
|
|
def decorator(f: Callable[[Any], Any]) -> Callable[[Any], Any]:
|
|
click.decorators._param_memo(f, JITOption(*args, **kwargs)) # type: ignore
|
|
return f
|
|
|
|
return decorator
|
|
|
|
|
|
def captive_prompt_callback(
|
|
processing_func: Callable[[str], str],
|
|
prompt: str
|
|
) -> Callable[[click.Context, Any, str], str]:
|
|
'''
|
|
Traps the user in a prompt until the value chosen is acceptable
|
|
as defined by `processing_func` not returning a ValidationError
|
|
:param processing_func: A function to process the user's input that possibly raises a ValidationError
|
|
:param prompt_func: the function that returns the text to prompt the user with
|
|
'''
|
|
def callback(ctx: click.Context, param: Any, user_input: str) -> str:
|
|
while True:
|
|
try:
|
|
return processing_func(user_input)
|
|
except ValidationError:
|
|
user_input = click.prompt(prompt)
|
|
return callback
|
|
|
|
|
|
def choice_prompt_func(prompt_func: Callable[[], str], choices: Sequence[str]) -> Callable[[], str]:
|
|
'''
|
|
Formats the prompt and choices in a printable manner.
|
|
'''
|
|
return lambda: '%s %s: ' % (prompt_func(), choices)
|