"""Integration tests for the EGoT algorithm workflow."""

import pytest
import os
import tempfile
import shutil
from pathlib import Path
from unittest.mock import patch, Mock
from src.main import load_and_run_algorithm
from src.data_models.task_config import TaskConfig


class TestEGoTWorkflow:
    """Integration tests for EGoT algorithm workflow."""
    
    @pytest.fixture
    def mock_task_config(self):
        """Create a mock TaskConfig for testing."""
        return TaskConfig(
            feasibility_check_points=["Check 1", "Check 2"],
            task_description="Test task description for EGoT",
            known_solutions=["Solution 1", "Solution 2"]
        )
    
    @pytest.fixture
    def temp_results_dir(self):
        """Create a temporary results directory for testing."""
        temp_dir = tempfile.mkdtemp()
        results_dir = Path(temp_dir) / "results"
        results_dir.mkdir()
        yield results_dir
        shutil.rmtree(temp_dir)
    
    def test_egot_algorithm_loading(self, mock_task_config):
        """Test that EGoT algorithm can be loaded and initialized."""
        # Setup mock LLM client
        mock_client = Mock()
        mock_client.call_openai.return_value = '{"ma": "method analysis", "me": "method explanation", "a": "answer", "ra": "rationale", "s": 85, "rs": "reasoning", "Pr(s)": 0.8, "rpr": "aggregated rationale"}'
        
        with patch('algorithms.egot.main.LLMAPIClient') as mock_llm_client_class:
            mock_llm_client_class.return_value = mock_client
            
            # Test algorithm loading
            solution_text, intermediate_logs = load_and_run_algorithm(
                algorithm_name='egot',
                task_config=mock_task_config,
                backbone_llm_name='gemini-pro',
                num_analogous_problems=10,
                num_solutions_per_problem=5,
                num_exploratory_ideas=50,
                num_new_rule_sets=3,
                num_final_solutions=3,
                num_solutions_combinational=10
            )
            
            # Verify results
            assert isinstance(solution_text, str)
            assert len(solution_text) > 0
            assert intermediate_logs == []  # EGoT should return empty intermediate logs
            
            # Verify EGoT-specific content in solution
            assert "Enhanced Graph of Thoughts" in solution_text
            assert "Final Solutions" in solution_text
    
    def test_egot_no_intermediate_logs_saved(self, mock_task_config, temp_results_dir):
        """Test that EGoT does not save intermediate logs to disk."""
        # Setup mock LLM client
        mock_client = Mock()
        mock_client.call_openai.return_value = '{"ma": "method analysis", "me": "method explanation", "a": "answer", "ra": "rationale", "s": 85, "rs": "reasoning", "Pr(s)": 0.8, "rpr": "aggregated rationale"}'
        
        with patch('algorithms.egot.main.LLMAPIClient') as mock_llm_client_class:
            mock_llm_client_class.return_value = mock_client
            
            # Mock the results directory
            with patch('src.main.Path') as mock_path:
                mock_path.return_value.parent.parent = temp_results_dir.parent
                
                # Run EGoT algorithm
                solution_text, intermediate_logs = load_and_run_algorithm(
                    algorithm_name='egot',
                    task_config=mock_task_config,
                    backbone_llm_name='gemini-pro',
                    num_analogous_problems=10,
                    num_solutions_per_problem=5,
                    num_exploratory_ideas=50,
                    num_new_rule_sets=3,
                    num_final_solutions=3,
                    num_solutions_combinational=10
                )
                
                # Verify no intermediate log files are created
                intermediate_logs_dir = temp_results_dir / "intermediate_logs"
                if intermediate_logs_dir.exists():
                    log_files = list(intermediate_logs_dir.glob("*.json"))
                    assert len(log_files) == 0, "EGoT should not create intermediate log files"
    
    def test_egot_algorithm_not_found(self, mock_task_config):
        """Test error handling when EGoT algorithm is not found."""
        # Test with a non-existent algorithm name
        with pytest.raises(ImportError, match="Failed to import algorithm module 'nonexistent'"):
            load_and_run_algorithm(
                algorithm_name='nonexistent',
                task_config=mock_task_config,
                backbone_llm_name='gemini-pro',
                num_analogous_problems=10,
                num_solutions_per_problem=5,
                num_exploratory_ideas=50,
                num_new_rule_sets=3,
                num_final_solutions=3,
                num_solutions_combinational=10
            )
    
    def test_egot_parameter_validation(self):
        """Test that EGoT validates parameters correctly."""
        # Setup mock LLM client
        mock_client = Mock()
        mock_client.call_openai.return_value = '{"ma": "method analysis", "me": "method explanation", "a": "answer", "ra": "rationale", "s": 85, "rs": "reasoning", "Pr(s)": 0.8, "rpr": "aggregated rationale"}'
        
        with patch('algorithms.egot.main.LLMAPIClient') as mock_llm_client_class:
            mock_llm_client_class.return_value = mock_client
            
            # Test with invalid task_config - this should raise ValueError during model initialization
            with pytest.raises(Exception, match="Algorithm execution failed.*task_config must be a TaskConfig object"):
                load_and_run_algorithm(
                    algorithm_name='egot',
                    task_config="invalid",
                    backbone_llm_name='gemini-pro',
                    num_analogous_problems=10,
                    num_solutions_per_problem=5,
                    num_exploratory_ideas=50,
                    num_new_rule_sets=3,
                    num_final_solutions=3,
                    num_solutions_combinational=10
                )
            
            # Test with invalid backbone_llm_name - this should raise ValueError during model initialization
            with pytest.raises(Exception, match="Algorithm execution failed.*backbone_llm_name must be a non-empty string"):
                load_and_run_algorithm(
                    algorithm_name='egot',
                    task_config=TaskConfig(
                        feasibility_check_points=["Check 1"],
                        task_description="Test task",
                        known_solutions=["Solution 1"]
                    ),
                    backbone_llm_name="",
                    num_analogous_problems=10,
                    num_solutions_per_problem=5,
                    num_exploratory_ideas=50,
                    num_new_rule_sets=3,
                    num_final_solutions=3,
                    num_solutions_combinational=10
                )
    
    def test_egot_solution_ranking(self, mock_task_config):
        """Test that EGoT correctly ranks solutions by confidence score."""
        # Setup mock LLM client
        mock_client = Mock()
        mock_client.call_openai.return_value = '{"ma": "method analysis", "me": "method explanation", "a": "answer", "ra": "rationale", "s": 85, "rs": "reasoning", "Pr(s)": 0.8, "rpr": "aggregated rationale"}'
        
        with patch('algorithms.egot.main.LLMAPIClient') as mock_llm_client_class:
            mock_llm_client_class.return_value = mock_client
            
            # Run EGoT with num_final_solutions=2
            solution_text, intermediate_logs = load_and_run_algorithm(
                algorithm_name='egot',
                task_config=mock_task_config,
                backbone_llm_name='gemini-pro',
                num_analogous_problems=10,
                num_solutions_per_problem=5,
                num_exploratory_ideas=50,
                num_new_rule_sets=3,
                num_final_solutions=2,
                num_solutions_combinational=10
            )
            
            # Verify that the top 2 solutions are selected
            assert "Solution 1" in solution_text
            assert "Solution 2" in solution_text
            assert "Solution 3" not in solution_text
            
            # Verify confidence scores are displayed
            assert "Confidence:" in solution_text
            
            # Since we're using the same mock response for all calls, all solutions will have the same confidence
            # The confidence should be 85 * 0.8 = 68.00
            assert "Confidence: 68.00" in solution_text