kiwi-scp/kiwi_scp/misc.py

112 lines
2.6 KiB
Python
Raw Normal View History

import re
from typing import Any, Type, List, Callable
2021-10-20 12:32:45 +00:00
import attr
2021-10-20 12:32:45 +00:00
import click
2021-10-27 11:33:26 +00:00
import ruamel.yaml
from click.decorators import FC
from ._constants import HEADER_KIWI_CONF_NAME
@attr.s
class _MultiDecorator:
options: List[Callable[[FC], FC]] = attr.ib(factory=list)
def __call__(self, target: FC):
for option in reversed(self.options):
target = option(target)
return target
2021-10-21 08:34:53 +00:00
_project_arg = click.argument(
"project",
required=False,
type=str,
2021-10-21 08:34:53 +00:00
)
_service_arg = click.argument(
"service",
required=False,
type=str,
)
instance_command = _MultiDecorator([])
project_command = _MultiDecorator([_project_arg])
service_command = _MultiDecorator([_project_arg, _service_arg])
2021-10-20 12:32:45 +00:00
def user_query(description: str, default: Any, cast_to: Type[Any] = str):
# prompt user as per argument
while True:
try:
str_value = input(f"Enter {description} [{default}] ").strip()
if str_value:
return cast_to(str_value)
else:
return default
except EOFError:
click.echo("Input aborted.")
return default
except Exception as e:
click.echo(f"Invalid input: {e}")
2021-10-27 11:33:26 +00:00
class YAML(ruamel.yaml.YAML):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.indent(offset=2)
def _format_kiwi_yml(yml_string: str):
# insert newline before every main key
yml_string = re.sub(r'^(\S)', r'\n\1', yml_string, flags=re.MULTILINE)
# load header comment from file
with open(HEADER_KIWI_CONF_NAME, 'r') as stream:
yml_string = stream.read() + yml_string
return yml_string
def _surround(string, bang):
midlane = f"{bang * 3} {string} {bang * 3}"
2021-09-22 01:06:43 +00:00
sidelane = bang * len(midlane)
return f"{sidelane}\n{midlane}\n{sidelane}"
2020-08-18 13:05:19 +00:00
def _emphasize(lines):
if isinstance(lines, list):
return '\n'.join([_emphasize(line) for line in lines])
elif lines:
return f">>> {lines} <<<"
else:
return lines
def are_you_sure(prompt, default="no"):
if default.lower() == 'yes':
suffix = "[YES|no]"
else:
suffix = "[yes|NO]"
answer = input(
2020-08-25 16:01:39 +00:00
f"{_surround('MUST HAVE CAREFULING IN PROCESS', '!')}\n"
f"\n"
2020-08-18 13:05:19 +00:00
f"{_emphasize(prompt)}\n"
f"\n"
f"Are you sure you want to proceed? {suffix} "
).strip().lower()
if answer == '':
answer = default
while answer not in ['yes', 'no']:
answer = input("Please type 'yes' or 'no' explicitly: ").strip().lower()
return answer == 'yes'