2025-07-10 05:14:06 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
2020-04-10 17:55:36 +02:00
|
|
|
from __future__ import annotations
|
2025-11-02 09:42:21 +01:00
|
|
|
from typing import Optional, List, Type, Union, TypeVar
|
2025-07-10 05:14:06 +02:00
|
|
|
import inspect, sys, re, abc, argparse
|
2020-04-10 17:55:36 +02:00
|
|
|
from argparse import ArgumentParser, _SubParsersAction
|
2017-10-14 17:25:24 +02:00
|
|
|
|
2020-04-04 16:32:39 +02:00
|
|
|
from jwutils import log
|
|
|
|
|
|
|
|
|
|
# full blown example of one level of nested subcommands
|
|
|
|
|
# git -C project remote -v show -n myremote
|
|
|
|
|
|
2025-07-10 05:14:06 +02:00
|
|
|
class Cmd(abc.ABC): # export
|
2017-10-14 17:25:24 +02:00
|
|
|
|
2019-03-10 16:38:59 +01:00
|
|
|
@abc.abstractmethod
|
2019-12-20 08:41:58 +01:00
|
|
|
async def run(self, args):
|
2017-10-14 17:25:24 +02:00
|
|
|
pass
|
|
|
|
|
|
2020-04-10 17:55:36 +02:00
|
|
|
def __init__(self, name: str, help: str) -> None:
|
2025-11-02 09:42:21 +01:00
|
|
|
from . import Cmds
|
2019-03-10 16:38:59 +01:00
|
|
|
self.name = name
|
2017-10-14 17:25:24 +02:00
|
|
|
self.help = help
|
2020-04-04 16:32:39 +02:00
|
|
|
self.parent = None
|
2020-04-10 17:55:36 +02:00
|
|
|
self.children: List[Cmd] = []
|
|
|
|
|
self.child_classes: List[Type[Cmd]] = []
|
2022-12-08 16:44:49 +01:00
|
|
|
self.app: Optional[Cmds] = None
|
2017-10-14 17:25:24 +02:00
|
|
|
|
2019-12-20 08:41:58 +01:00
|
|
|
async def _run(self, args):
|
2017-10-14 17:25:24 +02:00
|
|
|
pass
|
|
|
|
|
|
2020-04-10 17:55:36 +02:00
|
|
|
def add_parser(self, parsers) -> ArgumentParser:
|
2019-03-10 16:38:59 +01:00
|
|
|
r = parsers.add_parser(self.name, help=self.help,
|
2017-10-14 17:25:24 +02:00
|
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
|
2019-03-10 16:38:59 +01:00
|
|
|
r.set_defaults(func=self.run)
|
2017-10-14 17:25:24 +02:00
|
|
|
return r
|
|
|
|
|
|
2024-12-03 13:34:25 +01:00
|
|
|
def add_subcommands(self, cmd: Union[str, Type[Cmd], List[Type[Cmd]]]) -> None:
|
|
|
|
|
if isinstance(cmd, str):
|
|
|
|
|
sc = []
|
|
|
|
|
for name, obj in inspect.getmembers(sys.modules[self.__class__.__module__]):
|
|
|
|
|
if inspect.isclass(obj):
|
|
|
|
|
if re.search(cmd, str(obj)):
|
|
|
|
|
sc.append(obj)
|
|
|
|
|
log.slog(log.DEBUG, f"Found subcommand {obj}")
|
|
|
|
|
self.add_subcommands(sc)
|
|
|
|
|
return
|
2020-04-04 16:32:39 +02:00
|
|
|
if isinstance(cmd, list):
|
|
|
|
|
for c in cmd:
|
|
|
|
|
self.add_subcommands(c)
|
2020-04-07 09:23:13 +02:00
|
|
|
return
|
2020-04-04 16:32:39 +02:00
|
|
|
self.child_classes.append(cmd)
|
|
|
|
|
|
2025-10-13 12:45:51 +02:00
|
|
|
# To be overridden by derived class in case the command does take arguments.
|
2025-02-02 14:01:21 +01:00
|
|
|
# Will be called from App base class constructor and set up the parser hierarchy
|
2020-04-10 17:55:36 +02:00
|
|
|
def add_arguments(self, parser: ArgumentParser) -> None:
|
2020-04-04 16:32:39 +02:00
|
|
|
pass
|
2022-12-08 16:44:49 +01:00
|
|
|
|
|
|
|
|
def conf_value(self, path, default=None):
|
|
|
|
|
ret = None if self.app is None else self.app.conf_value(path, default)
|
|
|
|
|
if ret is None and default is not None:
|
|
|
|
|
return default
|
|
|
|
|
return ret
|