"""Tests for progressive subdirectory hint discovery.""" import os import pytest from pathlib import Path from agent.subdirectory_hints import SubdirectoryHintTracker @pytest.fixture def project(tmp_path): """Create a mock project tree with hint files in subdirectories.""" # Root — already loaded at startup (tmp_path / "AGENTS.md").write_text("Root project instructions") # backend/ — has its own AGENTS.md backend = tmp_path / "backend" backend.mkdir() (backend / "AGENTS.md").write_text("Backend-specific instructions:\n- Use FastAPI\n- Always add type hints") # backend/src/ — no hints (backend / "src").mkdir() (backend / "src" / "main.py").write_text("print('hello')") # frontend/ — has CLAUDE.md frontend = tmp_path / "frontend" frontend.mkdir() (frontend / "CLAUDE.md").write_text("Frontend rules:\n- Use TypeScript\n- No any types") # docs/ — no hints (tmp_path / "docs").mkdir() (tmp_path / "docs" / "README.md").write_text("Documentation") # deep/nested/path/ — has .cursorrules deep = tmp_path / "deep" / "nested" / "path" deep.mkdir(parents=True) (deep / ".cursorrules").write_text("Cursor rules for nested path") return tmp_path class TestSubdirectoryHintTracker: """Unit tests for SubdirectoryHintTracker.""" def test_working_dir_not_loaded(self, project): """Working dir is pre-marked as loaded (startup handles it).""" tracker = SubdirectoryHintTracker(working_dir=str(project)) # Reading a file in the root should NOT trigger hints result = tracker.check_tool_call("read_file", {"path": str(project / "AGENTS.md")}) assert result is None def test_discovers_agents_md_via_ancestor_walk(self, project): """Reading backend/src/main.py discovers backend/AGENTS.md via ancestor walk.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": str(project / "backend" / "src" / "main.py")} ) # backend/src/ has no hints, but ancestor walk finds backend/AGENTS.md assert result is not None assert "Backend-specific instructions" in result # Second read in same subtree should not re-trigger result2 = tracker.check_tool_call( "read_file", {"path": str(project / "backend" / "AGENTS.md")} ) assert result2 is None # backend/ already loaded def test_discovers_claude_md(self, project): """Frontend CLAUDE.md should be discovered.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": str(project / "frontend" / "index.ts")} ) assert result is not None assert "Frontend rules" in result def test_no_duplicate_loading(self, project): """Same directory should not be loaded twice.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result1 = tracker.check_tool_call( "read_file", {"path": str(project / "frontend" / "a.ts")} ) assert result1 is not None result2 = tracker.check_tool_call( "read_file", {"path": str(project / "frontend" / "b.ts")} ) assert result2 is None # already loaded def test_no_hints_in_empty_directory(self, project): """Directories without hint files return None.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": str(project / "docs" / "README.md")} ) assert result is None def test_terminal_command_path_extraction(self, project): """Paths extracted from terminal commands.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "terminal", {"command": f"cat {project / 'frontend' / 'index.ts'}"} ) assert result is not None assert "Frontend rules" in result def test_terminal_cd_command(self, project): """cd into a directory with hints.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "terminal", {"command": f"cd {project / 'backend'} && ls"} ) assert result is not None assert "Backend-specific instructions" in result def test_relative_path(self, project): """Relative paths resolved against working_dir.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": "frontend/index.ts"} ) assert result is not None assert "Frontend rules" in result def test_outside_working_dir_still_checked(self, tmp_path, project): """Paths outside working_dir are still checked for hints.""" other_project = tmp_path / "other" other_project.mkdir() (other_project / "AGENTS.md").write_text("Other project rules") tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": str(other_project / "file.py")} ) assert result is not None assert "Other project rules" in result def test_workdir_arg(self, project): """The workdir argument from terminal tool is checked.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "terminal", {"command": "ls", "workdir": str(project / "frontend")} ) assert result is not None assert "Frontend rules" in result def test_deeply_nested_cursorrules(self, project): """Deeply nested .cursorrules should be discovered.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": str(project / "deep" / "nested" / "path" / "file.py")} ) assert result is not None assert "Cursor rules for nested path" in result def test_hint_format_includes_path(self, project): """Discovered hints should indicate which file they came from.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "read_file", {"path": str(project / "backend" / "file.py")} ) assert result is not None assert "Subdirectory context discovered:" in result assert "AGENTS.md" in result def test_truncation_of_large_hints(self, tmp_path): """Hint files over the limit are truncated.""" sub = tmp_path / "bigdir" sub.mkdir() (sub / "AGENTS.md").write_text("x" * 20_000) tracker = SubdirectoryHintTracker(working_dir=str(tmp_path)) result = tracker.check_tool_call( "read_file", {"path": str(sub / "file.py")} ) assert result is not None assert "truncated" in result.lower() # Should be capped assert len(result) < 20_000 def test_empty_args(self, project): """Empty args should not crash.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) assert tracker.check_tool_call("read_file", {}) is None assert tracker.check_tool_call("terminal", {"command": ""}) is None def test_url_in_command_ignored(self, project): """URLs in shell commands should not be treated as paths.""" tracker = SubdirectoryHintTracker(working_dir=str(project)) result = tracker.check_tool_call( "terminal", {"command": "curl https://example.com/frontend/api"} ) assert result is None