2021-11-17 15:23:55 +00:00
|
|
|
import importlib
|
2021-10-20 06:31:36 +00:00
|
|
|
import os
|
2021-11-03 15:32:01 +00:00
|
|
|
import sys
|
2021-11-02 16:21:01 +00:00
|
|
|
from enum import Enum, auto
|
2021-11-17 15:23:55 +00:00
|
|
|
from typing import List, Tuple, Iterable, Type, Optional, TypeVar
|
2021-10-20 06:31:36 +00:00
|
|
|
|
|
|
|
import click
|
|
|
|
|
2021-11-13 00:12:27 +00:00
|
|
|
from ..instance import Instance, Project
|
2021-11-27 16:39:43 +00:00
|
|
|
from ..wstring import WParagraph, WAlignment
|
2021-10-29 11:37:59 +00:00
|
|
|
|
2021-10-20 06:31:36 +00:00
|
|
|
|
|
|
|
class KiwiCLI(click.MultiCommand):
|
2021-10-20 08:54:41 +00:00
|
|
|
"""Command Line Interface spread over multiple files in this directory"""
|
|
|
|
|
2021-11-17 15:23:55 +00:00
|
|
|
def list_commands(self, ctx: click.Context) -> List[str]:
|
2021-10-20 08:54:41 +00:00
|
|
|
"""list all the commands defined by cmd_*.py files in this directory"""
|
|
|
|
|
2021-11-17 15:23:55 +00:00
|
|
|
return [
|
2021-10-20 08:54:41 +00:00
|
|
|
filename[4:-3]
|
|
|
|
for filename in os.listdir(os.path.abspath(os.path.dirname(__file__)))
|
|
|
|
if filename.startswith("cmd_") and filename.endswith(".py")
|
2021-11-17 15:23:55 +00:00
|
|
|
]
|
2021-10-20 06:31:36 +00:00
|
|
|
|
2021-11-17 15:23:55 +00:00
|
|
|
def get_command(self, ctx: click.Context, cmd_name: str) -> Optional[click.Command]:
|
2021-10-20 08:54:41 +00:00
|
|
|
"""import and return a specific command"""
|
|
|
|
|
2021-10-20 06:31:36 +00:00
|
|
|
try:
|
2021-11-17 15:23:55 +00:00
|
|
|
cmd_module = importlib.import_module(f"kiwi_scp.commands.cmd_{cmd_name}")
|
|
|
|
|
2021-10-20 06:31:36 +00:00
|
|
|
except ImportError:
|
|
|
|
return
|
2021-11-17 15:23:55 +00:00
|
|
|
|
|
|
|
for cmd_name in dir(cmd_module):
|
|
|
|
member = getattr(cmd_module, cmd_name)
|
|
|
|
if isinstance(member, click.Command):
|
|
|
|
return member
|
|
|
|
|
|
|
|
|
|
|
|
T = TypeVar("T")
|
2021-11-02 16:21:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
class KiwiCommand:
|
2021-11-03 15:32:01 +00:00
|
|
|
@staticmethod
|
2021-11-17 15:23:55 +00:00
|
|
|
def print_multi_color(*content: Tuple[str, str]) -> None:
|
2021-11-03 15:32:01 +00:00
|
|
|
for message, color in content:
|
|
|
|
click.secho(message, fg=color, nl=False)
|
|
|
|
click.echo()
|
|
|
|
|
|
|
|
@staticmethod
|
2021-11-17 15:23:55 +00:00
|
|
|
def print_header(header: str) -> None:
|
2021-11-03 15:32:01 +00:00
|
|
|
click.secho(header, fg="green", bold=True)
|
|
|
|
|
|
|
|
@staticmethod
|
2021-11-17 15:23:55 +00:00
|
|
|
def print_error(error: str) -> None:
|
2021-11-06 02:45:27 +00:00
|
|
|
click.secho(error, file=sys.stderr, fg="red", bold=True)
|
2021-11-03 15:32:01 +00:00
|
|
|
|
|
|
|
@staticmethod
|
2021-11-17 15:23:55 +00:00
|
|
|
def print_list(content: Iterable[str]) -> None:
|
2021-11-03 15:32:01 +00:00
|
|
|
for item in content:
|
|
|
|
KiwiCommand.print_multi_color(
|
|
|
|
(" - ", "green"),
|
|
|
|
(item, "blue"),
|
|
|
|
)
|
|
|
|
|
2021-11-03 16:58:18 +00:00
|
|
|
@staticmethod
|
2021-11-17 15:23:55 +00:00
|
|
|
def user_query(description: str, default: T, cast_to: Type[T] = str) -> T:
|
2021-11-03 16:58:18 +00:00
|
|
|
# prompt user as per argument
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
prompt = \
|
|
|
|
click.style(f"Enter {description} [", fg="green") + \
|
|
|
|
click.style(default, fg="blue") + \
|
|
|
|
click.style("] ", fg="green")
|
|
|
|
str_value = input(prompt).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-11-27 16:39:43 +00:00
|
|
|
@staticmethod
|
|
|
|
def danger_confirm(*prompt_lines: str, default: Optional[bool] = None) -> bool:
|
|
|
|
if default is True:
|
|
|
|
suffix = "[YES|no]"
|
|
|
|
elif default is False:
|
|
|
|
suffix = "[yes|NO]"
|
|
|
|
else:
|
|
|
|
suffix = "[yes|no]"
|
|
|
|
|
|
|
|
dumb = WParagraph.from_strings(
|
|
|
|
click.style("WARNING", bold=True, underline=True, blink=True, fg="red"),
|
|
|
|
click.style("ここにゴミ", fg="cyan"),
|
|
|
|
click.style("を捨てないで下さい", fg="cyan"),
|
|
|
|
click.style("DO NOT DUMB HERE", fg="yellow"),
|
|
|
|
click.style("NO DUMB AREA", fg="yellow"),
|
|
|
|
).align().surround("!")
|
|
|
|
|
|
|
|
prompt = WParagraph.from_strings(*prompt_lines).align(WAlignment.LEFT).emphasize(3)
|
|
|
|
|
|
|
|
answer = input(
|
|
|
|
f"{dumb}\n\n"
|
|
|
|
f"{prompt}\n\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'
|
|
|
|
|
2021-11-02 16:21:01 +00:00
|
|
|
@classmethod
|
2021-11-03 00:12:32 +00:00
|
|
|
def run_for_instance(cls, instance: Instance, **kwargs) -> None:
|
2021-11-06 02:45:27 +00:00
|
|
|
for project_config in instance.config.projects:
|
|
|
|
project = instance.get_project(project_config.name)
|
2021-11-27 17:32:51 +00:00
|
|
|
cls.run_for_project(instance, project, **kwargs)
|
2021-11-02 16:21:01 +00:00
|
|
|
|
|
|
|
@classmethod
|
2021-11-27 17:32:51 +00:00
|
|
|
def run_for_project(cls, instance: Instance, project: Project, **kwargs) -> None:
|
2021-11-06 02:45:27 +00:00
|
|
|
service_names = [service.name for service in project.services.content]
|
|
|
|
cls.run_for_services(instance, project, service_names, **kwargs)
|
2021-11-02 16:21:01 +00:00
|
|
|
|
2021-11-27 17:32:51 +00:00
|
|
|
@classmethod
|
|
|
|
def run_for_new_project(cls, instance: Instance, project_name: str, **kwargs) -> None:
|
|
|
|
cls.print_error(f"Project '{project_name}' not in kiwi-scp instance at '{instance.directory}'!")
|
|
|
|
|
2021-11-02 16:21:01 +00:00
|
|
|
@classmethod
|
2021-11-06 02:45:27 +00:00
|
|
|
def run_for_services(cls, instance: Instance, project: Project, service_names: List[str], **kwargs) -> None:
|
|
|
|
raise Exception
|
2021-11-02 16:21:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
class KiwiCommandType(Enum):
|
|
|
|
INSTANCE = auto()
|
|
|
|
PROJECT = auto()
|
|
|
|
SERVICE = auto()
|