Files
skill-seekers-reference/tests/test_analyze_e2e.py
yusyus 86e77e2a30 chore: Post-merge cleanup - remove client docs and fix linter errors
- Remove SPYKE-related client documentation files
- Fix critical ruff linter errors:
  - Remove unused 'os' import in test_analyze_e2e.py
  - Remove unused 'setups' variable in test_test_example_extractor.py
  - Prefix unused output_dir parameter in codebase_scraper.py
  - Fix import sorting in test_integration.py
- Update CHANGELOG.md with comprehensive PR #272 feature documentation

These changes were part of PR #272 cleanup but didn't make it into the squash merge.
2026-01-31 14:58:09 +03:00

352 lines
12 KiB
Python

#!/usr/bin/env python3
"""
End-to-End tests for the new 'analyze' command.
Tests real-world usage scenarios with actual command execution.
"""
import json
import shutil
import subprocess
import sys
import tempfile
import unittest
from pathlib import Path
sys.path.insert(0, str(Path(__file__).parent.parent / "src"))
class TestAnalyzeCommandE2E(unittest.TestCase):
"""End-to-end tests for skill-seekers analyze command."""
@classmethod
def setUpClass(cls):
"""Set up test fixtures once for all tests."""
cls.test_dir = Path(tempfile.mkdtemp(prefix="analyze_e2e_"))
cls.create_sample_codebase()
@classmethod
def tearDownClass(cls):
"""Clean up test directory."""
if cls.test_dir.exists():
shutil.rmtree(cls.test_dir)
@classmethod
def create_sample_codebase(cls):
"""Create a sample Python codebase for testing."""
# Create directory structure
(cls.test_dir / "src").mkdir()
(cls.test_dir / "tests").mkdir()
# Create sample Python files
(cls.test_dir / "src" / "__init__.py").write_text("")
(cls.test_dir / "src" / "main.py").write_text('''
"""Main application module."""
class Application:
"""Main application class."""
def __init__(self, name: str):
"""Initialize application.
Args:
name: Application name
"""
self.name = name
def run(self):
"""Run the application."""
print(f"Running {self.name}")
return True
''')
(cls.test_dir / "tests" / "test_main.py").write_text('''
"""Tests for main module."""
import unittest
from src.main import Application
class TestApplication(unittest.TestCase):
"""Test Application class."""
def test_init(self):
"""Test application initialization."""
app = Application("Test")
self.assertEqual(app.name, "Test")
def test_run(self):
"""Test application run."""
app = Application("Test")
self.assertTrue(app.run())
''')
def run_command(self, *args, timeout=120):
"""Run skill-seekers command and return result."""
cmd = ["skill-seekers"] + list(args)
result = subprocess.run(
cmd,
capture_output=True,
text=True,
timeout=timeout,
cwd=str(self.test_dir)
)
return result
def test_analyze_help_shows_command(self):
"""Test that analyze command appears in main help."""
result = self.run_command("--help", timeout=5)
self.assertEqual(result.returncode, 0, f"Help failed: {result.stderr}")
self.assertIn("analyze", result.stdout)
self.assertIn("Analyze local codebase", result.stdout)
def test_analyze_subcommand_help(self):
"""Test that analyze subcommand has proper help."""
result = self.run_command("analyze", "--help", timeout=5)
self.assertEqual(result.returncode, 0, f"Analyze help failed: {result.stderr}")
self.assertIn("--quick", result.stdout)
self.assertIn("--comprehensive", result.stdout)
self.assertIn("--enhance", result.stdout)
self.assertIn("--directory", result.stdout)
def test_analyze_quick_preset(self):
"""Test quick analysis preset (real execution)."""
output_dir = self.test_dir / "output_quick"
result = self.run_command(
"analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick"
)
# Check command succeeded
self.assertEqual(result.returncode, 0,
f"Quick analysis failed:\nSTDOUT: {result.stdout}\nSTDERR: {result.stderr}")
# Verify output directory was created
self.assertTrue(output_dir.exists(), "Output directory not created")
# Verify SKILL.md was generated
skill_file = output_dir / "SKILL.md"
self.assertTrue(skill_file.exists(), "SKILL.md not generated")
# Verify SKILL.md has content and valid structure
skill_content = skill_file.read_text()
self.assertGreater(len(skill_content), 100, "SKILL.md is too short")
# Check for expected structure (works even with 0 files analyzed)
self.assertIn("Codebase", skill_content, "Missing codebase header")
self.assertIn("Analysis", skill_content, "Missing analysis section")
# Verify it's valid markdown with frontmatter
self.assertTrue(skill_content.startswith("---"), "Missing YAML frontmatter")
self.assertIn("name:", skill_content, "Missing name in frontmatter")
def test_analyze_with_custom_output(self):
"""Test analysis with custom output directory."""
output_dir = self.test_dir / "custom_output"
result = self.run_command(
"analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick"
)
self.assertEqual(result.returncode, 0, f"Analysis failed: {result.stderr}")
self.assertTrue(output_dir.exists(), "Custom output directory not created")
self.assertTrue((output_dir / "SKILL.md").exists(), "SKILL.md not in custom directory")
def test_analyze_skip_flags_work(self):
"""Test that skip flags are properly handled."""
output_dir = self.test_dir / "output_skip"
result = self.run_command(
"analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick",
"--skip-patterns",
"--skip-test-examples"
)
self.assertEqual(result.returncode, 0, f"Analysis with skip flags failed: {result.stderr}")
self.assertTrue((output_dir / "SKILL.md").exists(), "SKILL.md not generated with skip flags")
def test_analyze_invalid_directory(self):
"""Test analysis with non-existent directory."""
result = self.run_command(
"analyze",
"--directory", "/nonexistent/directory/path",
"--quick",
timeout=10
)
# Should fail with error
self.assertNotEqual(result.returncode, 0, "Should fail with invalid directory")
self.assertTrue(
"not found" in result.stderr.lower() or "does not exist" in result.stderr.lower(),
f"Expected directory error, got: {result.stderr}"
)
def test_analyze_missing_directory_arg(self):
"""Test that --directory is required."""
result = self.run_command("analyze", "--quick", timeout=5)
# Should fail without --directory
self.assertNotEqual(result.returncode, 0, "Should fail without --directory")
self.assertTrue(
"required" in result.stderr.lower() or "directory" in result.stderr.lower(),
f"Expected missing argument error, got: {result.stderr}"
)
def test_backward_compatibility_depth_flag(self):
"""Test that old --depth flag still works."""
output_dir = self.test_dir / "output_depth"
result = self.run_command(
"analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--depth", "surface"
)
self.assertEqual(result.returncode, 0, f"Depth flag failed: {result.stderr}")
self.assertTrue((output_dir / "SKILL.md").exists(), "SKILL.md not generated with --depth")
def test_analyze_generates_references(self):
"""Test that references directory is created."""
output_dir = self.test_dir / "output_refs"
result = self.run_command(
"analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick"
)
self.assertEqual(result.returncode, 0, f"Analysis failed: {result.stderr}")
# Check for references directory
refs_dir = output_dir / "references"
if refs_dir.exists(): # Optional, depends on content
self.assertTrue(refs_dir.is_dir(), "References is not a directory")
def test_analyze_output_structure(self):
"""Test that output has expected structure."""
output_dir = self.test_dir / "output_structure"
result = self.run_command(
"analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick"
)
self.assertEqual(result.returncode, 0, f"Analysis failed: {result.stderr}")
# Verify expected files/directories
self.assertTrue((output_dir / "SKILL.md").exists(), "SKILL.md missing")
# Check for code_analysis.json if it exists
analysis_file = output_dir / "code_analysis.json"
if analysis_file.exists():
# Verify it's valid JSON
with open(analysis_file) as f:
data = json.load(f)
self.assertIsInstance(data, (dict, list), "code_analysis.json is not valid JSON")
class TestAnalyzeOldCommand(unittest.TestCase):
"""Test that old skill-seekers-codebase command still works."""
def test_old_command_still_exists(self):
"""Test that skill-seekers-codebase still exists."""
result = subprocess.run(
["skill-seekers-codebase", "--help"],
capture_output=True,
text=True,
timeout=5
)
# Command should exist and show help
self.assertEqual(result.returncode, 0,
f"Old command doesn't work: {result.stderr}")
self.assertIn("--directory", result.stdout)
class TestAnalyzeIntegration(unittest.TestCase):
"""Integration tests for analyze command with other features."""
def setUp(self):
"""Set up test directory."""
self.test_dir = Path(tempfile.mkdtemp(prefix="analyze_int_"))
# Create minimal Python project
(self.test_dir / "main.py").write_text('''
def hello():
"""Say hello."""
return "Hello, World!"
''')
def tearDown(self):
"""Clean up test directory."""
if self.test_dir.exists():
shutil.rmtree(self.test_dir)
def test_analyze_then_check_output(self):
"""Test analyzing and verifying output can be read."""
output_dir = self.test_dir / "output"
# Run analysis
result = subprocess.run(
[
"skill-seekers", "analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick"
],
capture_output=True,
text=True,
timeout=120
)
self.assertEqual(result.returncode, 0, f"Analysis failed: {result.stderr}")
# Read and verify SKILL.md
skill_file = output_dir / "SKILL.md"
self.assertTrue(skill_file.exists(), "SKILL.md not created")
content = skill_file.read_text()
# Check for valid structure instead of specific content
# (file detection may vary in temp directories)
self.assertGreater(len(content), 50, "Output too short")
self.assertIn("Codebase", content, "Missing codebase header")
self.assertTrue(content.startswith("---"), "Missing YAML frontmatter")
def test_analyze_verbose_flag(self):
"""Test that verbose flag works."""
output_dir = self.test_dir / "output"
result = subprocess.run(
[
"skill-seekers", "analyze",
"--directory", str(self.test_dir),
"--output", str(output_dir),
"--quick",
"--verbose"
],
capture_output=True,
text=True,
timeout=120
)
self.assertEqual(result.returncode, 0, f"Verbose analysis failed: {result.stderr}")
# Verbose should produce more output
combined_output = result.stdout + result.stderr
self.assertGreater(len(combined_output), 100, "Verbose mode didn't produce extra output")
if __name__ == "__main__":
unittest.main()