239 lines
7.7 KiB
Python
239 lines
7.7 KiB
Python
#!/usr/bin/env python3
|
|
"""
|
|
Package discovery and context output.
|
|
|
|
Provides:
|
|
get_packages_info - Get structured package info
|
|
get_packages_section - Build PACKAGES text section
|
|
get_context_packages_text - Full packages text output (--mode packages)
|
|
get_context_packages_json - Full packages JSON output (--mode packages --json)
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from pathlib import Path
|
|
|
|
from .config import _is_true_config_value, get_default_package, get_packages, get_spec_scope
|
|
from .paths import (
|
|
DIR_SPEC,
|
|
DIR_WORKFLOW,
|
|
get_current_task,
|
|
get_repo_root,
|
|
)
|
|
from .tasks import load_task
|
|
|
|
|
|
# =============================================================================
|
|
# Internal Helpers
|
|
# =============================================================================
|
|
|
|
def _scan_spec_layers(spec_dir: Path, package: str | None = None) -> list[str]:
|
|
"""Scan spec directory for available layers (subdirectories).
|
|
|
|
For monorepo: scans spec/<package>/
|
|
For single-repo: scans spec/
|
|
"""
|
|
target = spec_dir / package if package else spec_dir
|
|
if not target.is_dir():
|
|
return []
|
|
return sorted(
|
|
d.name for d in target.iterdir() if d.is_dir() and d.name != "guides"
|
|
)
|
|
|
|
|
|
def _get_active_task_package(repo_root: Path) -> str | None:
|
|
"""Get the package field from the active task's task.json."""
|
|
current = get_current_task(repo_root)
|
|
if not current:
|
|
return None
|
|
ct = load_task(repo_root / current)
|
|
return ct.package if ct and ct.package else None
|
|
|
|
|
|
def _resolve_scope_set(
|
|
packages: dict,
|
|
spec_scope,
|
|
task_pkg: str | None,
|
|
default_pkg: str | None,
|
|
) -> set | None:
|
|
"""Resolve spec_scope to a set of allowed package names, or None for full scan."""
|
|
if not packages:
|
|
return None
|
|
|
|
if spec_scope is None:
|
|
return None
|
|
|
|
if isinstance(spec_scope, str) and spec_scope == "active_task":
|
|
if task_pkg and task_pkg in packages:
|
|
return {task_pkg}
|
|
if default_pkg and default_pkg in packages:
|
|
return {default_pkg}
|
|
return None
|
|
|
|
if isinstance(spec_scope, list):
|
|
valid = {e for e in spec_scope if e in packages}
|
|
if valid:
|
|
return valid
|
|
# All invalid: fallback
|
|
if task_pkg and task_pkg in packages:
|
|
return {task_pkg}
|
|
if default_pkg and default_pkg in packages:
|
|
return {default_pkg}
|
|
return None
|
|
|
|
return None
|
|
|
|
|
|
# =============================================================================
|
|
# Public Functions
|
|
# =============================================================================
|
|
|
|
def get_packages_info(repo_root: Path) -> list[dict]:
|
|
"""Get structured package info for monorepo projects.
|
|
|
|
Returns list of dicts with keys: name, path, type, default, specLayers,
|
|
isSubmodule, isGitRepo.
|
|
Returns empty list for single-repo projects.
|
|
"""
|
|
packages = get_packages(repo_root)
|
|
if not packages:
|
|
return []
|
|
|
|
default_pkg = get_default_package(repo_root)
|
|
spec_dir = repo_root / DIR_WORKFLOW / DIR_SPEC
|
|
result = []
|
|
|
|
for pkg_name, pkg_config in packages.items():
|
|
pkg_path = pkg_config.get("path", pkg_name) if isinstance(pkg_config, dict) else str(pkg_config)
|
|
pkg_type = pkg_config.get("type", "local") if isinstance(pkg_config, dict) else "local"
|
|
pkg_git = pkg_config.get("git", False) if isinstance(pkg_config, dict) else False
|
|
layers = _scan_spec_layers(spec_dir, pkg_name)
|
|
|
|
result.append({
|
|
"name": pkg_name,
|
|
"path": pkg_path,
|
|
"type": pkg_type,
|
|
"default": pkg_name == default_pkg,
|
|
"specLayers": layers,
|
|
"isSubmodule": pkg_type == "submodule",
|
|
"isGitRepo": _is_true_config_value(pkg_git),
|
|
})
|
|
|
|
return result
|
|
|
|
|
|
def get_packages_section(repo_root: Path) -> str:
|
|
"""Build the PACKAGES section for text output."""
|
|
spec_dir = repo_root / DIR_WORKFLOW / DIR_SPEC
|
|
pkg_info = get_packages_info(repo_root)
|
|
|
|
lines: list[str] = []
|
|
lines.append("## PACKAGES")
|
|
|
|
if not pkg_info:
|
|
lines.append("(single-repo mode)")
|
|
layers = _scan_spec_layers(spec_dir)
|
|
if layers:
|
|
lines.append(f"Spec layers: {', '.join(layers)}")
|
|
return "\n".join(lines)
|
|
|
|
default_pkg = get_default_package(repo_root)
|
|
|
|
for pkg in pkg_info:
|
|
layers_str = f" [{', '.join(pkg['specLayers'])}]" if pkg["specLayers"] else ""
|
|
submodule_tag = " (submodule)" if pkg["isSubmodule"] else ""
|
|
git_repo_tag = " (git repo)" if pkg["isGitRepo"] else ""
|
|
default_tag = " *" if pkg["default"] else ""
|
|
lines.append(
|
|
f"- {pkg['name']:<16} {pkg['path']:<20}{layers_str}{submodule_tag}{git_repo_tag}{default_tag}"
|
|
)
|
|
|
|
if default_pkg:
|
|
lines.append(f"Default package: {default_pkg}")
|
|
|
|
return "\n".join(lines)
|
|
|
|
|
|
def get_context_packages_text(repo_root: Path | None = None) -> str:
|
|
"""Get packages context as formatted text (for --mode packages)."""
|
|
if repo_root is None:
|
|
repo_root = get_repo_root()
|
|
|
|
pkg_info = get_packages_info(repo_root)
|
|
lines: list[str] = []
|
|
|
|
if not pkg_info:
|
|
spec_dir = repo_root / DIR_WORKFLOW / DIR_SPEC
|
|
lines.append("Single-repo project (no packages configured)")
|
|
lines.append("")
|
|
layers = _scan_spec_layers(spec_dir)
|
|
if layers:
|
|
lines.append(f"Spec layers: {', '.join(layers)}")
|
|
return "\n".join(lines)
|
|
|
|
# Resolve scope for annotations
|
|
packages_dict = get_packages(repo_root) or {}
|
|
default_pkg = get_default_package(repo_root)
|
|
spec_scope = get_spec_scope(repo_root)
|
|
task_pkg = _get_active_task_package(repo_root)
|
|
scope_set = _resolve_scope_set(packages_dict, spec_scope, task_pkg, default_pkg)
|
|
|
|
lines.append("## PACKAGES")
|
|
lines.append("")
|
|
for pkg in pkg_info:
|
|
default_tag = " (default)" if pkg["default"] else ""
|
|
type_tag = f" [{pkg['type']}]" if pkg["type"] != "local" else ""
|
|
git_tag = " [git repo]" if pkg["isGitRepo"] else ""
|
|
|
|
# Scope annotation
|
|
scope_tag = ""
|
|
if scope_set is not None and pkg["name"] not in scope_set:
|
|
scope_tag = " (out of scope)"
|
|
|
|
lines.append(f"### {pkg['name']}{default_tag}{type_tag}{git_tag}{scope_tag}")
|
|
lines.append(f"Path: {pkg['path']}")
|
|
if pkg["specLayers"]:
|
|
lines.append(f"Spec layers: {', '.join(pkg['specLayers'])}")
|
|
for layer in pkg["specLayers"]:
|
|
lines.append(f" - .trellis/spec/{pkg['name']}/{layer}/index.md")
|
|
else:
|
|
lines.append("Spec: not configured")
|
|
lines.append("")
|
|
|
|
# Also show shared guides
|
|
guides_dir = repo_root / DIR_WORKFLOW / DIR_SPEC / "guides"
|
|
if guides_dir.is_dir():
|
|
lines.append("### Shared Guides (always included)")
|
|
lines.append("Path: .trellis/spec/guides/index.md")
|
|
lines.append("")
|
|
|
|
return "\n".join(lines)
|
|
|
|
|
|
def get_context_packages_json(repo_root: Path | None = None) -> dict:
|
|
"""Get packages context as a dictionary (for --mode packages --json)."""
|
|
if repo_root is None:
|
|
repo_root = get_repo_root()
|
|
|
|
pkg_info = get_packages_info(repo_root)
|
|
|
|
if not pkg_info:
|
|
spec_dir = repo_root / DIR_WORKFLOW / DIR_SPEC
|
|
layers = _scan_spec_layers(spec_dir)
|
|
return {
|
|
"mode": "single-repo",
|
|
"specLayers": layers,
|
|
}
|
|
|
|
default_pkg = get_default_package(repo_root)
|
|
spec_scope = get_spec_scope(repo_root)
|
|
task_pkg = _get_active_task_package(repo_root)
|
|
|
|
return {
|
|
"mode": "monorepo",
|
|
"packages": pkg_info,
|
|
"defaultPackage": default_pkg,
|
|
"specScope": spec_scope,
|
|
"activeTaskPackage": task_pkg,
|
|
}
|