97adfbda45
Rename workflow files and their 'name' field with a 'media.' domain prefix to anticipate future multi-domain expansion (mail.*, calendar.*, ...). - organize_media -> media.organize_media - manage_subtitles -> media.manage_subtitles WorkflowLoader picks them up unchanged (uses data['name']).
53 lines
1.6 KiB
Python
53 lines
1.6 KiB
Python
"""WorkflowLoader — autodiscovers and loads workflow YAML files.
|
|
|
|
Scans the workflows/ directory for all .yaml files and exposes them
|
|
as dicts. No manual registration needed — drop a new .yaml file and
|
|
it will be picked up automatically.
|
|
"""
|
|
|
|
import logging
|
|
from pathlib import Path
|
|
|
|
import yaml
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
_WORKFLOWS_DIR = Path(__file__).parent
|
|
|
|
|
|
class WorkflowLoader:
|
|
"""
|
|
Loads all workflow definitions from the workflows/ directory.
|
|
|
|
Usage:
|
|
loader = WorkflowLoader()
|
|
all_workflows = loader.all()
|
|
workflow = loader.get("media.organize_media")
|
|
"""
|
|
|
|
def __init__(self):
|
|
self._workflows: dict[str, dict] = {}
|
|
self._load()
|
|
|
|
def _load(self) -> None:
|
|
for path in sorted(_WORKFLOWS_DIR.glob("*.yaml")):
|
|
try:
|
|
data = yaml.safe_load(path.read_text(encoding="utf-8"))
|
|
name = data.get("name") or path.stem
|
|
self._workflows[name] = data
|
|
logger.info(f"WorkflowLoader: Loaded '{name}' from {path.name}")
|
|
except Exception as e:
|
|
logger.warning(f"WorkflowLoader: Could not load {path.name}: {e}")
|
|
|
|
def all(self) -> dict[str, dict]:
|
|
"""Return all loaded workflows keyed by name."""
|
|
return self._workflows
|
|
|
|
def get(self, name: str) -> dict | None:
|
|
"""Return a specific workflow by name, or None if not found."""
|
|
return self._workflows.get(name)
|
|
|
|
def names(self) -> list[str]:
|
|
"""Return all available workflow names."""
|
|
return list(self._workflows.keys())
|