mirror of
https://github.com/daylinmorgan/swydd.git
synced 2025-01-04 19:20:44 -06:00
refactor!: allow native datatypes to be strings
This commit is contained in:
parent
d7fa879eb9
commit
83f94cd4c2
1 changed files with 83 additions and 42 deletions
|
@ -12,7 +12,6 @@ from argparse import (
|
|||
RawDescriptionHelpFormatter,
|
||||
)
|
||||
from functools import wraps
|
||||
from inspect import Parameter
|
||||
from pathlib import Path
|
||||
from subprocess import PIPE, Popen
|
||||
from typing import TYPE_CHECKING
|
||||
|
@ -81,6 +80,16 @@ class Task:
|
|||
def _mark(self) -> None:
|
||||
self.show = True
|
||||
|
||||
def _task_repr(self) -> str:
|
||||
return (
|
||||
"\n".join(
|
||||
f" {line}"
|
||||
for line in inspect.getsource(self.func).splitlines()
|
||||
if not line.startswith("@")
|
||||
)
|
||||
+ "\n"
|
||||
)
|
||||
|
||||
|
||||
class Graph:
|
||||
def __init__(self) -> None:
|
||||
|
@ -105,8 +114,6 @@ class Context:
|
|||
self._ids: Dict[str, str] = {}
|
||||
self.targets: Dict[str, Any] = {}
|
||||
self.data: Any = None
|
||||
self.flags: Dict[str, Any] = {}
|
||||
self._flag_defs: List[Tuple[Tuple[str, ...], Any]] = []
|
||||
self.show_targets = True
|
||||
self._graph = Graph()
|
||||
self.rest = [] # remaining positional args
|
||||
|
@ -154,13 +161,35 @@ class Context:
|
|||
assert name in self._ids
|
||||
return self._tasks[self._ids[name]]
|
||||
|
||||
def add_flag(self, *args: str, **kwargs: Any) -> None:
|
||||
|
||||
class SwyddFlags:
|
||||
def __init__(self) -> None:
|
||||
self._flags: Dict[str, Any] = {}
|
||||
self._flag_defs: List[Tuple[Tuple[str, ...], Any]] = []
|
||||
|
||||
def __getattr__(self, name: str):
|
||||
if name in self._flags:
|
||||
return self._flags[name]
|
||||
else:
|
||||
# TODO: better error/sys.exit
|
||||
raise AttributeError
|
||||
|
||||
def add(self, *args: str, **kwargs: Any) -> None:
|
||||
name = max(args, key=len).split("-")[-1]
|
||||
self.flags[name] = None
|
||||
self._flags[name] = None
|
||||
self._flag_defs.append((args, kwargs))
|
||||
|
||||
def _parse_args(self, args):
|
||||
for name in self._flags:
|
||||
self._flags[name] = args.pop(name)
|
||||
|
||||
def _add_arguments(self, shared):
|
||||
for flag_args, flag_kwargs in self._flag_defs:
|
||||
shared.add_argument(*flag_args, **flag_kwargs)
|
||||
|
||||
|
||||
ctx = Context()
|
||||
flags = SwyddFlags()
|
||||
|
||||
|
||||
class SwyddSubResult:
|
||||
|
@ -230,7 +259,7 @@ class SwyddProc:
|
|||
|
||||
if self.output:
|
||||
sub_kwargs["text"] = True # assume text is the desired output
|
||||
sub_kwargs["capture_output"] = True
|
||||
sub_kwargs["stdout"] = PIPE
|
||||
|
||||
sub_kwargs.update(**self.cmd_kwargs)
|
||||
return sub_kwargs
|
||||
|
@ -512,11 +541,17 @@ def _inspect_wrapper(place, func):
|
|||
|
||||
|
||||
def task(
|
||||
arg=None,
|
||||
arg: Callable[..., Any] | None = None,
|
||||
):
|
||||
def wrapper(func: Callable[..., Any]) -> Callable[..., Callable[..., None]]:
|
||||
def wrapper(
|
||||
func: Callable[..., Any] | None = None,
|
||||
) -> Callable[..., Callable[..., None]]:
|
||||
# added this to prevent lsp errors
|
||||
# about missing argument on wrapped functions
|
||||
if func is None:
|
||||
raise ValueError
|
||||
|
||||
ctx._add_task(func, show=not func.__name__.startswith("_"))
|
||||
# _inspect_wrapper("task", func)
|
||||
|
||||
@wraps(func)
|
||||
def inner(*args: Any, **kwargs: Any) -> Callable[..., None]:
|
||||
|
@ -581,12 +616,6 @@ def option(
|
|||
return wrapper
|
||||
|
||||
|
||||
def manage(version: bool = False) -> None:
|
||||
"""internal cli"""
|
||||
if version:
|
||||
print("current version", __version__)
|
||||
|
||||
|
||||
def noop(*args, **kwargs) -> Any:
|
||||
_ = args, kwargs
|
||||
|
||||
|
@ -617,7 +646,27 @@ def _target_generator(
|
|||
return wrapper
|
||||
|
||||
|
||||
# TODO: reduce how the load bearing on this function
|
||||
def _internal_cli():
|
||||
name = "+swydd"
|
||||
parser = ArgumentParser(
|
||||
usage=f"%(prog)s {name} [opts]",
|
||||
description="""internal cli""",
|
||||
prog=os.path.basename(sys.argv[0]),
|
||||
)
|
||||
parser.add_argument("-V", "--version", help="show version", action="store_true")
|
||||
parser.add_argument("--which", help="show path to swydd", action="store_true")
|
||||
|
||||
args = parser.parse_args(sys.argv[2:])
|
||||
|
||||
if args.version:
|
||||
print("swydd, v" + __version__)
|
||||
if args.which:
|
||||
print("current swydd:", __file__)
|
||||
|
||||
sys.exit(0)
|
||||
|
||||
|
||||
# TODO: reduce the load bearing on this function
|
||||
# seperate by subparser for tasks vs subparser for target
|
||||
def _generate_task_subparser(
|
||||
shared: ArgumentParser,
|
||||
|
@ -642,21 +691,28 @@ def _generate_task_subparser(
|
|||
usage=f"%(prog)s {name} [opts]",
|
||||
prog=prog,
|
||||
)
|
||||
|
||||
for name, info in task.params.items():
|
||||
param = info.get("Parameter") # must check signature for args?
|
||||
annotation = param.annotation
|
||||
|
||||
args = []
|
||||
if "short" in info:
|
||||
args.append("-" + info["short"])
|
||||
args.append(f"--{name.replace('_','-')}")
|
||||
kwargs = {"help": info.get("help", "")}
|
||||
|
||||
if param.annotation is bool:
|
||||
str_lit_types = {"bool": bool, "int": int, "float": float, "Path": Path}
|
||||
annotation = str_lit_types.get(annotation, annotation)
|
||||
|
||||
if annotation is bool:
|
||||
kwargs.update({"default": False, "action": "store_true"})
|
||||
elif param.annotation != Parameter.empty:
|
||||
kwargs.update({"type": param.annotation})
|
||||
elif (annotation != inspect._empty) and (type(annotations) is not str):
|
||||
kwargs.update({"type": annotation})
|
||||
|
||||
kwargs.update(
|
||||
{"required": True}
|
||||
if param.default == Parameter.empty
|
||||
if param.default == inspect._empty
|
||||
else {"default": param.default}
|
||||
)
|
||||
|
||||
|
@ -692,7 +748,7 @@ def _target_status(target: str) -> str:
|
|||
for need in needs:
|
||||
if not (p := Path(need)).is_file():
|
||||
return "missing inputs!"
|
||||
needs_stats.append(p)
|
||||
needs_stats.append(p.stat())
|
||||
|
||||
if any((stat.st_mtime > target_stat.st_mtime for stat in needs_stats)):
|
||||
return "out of date"
|
||||
|
@ -714,27 +770,17 @@ def _add_targets(
|
|||
)
|
||||
|
||||
|
||||
def _task_repr(func: Callable) -> str:
|
||||
return (
|
||||
"\n".join(
|
||||
f" {line}"
|
||||
for line in inspect.getsource(func).splitlines()
|
||||
if not line.startswith("@")
|
||||
)
|
||||
+ "\n"
|
||||
)
|
||||
|
||||
|
||||
def cli(default: str | None = None) -> None:
|
||||
ctx._generate_graph()
|
||||
if len(sys.argv) > 1 and sys.argv[1] == "+swydd":
|
||||
_internal_cli()
|
||||
|
||||
parser = ArgumentParser(
|
||||
formatter_class=SubcommandHelpFormatter, usage="%(prog)s <task/target> [opts]"
|
||||
)
|
||||
shared = ArgumentParser(add_help=False)
|
||||
|
||||
for flag_args, flag_kwargs in ctx._flag_defs:
|
||||
shared.add_argument(*flag_args, **flag_kwargs)
|
||||
flags._add_arguments(shared)
|
||||
|
||||
shared.add_argument(
|
||||
"-v", "--verbose", help="use verbose output", action="store_true"
|
||||
|
@ -747,11 +793,6 @@ def cli(default: str | None = None) -> None:
|
|||
title="tasks", required=True, dest="pos-arg", metavar="<task/target>"
|
||||
)
|
||||
|
||||
if len(sys.argv) > 1 and sys.argv[1] == "+swydd":
|
||||
_generate_task_subparser(
|
||||
shared, subparsers, Task(manage, name="+swydd", show=True)
|
||||
)
|
||||
|
||||
_add_targets(shared, subparsers, ctx)
|
||||
|
||||
for task in ctx._tasks.values():
|
||||
|
@ -776,15 +817,14 @@ def cli(default: str | None = None) -> None:
|
|||
ctx.dry = args.pop("dry_run", False)
|
||||
ctx.dag = args.pop("dag", False)
|
||||
ctx.force = args.pop("force", False)
|
||||
for name in ctx.flags:
|
||||
ctx.flags[name] = args.pop(name)
|
||||
flags._parse_args(args)
|
||||
|
||||
if f := args.pop("func", None):
|
||||
if ctx.dry:
|
||||
sys.stderr.write("dry run >>>\n" f" args: {args}\n")
|
||||
if ctx._env:
|
||||
sys.stderr.write(f" env: {ctx._env}\n")
|
||||
sys.stderr.write(_task_repr(ctx._get_task(pos_arg).func))
|
||||
sys.stderr.write(ctx._get_task(pos_arg)._task_repr())
|
||||
elif ctx.dag:
|
||||
sys.stderr.write(
|
||||
"currently --dag is a noop\n"
|
||||
|
@ -833,6 +873,7 @@ __all__ = [
|
|||
"setenv",
|
||||
"cli",
|
||||
"task",
|
||||
"flags",
|
||||
]
|
||||
|
||||
if __name__ == "__main__":
|
||||
|
|
Loading…
Reference in a new issue