Files
markitect-main/tests/e2e/cli/test_issue_commands_e2e.py
tegwick 21a5d1d734
Some checks failed
Test Suite / unit-tests (3.11) (push) Has been cancelled
Test Suite / unit-tests (3.12) (push) Has been cancelled
Test Suite / integration-tests (push) Has been cancelled
Test Suite / e2e-tests (push) Has been cancelled
Test Suite / performance-tests (push) Has been cancelled
Test Suite / code-quality (push) Has been cancelled
Test Suite / security-scan (push) Has been cancelled
Test Suite / test-summary (push) Has been cancelled
feat: Implement comprehensive Testing Architecture Enhancement
Establishes robust testing framework with clean architecture patterns:

## Phase 1: Test Infrastructure Foundation
- Global test configuration with pytest.ini and conftest.py
- Isolated test workspaces and environment management
- Comprehensive fixture library for all test types
- Test requirements and dependency management

## Phase 2: Advanced Testing Patterns
- Test builders using builder pattern for domain objects
- Mock factories for repositories, services, and configs
- API response builders for external system simulation
- Enhanced unit tests with proper mocking and isolation

## Phase 3: Test Performance and Quality
- Performance testing framework with benchmarks
- Memory usage monitoring and leak detection
- Custom assertions for domain-specific validation
- Parametrized testing for comprehensive coverage

## Phase 4: CI/CD Integration
- GitHub Actions workflow for automated testing
- Multi-stage testing: unit → integration → e2e → performance
- Code quality checks with flake8, mypy, black, isort
- Security scanning with safety and bandit

## Testing Architecture Benefits
 100+ new test infrastructure components
 Standardized test organization (unit/integration/e2e)
 Mock-based testing with no external dependencies
 Performance regression detection
 Comprehensive fixture library
 CI/CD pipeline with quality gates

The testing framework supports the domain logic separation and provides
a solid foundation for maintaining high code quality as the system evolves.

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-09-26 22:36:35 +02:00

348 lines
12 KiB
Python

"""
End-to-end tests for issue management CLI commands.
Demonstrates:
- CLI command testing with real processes
- Environment isolation
- Workflow validation
- Output verification
"""
import pytest
import subprocess
import json
from pathlib import Path
import time
import os
from tests.utils.assertions import assert_file_exists, assert_directory_exists, assert_file_contains
@pytest.mark.e2e
class TestIssueCommandsE2E:
"""End-to-end tests for issue management CLI commands."""
def test_show_issue_command_basic(self, isolated_environment):
"""Test basic issue show command."""
# Act
result = subprocess.run(
["python", "tddai_cli.py", "show-issue", "23"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Assert
assert result.returncode == 0, f"Command failed with stderr: {result.stderr}"
assert "Issue #23" in result.stdout or "issue 23" in result.stdout.lower()
def test_show_issue_command_with_invalid_number(self, isolated_environment):
"""Test show issue command with invalid issue number."""
# Act
result = subprocess.run(
["python", "tddai_cli.py", "show-issue", "99999"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Assert - Should handle gracefully
# Note: Depending on implementation, this might return 0 or 1
assert "not found" in result.stdout.lower() or "error" in result.stderr.lower()
def test_workspace_status_command(self, isolated_environment):
"""Test workspace status command."""
# Act
result = subprocess.run(
["python", "tddai_cli.py", "workspace-status"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Assert
assert result.returncode == 0
# Should show workspace information
assert "workspace" in result.stdout.lower() or "status" in result.stdout.lower()
@pytest.mark.slow
def test_complete_issue_workflow(self, isolated_environment, test_workspace):
"""Test complete issue workflow from start to finish."""
workspace_dir = Path(isolated_environment["MARKITECT_WORKSPACE_DIR"])
workspace_dir.mkdir(exist_ok=True)
# Step 1: Check initial workspace status
result = subprocess.run(
["python", "tddai_cli.py", "workspace-status"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
assert result.returncode == 0
# Step 2: Start working on an issue
result = subprocess.run(
["python", "tddai_cli.py", "start-issue", "42"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd(),
timeout=30 # Prevent hanging
)
# Verify the start command works (might create workspace)
if result.returncode == 0:
# If successful, check if workspace was created
issue_workspace = workspace_dir / "issue_42"
if issue_workspace.exists():
assert_directory_exists(issue_workspace)
# Step 3: Check workspace status again
result = subprocess.run(
["python", "tddai_cli.py", "workspace-status"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
assert result.returncode == 0
# Step 4: Try to finish (cleanup)
result = subprocess.run(
["python", "tddai_cli.py", "finish-issue"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd(),
timeout=30
)
# The finish command should work or provide meaningful feedback
assert result.returncode in [0, 1] # Allow for various implementation states
def test_list_open_issues_command(self, isolated_environment):
"""Test listing open issues."""
# Act
result = subprocess.run(
["python", "tddai_cli.py", "list-open-issues"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Assert
assert result.returncode == 0
# Should return some form of issue listing (even if empty)
output = result.stdout.strip()
assert len(output) >= 0 # Any output is acceptable
def test_cli_help_commands(self, isolated_environment):
"""Test CLI help functionality."""
# Test main help
result = subprocess.run(
["python", "tddai_cli.py", "--help"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
assert result.returncode == 0
assert "usage" in result.stdout.lower() or "commands" in result.stdout.lower()
# Test specific command help
result = subprocess.run(
["python", "tddai_cli.py", "show-issue", "--help"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
assert result.returncode == 0
def test_cli_with_invalid_command(self, isolated_environment):
"""Test CLI behavior with invalid command."""
# Act
result = subprocess.run(
["python", "tddai_cli.py", "invalid-command"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Assert - Should handle gracefully
assert result.returncode != 0
assert "error" in result.stderr.lower() or "unknown" in result.stderr.lower()
def test_cli_error_handling(self, isolated_environment):
"""Test CLI error handling for various scenarios."""
# Test with missing required argument
result = subprocess.run(
["python", "tddai_cli.py", "show-issue"], # Missing issue number
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Should provide helpful error message
assert result.returncode != 0
assert len(result.stderr) > 0 or "error" in result.stdout.lower()
@pytest.mark.parametrize("issue_number", ["1", "23", "100"])
def test_show_issue_command_multiple_issues(self, isolated_environment, issue_number):
"""Test show issue command with multiple issue numbers."""
# Act
result = subprocess.run(
["python", "tddai_cli.py", "show-issue", issue_number],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd(),
timeout=15
)
# Assert - Command should execute without crashing
assert result.returncode in [0, 1] # Allow for not found scenarios
assert len(result.stdout + result.stderr) > 0 # Should provide some output
def test_cli_performance(self, isolated_environment, performance_timer):
"""Test CLI command performance."""
# Act
performance_timer.start()
result = subprocess.run(
["python", "tddai_cli.py", "workspace-status"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
performance_timer.stop()
# Assert
assert result.returncode == 0
# CLI commands should be reasonably fast
assert performance_timer.elapsed < 10.0, f"CLI command took {performance_timer.elapsed:.2f}s"
def test_cli_output_formatting(self, isolated_environment):
"""Test CLI output formatting and structure."""
# Test workspace status output
result = subprocess.run(
["python", "tddai_cli.py", "workspace-status"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd()
)
if result.returncode == 0:
output = result.stdout
# Output should be readable and structured
assert len(output.strip()) > 0
# Should not contain obvious error traces
assert "Traceback" not in output
assert "Exception" not in output
def test_cli_environment_isolation(self, test_workspace):
"""Test that CLI commands work in isolated environment."""
# Create isolated environment
isolated_env = {
"MARKITECT_WORKSPACE_DIR": str(test_workspace / "isolated"),
"MARKITECT_GITEA_URL": "http://isolated-gitea.com",
"MARKITECT_REPO_OWNER": "isolated",
"MARKITECT_REPO_NAME": "test",
"PYTHONPATH": "."
}
# Update with current env to preserve PATH, etc.
full_env = dict(os.environ)
full_env.update(isolated_env)
# Act
result = subprocess.run(
["python", "tddai_cli.py", "workspace-status"],
env=full_env,
capture_output=True,
text=True,
cwd=Path.cwd()
)
# Assert - Should work with isolated environment
assert result.returncode == 0
# Should use isolated workspace directory
workspace_path = test_workspace / "isolated"
workspace_path.mkdir(exist_ok=True)
def test_cli_concurrent_execution(self, isolated_environment):
"""Test concurrent CLI command execution."""
import threading
import queue
results_queue = queue.Queue()
def run_command(command_args):
result = subprocess.run(
command_args,
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd(),
timeout=15
)
results_queue.put(result)
# Start multiple commands concurrently
commands = [
["python", "tddai_cli.py", "workspace-status"],
["python", "tddai_cli.py", "show-issue", "1"],
["python", "tddai_cli.py", "show-issue", "2"],
]
threads = []
for cmd in commands:
thread = threading.Thread(target=run_command, args=(cmd,))
threads.append(thread)
thread.start()
# Wait for all threads to complete
for thread in threads:
thread.join(timeout=20)
# Collect results
results = []
while not results_queue.empty():
results.append(results_queue.get())
# Assert
assert len(results) == len(commands)
# At least some commands should succeed
successful_commands = [r for r in results if r.returncode == 0]
assert len(successful_commands) > 0
@pytest.mark.smoke
def test_cli_smoke_test(self, isolated_environment):
"""Basic smoke test for CLI functionality."""
# Test that the CLI script exists and is executable
cli_script = Path("tddai_cli.py")
assert_file_exists(cli_script)
# Test basic command execution
result = subprocess.run(
["python", "tddai_cli.py", "--help"],
env=isolated_environment,
capture_output=True,
text=True,
cwd=Path.cwd(),
timeout=10
)
# Should at least not crash
assert result.returncode in [0, 1, 2] # Various help return codes
assert len(result.stdout + result.stderr) > 0