- Add comprehensive module docstrings across the codebase for better clarity and documentation. - Refactor Enum classes (e.g., FileType, ConfirmType) to use `_missing_` for built-in coercion from strings. - Add `encoding` attribute to `LoadFileAction`, `SaveFileAction`, and `SelectFileAction` for more flexible file handling. - Enable lazy file loading by default in `SelectFileAction` to improve performance. - Simplify bottom bar toggle behavior: all toggles now use `ctrl+<key>`, eliminating the need for key conflict checks with Falyx commands. - Add `ignore_in_history` attribute to `Command` to refine how `ExecutionRegistry` identifies the last valid result. - Improve History command output: now includes tracebacks when displaying exceptions.
173 lines
6.2 KiB
Python
173 lines
6.2 KiB
Python
# Falyx CLI Framework — (c) 2025 rtj.dev LLC — MIT Licensed
|
|
"""
|
|
BaseIOAction: A base class for stream- or buffer-based IO-driven Actions.
|
|
|
|
This module defines `BaseIOAction`, a specialized variant of `BaseAction`
|
|
that interacts with standard input and output, enabling command-line pipelines,
|
|
text filters, and stream processing tasks.
|
|
|
|
Features:
|
|
- Supports buffered or streaming input modes.
|
|
- Reads from stdin and writes to stdout automatically.
|
|
- Integrates with lifecycle hooks and retry logic.
|
|
- Subclasses must implement `from_input()`, `to_output()`, and `_run()`.
|
|
|
|
Common usage includes shell-like filters, input transformers, or any tool that
|
|
needs to consume input from another process or pipeline.
|
|
"""
|
|
import asyncio
|
|
import sys
|
|
from typing import Any, Callable
|
|
|
|
from rich.tree import Tree
|
|
|
|
from falyx.action.base_action import BaseAction
|
|
from falyx.context import ExecutionContext
|
|
from falyx.execution_registry import ExecutionRegistry as er
|
|
from falyx.hook_manager import HookManager, HookType
|
|
from falyx.themes import OneColors
|
|
|
|
|
|
class BaseIOAction(BaseAction):
|
|
"""
|
|
Base class for IO-driven Actions that operate on stdin/stdout input streams.
|
|
|
|
Designed for use in shell pipelines or programmatic workflows that pass data
|
|
through chained commands. It handles reading input, transforming it, and
|
|
emitting output — either as a one-time buffered operation or line-by-line streaming.
|
|
|
|
Core responsibilities:
|
|
- Reads input from stdin or previous action result.
|
|
- Supports buffered or streaming modes via `mode`.
|
|
- Parses input via `from_input()` and formats output via `to_output()`.
|
|
- Executes `_run()` with the parsed input.
|
|
- Writes output to stdout.
|
|
|
|
Subclasses must implement:
|
|
- `from_input(raw)`: Convert raw stdin or injected data into typed input.
|
|
- `to_output(data)`: Convert result into output string or bytes.
|
|
- `_run(parsed_input, *args, **kwargs)`: Core execution logic.
|
|
|
|
Args:
|
|
name (str): Name of the action. Used for logging and debugging.
|
|
hooks (HookManager | None): Hook manager for lifecycle events.
|
|
mode (str): Either "buffered" or "stream". Controls input behavior.
|
|
logging_hooks (bool): Whether to register debug hooks for logging.
|
|
inject_last_result (bool): Whether to inject shared context input.
|
|
"""
|
|
|
|
def __init__(
|
|
self,
|
|
name: str,
|
|
*,
|
|
hooks: HookManager | None = None,
|
|
mode: str = "buffered",
|
|
logging_hooks: bool = True,
|
|
inject_last_result: bool = True,
|
|
):
|
|
super().__init__(
|
|
name=name,
|
|
hooks=hooks,
|
|
logging_hooks=logging_hooks,
|
|
inject_last_result=inject_last_result,
|
|
)
|
|
self.mode = mode
|
|
|
|
def from_input(self, raw: str | bytes) -> Any:
|
|
raise NotImplementedError
|
|
|
|
def to_output(self, result: Any) -> str | bytes:
|
|
raise NotImplementedError
|
|
|
|
async def _resolve_input(
|
|
self, args: tuple[Any], kwargs: dict[str, Any]
|
|
) -> str | bytes:
|
|
data = await self._read_stdin()
|
|
if data:
|
|
return self.from_input(data)
|
|
|
|
if len(args) == 1:
|
|
return self.from_input(args[0])
|
|
|
|
if self.inject_last_result and self.shared_context:
|
|
return self.shared_context.last_result()
|
|
|
|
return ""
|
|
|
|
def get_infer_target(self) -> tuple[Callable[..., Any] | None, dict[str, Any] | None]:
|
|
return None, None
|
|
|
|
async def __call__(self, *args, **kwargs):
|
|
context = ExecutionContext(
|
|
name=self.name,
|
|
args=args,
|
|
kwargs=kwargs,
|
|
action=self,
|
|
)
|
|
|
|
context.start_timer()
|
|
await self.hooks.trigger(HookType.BEFORE, context)
|
|
|
|
try:
|
|
if self.mode == "stream":
|
|
line_gen = await self._read_stdin_stream()
|
|
async for _ in self._stream_lines(line_gen, args, kwargs):
|
|
pass
|
|
result = getattr(self, "_last_result", None)
|
|
else:
|
|
parsed_input = await self._resolve_input(args, kwargs)
|
|
result = await self._run(parsed_input)
|
|
output = self.to_output(result)
|
|
await self._write_stdout(output)
|
|
context.result = result
|
|
await self.hooks.trigger(HookType.ON_SUCCESS, context)
|
|
return result
|
|
except Exception as error:
|
|
context.exception = error
|
|
await self.hooks.trigger(HookType.ON_ERROR, context)
|
|
raise
|
|
finally:
|
|
context.stop_timer()
|
|
await self.hooks.trigger(HookType.AFTER, context)
|
|
await self.hooks.trigger(HookType.ON_TEARDOWN, context)
|
|
er.record(context)
|
|
|
|
async def _read_stdin(self) -> str:
|
|
if not sys.stdin.isatty():
|
|
return await asyncio.to_thread(sys.stdin.read)
|
|
return ""
|
|
|
|
async def _read_stdin_stream(self) -> Any:
|
|
"""Returns a generator that yields lines from stdin in a background thread."""
|
|
loop = asyncio.get_running_loop()
|
|
return await loop.run_in_executor(None, lambda: iter(sys.stdin))
|
|
|
|
async def _stream_lines(self, line_gen, args, kwargs):
|
|
for line in line_gen:
|
|
parsed = self.from_input(line)
|
|
result = await self._run(parsed, *args, **kwargs)
|
|
self._last_result = result
|
|
output = self.to_output(result)
|
|
await self._write_stdout(output)
|
|
yield result
|
|
|
|
async def _write_stdout(self, data: str) -> None:
|
|
await asyncio.to_thread(sys.stdout.write, data)
|
|
await asyncio.to_thread(sys.stdout.flush)
|
|
|
|
async def _run(self, parsed_input: Any, *args, **kwargs) -> Any:
|
|
"""Subclasses should override this with actual logic."""
|
|
raise NotImplementedError("Must implement _run()")
|
|
|
|
def __str__(self):
|
|
return f"<{self.__class__.__name__} '{self.name}' IOAction>"
|
|
|
|
async def preview(self, parent: Tree | None = None):
|
|
label = [f"[{OneColors.GREEN_b}]⚙ IOAction[/] '{self.name}'"]
|
|
if self.inject_last_result:
|
|
label.append(f" [dim](injects '{self.inject_into}')[/dim]")
|
|
if parent:
|
|
parent.add("".join(label))
|
|
else:
|
|
self.console.print(Tree("".join(label)))
|