#!/usr/bin/env python3 # -*- coding: utf-8 -*- """ Session Start Hook - Inject structured context """ # IMPORTANT: Suppress all warnings FIRST import warnings warnings.filterwarnings("ignore") import json import os import subprocess import sys from io import StringIO from pathlib import Path # IMPORTANT: Force stdout to use UTF-8 on Windows # This fixes UnicodeEncodeError when outputting non-ASCII characters if sys.platform == "win32": import io as _io if hasattr(sys.stdout, "reconfigure"): sys.stdout.reconfigure(encoding="utf-8", errors="replace") # type: ignore[union-attr] elif hasattr(sys.stdout, "detach"): sys.stdout = _io.TextIOWrapper(sys.stdout.detach(), encoding="utf-8", errors="replace") # type: ignore[union-attr] def should_skip_injection() -> bool: return ( os.environ.get("CLAUDE_NON_INTERACTIVE") == "1" or os.environ.get("OPENCODE_NON_INTERACTIVE") == "1" ) def read_file(path: Path, fallback: str = "") -> str: try: return path.read_text(encoding="utf-8") except (FileNotFoundError, PermissionError): return fallback def run_script(script_path: Path) -> str: try: if script_path.suffix == ".py": # Add PYTHONIOENCODING to force UTF-8 in subprocess env = os.environ.copy() env["PYTHONIOENCODING"] = "utf-8" cmd = [sys.executable, "-W", "ignore", str(script_path)] else: env = os.environ cmd = [str(script_path)] result = subprocess.run( cmd, capture_output=True, text=True, encoding="utf-8", errors="replace", timeout=5, cwd=script_path.parent.parent.parent, env=env, ) return result.stdout if result.returncode == 0 else "No context available" except (subprocess.TimeoutExpired, FileNotFoundError, PermissionError): return "No context available" def _get_task_status(trellis_dir: Path) -> str: """Check current task status and return structured status string.""" current_task_file = trellis_dir / ".current-task" if not current_task_file.is_file(): return "Status: NO ACTIVE TASK\nNext: Describe what you want to work on" task_ref = current_task_file.read_text(encoding="utf-8").strip() if not task_ref: return "Status: NO ACTIVE TASK\nNext: Describe what you want to work on" # Resolve task directory if Path(task_ref).is_absolute(): task_dir = Path(task_ref) elif task_ref.startswith(".trellis/"): task_dir = trellis_dir.parent / task_ref else: task_dir = trellis_dir / "tasks" / task_ref if not task_dir.is_dir(): return f"Status: STALE POINTER\nTask: {task_ref}\nNext: Task directory not found. Run: python3 ./.trellis/scripts/task.py finish" # Read task.json task_json_path = task_dir / "task.json" task_data = {} if task_json_path.is_file(): try: task_data = json.loads(task_json_path.read_text(encoding="utf-8")) except (json.JSONDecodeError, PermissionError): pass task_title = task_data.get("title", task_ref) task_status = task_data.get("status", "unknown") if task_status == "completed": return f"Status: COMPLETED\nTask: {task_title}\nNext: Archive with `python3 ./.trellis/scripts/task.py archive {task_dir.name}` or start a new task" # Check if context is configured (jsonl files exist and non-empty) has_context = False for jsonl_name in ("implement.jsonl", "check.jsonl", "spec.jsonl"): jsonl_path = task_dir / jsonl_name if jsonl_path.is_file() and jsonl_path.stat().st_size > 0: has_context = True break has_prd = (task_dir / "prd.md").is_file() if not has_prd: return f"Status: NOT READY\nTask: {task_title}\nMissing: prd.md not created\nNext: Write PRD, then research → init-context → start" if not has_context: return f"Status: NOT READY\nTask: {task_title}\nMissing: Context not configured (no jsonl files)\nNext: Complete Phase 2 (research → init-context → start) before implementing" return f"Status: READY\nTask: {task_title}\nNext: Continue with implement or check" def main(): if should_skip_injection(): sys.exit(0) project_dir = Path(os.environ.get("CLAUDE_PROJECT_DIR", ".")).resolve() trellis_dir = project_dir / ".trellis" claude_dir = project_dir / ".claude" output = StringIO() output.write(""" You are starting a new session in a Trellis-managed project. Read and follow all instructions below carefully. """) output.write("\n") context_script = trellis_dir / "scripts" / "get_context.py" output.write(run_script(context_script)) output.write("\n\n\n") output.write("\n") workflow_content = read_file(trellis_dir / "workflow.md", "No workflow.md found") output.write(workflow_content) output.write("\n\n\n") output.write("\n") output.write("**Note**: The guidelines below are index files — they list available guideline documents and their locations.\n") output.write("During actual development, you MUST read the specific guideline files listed in each index's Pre-Development Checklist.\n\n") spec_dir = trellis_dir / "spec" if spec_dir.is_dir(): for sub in sorted(spec_dir.iterdir()): if not sub.is_dir() or sub.name.startswith("."): continue index_file = sub / "index.md" if index_file.is_file(): output.write(f"## {sub.name}\n") output.write(read_file(index_file)) output.write("\n\n") else: # Check for nested package dirs (monorepo: spec///index.md) for nested in sorted(sub.iterdir()): if not nested.is_dir(): continue nested_index = nested / "index.md" if nested_index.is_file(): output.write(f"## {sub.name}/{nested.name}\n") output.write(read_file(nested_index)) output.write("\n\n") output.write("\n\n") output.write("\n") start_md = read_file( claude_dir / "commands" / "trellis" / "start.md", "No start.md found" ) output.write(start_md) output.write("\n\n\n") # R2: Check task status and inject structured tag task_status = _get_task_status(trellis_dir) output.write(f"\n{task_status}\n\n\n") output.write(""" Context loaded. Steps 1-3 (workflow, context, guidelines) are already injected above — do NOT re-read them. Start from Step 4. Wait for user's first message, then follow to handle their request. If there is an active task, ask whether to continue it. """) result = { "hookSpecificOutput": { "hookEventName": "SessionStart", "additionalContext": output.getvalue(), } } # Output JSON - stdout is already configured for UTF-8 print(json.dumps(result, ensure_ascii=False), flush=True) if __name__ == "__main__": main()