"""Unit tests for main.py functionality."""

import pytest
import argparse
from unittest.mock import patch, MagicMock
from src.main import load_and_run_algorithm
from src.data_models.task_config import TaskConfig


class TestMain:
    """Test cases for main.py functions."""
    
    def setup_method(self):
        """Set up test fixtures."""
        self.task_config = TaskConfig(
            feasibility_check_points=["Check 1", "Check 2"],
            task_description="Test task description",
            known_solutions=["Solution 1", "Solution 2"]
        )
    
    def test_load_and_run_algorithm_success(self):
        """Test successful loading and running of an algorithm."""
        with patch('src.main.importlib.import_module') as mock_import:
            # Mock the algorithm module
            mock_module = MagicMock()
            mock_reasoning_model_class = MagicMock()
            mock_model_instance = MagicMock()
            mock_model_instance.run.return_value = ("Generated solution", [("prompt", "response")])
            
            mock_reasoning_model_class.return_value = mock_model_instance
            mock_module.reasoning_model = mock_reasoning_model_class
            mock_import.return_value = mock_module
            
            result = load_and_run_algorithm(
                "test_algorithm",
                self.task_config,
                "gpt-4",
                10,
                5,
                50,
                3,
                3,
                10
            )
            
            solution_text, intermediate_logs = result
            assert solution_text == "Generated solution"
            assert len(intermediate_logs) == 1
            
            # Verify the algorithm was instantiated with correct parameters
            mock_reasoning_model_class.assert_called_once_with(
                self.task_config,
                "gpt-4",
                10,
                5,
                3,
                10
            )
            
            # Verify the run method was called
            mock_model_instance.run.assert_called_once()
    
    def test_load_and_run_algorithm_import_error(self):
        """Test handling of import errors."""
        with patch('src.main.importlib.import_module') as mock_import:
            mock_import.side_effect = ImportError("No module named 'test_algorithm'")
            
            with pytest.raises(ImportError, match="Failed to import algorithm module 'test_algorithm': No module named 'test_algorithm'"):
                load_and_run_algorithm(
                    "test_algorithm",
                    self.task_config,
                    "gpt-4",
                    10,
                    5,
                    50,
                    3,
                    3,
                    10
                )
    
    def test_load_and_run_algorithm_attribute_error(self):
        """Test handling of missing reasoning_model class."""
        with patch('src.main.importlib.import_module') as mock_import:
            # Mock module without reasoning_model class
            mock_module = MagicMock()
            del mock_module.reasoning_model
            mock_import.return_value = mock_module
            
            with pytest.raises(AttributeError, match="Algorithm module 'test_algorithm' missing required 'reasoning_model' class"):
                load_and_run_algorithm(
                    "test_algorithm",
                    self.task_config,
                    "gpt-4",
                    10,
                    5,
                    50,
                    3,
                    3,
                    10
                )
    
    def test_load_and_run_algorithm_execution_error(self):
        """Test handling of algorithm execution errors."""
        with patch('src.main.importlib.import_module') as mock_import:
            # Mock the algorithm module
            mock_module = MagicMock()
            mock_reasoning_model_class = MagicMock()
            mock_model_instance = MagicMock()
            mock_model_instance.run.side_effect = Exception("Algorithm failed")
            
            mock_reasoning_model_class.return_value = mock_model_instance
            mock_module.reasoning_model = mock_reasoning_model_class
            mock_import.return_value = mock_module
            
            with pytest.raises(Exception, match="Algorithm execution failed: Algorithm failed"):
                load_and_run_algorithm(
                    "test_algorithm",
                    self.task_config,
                    "gpt-4",
                    10,
                    5,
                    50,
                    3,
                    3,
                    10
                )
    
    def test_load_and_run_algorithm_non_string_return(self):
        """Test handling of non-string return value."""
        with patch('src.main.importlib.import_module') as mock_import:
            # Mock the algorithm module
            mock_module = MagicMock()
            mock_reasoning_model_class = MagicMock()
            mock_model_instance = MagicMock()
            mock_model_instance.run.return_value = (123, [("prompt", "response")])  # Non-string return
            
            mock_reasoning_model_class.return_value = mock_model_instance
            mock_module.reasoning_model = mock_reasoning_model_class
            mock_import.return_value = mock_module
            
            with pytest.raises(Exception, match="Algorithm execution failed: Algorithm must return a string solution"):
                load_and_run_algorithm(
                    "test_algorithm",
                    self.task_config,
                    "gpt-4",
                    10,
                    5,
                    50,
                    3,
                    3,
                    10
                )
    
    def test_load_and_run_algorithm_with_different_parameters(self):
        """Test loading algorithm with different parameter values."""
        with patch('src.main.importlib.import_module') as mock_import:
            # Mock the algorithm module
            mock_module = MagicMock()
            mock_reasoning_model_class = MagicMock()
            mock_model_instance = MagicMock()
            mock_model_instance.run.return_value = ("Solution with different params", [("prompt", "response")])
            
            mock_reasoning_model_class.return_value = mock_model_instance
            mock_module.reasoning_model = mock_reasoning_model_class
            mock_import.return_value = mock_module
            
            result = load_and_run_algorithm(
                "different_algorithm",
                self.task_config,
                "claude-3",
                15,
                8,
                50,
                3,
                3,
                15
            )
            
            solution_text, intermediate_logs = result
            assert solution_text == "Solution with different params"
            assert len(intermediate_logs) == 1
            
            # Verify the algorithm was instantiated with different parameters
            mock_reasoning_model_class.assert_called_once_with(
                self.task_config,
                "claude-3",
                15,
                8,
                3,
                15
            )
    
    def test_argument_parser_help(self):
        """Test that argument parser includes new arguments."""
        from src.main import main
        
        # This test verifies that the argument parser can be created without errors
        # and that the new arguments are present
        parser = argparse.ArgumentParser()
        parser.add_argument("--task-name", required=True, help="Name of the task to solve")
        parser.add_argument("--algorithm-name", required=True, help="Name of the algorithm to use")
        parser.add_argument("--backbone-llm-name", required=True, help="Name of the backbone LLM")
        parser.add_argument("--num-analogous-problems", type=int, default=10, help="Number of analogous problems to find")
        parser.add_argument("--num-solutions-per-problem", type=int, default=5, help="Number of solutions per analogous problem")
        parser.add_argument("--num-final-solutions", type=int, default=3, help="Number of final solutions to generate or select")
        parser.add_argument("--num-solutions-combinational", type=int, default=10, help="Number of new solutions to synthesize by the Combinational Creative Reasoning algorithm")
        
        # Test parsing with new arguments
        args = parser.parse_args([
            "--task-name", "test_task",
            "--algorithm-name", "test_algorithm", 
            "--backbone-llm-name", "gpt-4",
            "--num-analogous-problems", "15",
            "--num-solutions-per-problem", "7",
            "--num-final-solutions", "5",
            "--num-solutions-combinational", "12"
        ])
        
        assert args.task_name == "test_task"
        assert args.algorithm_name == "test_algorithm"
        assert args.backbone_llm_name == "gpt-4"
        assert args.num_analogous_problems == 15
        assert args.num_solutions_per_problem == 7
        assert args.num_final_solutions == 5
        assert args.num_solutions_combinational == 12
        
        # Test default values
        args_default = parser.parse_args([
            "--task-name", "test_task",
            "--algorithm-name", "test_algorithm",
            "--backbone-llm-name", "gpt-4"
        ])
        
        assert args_default.num_analogous_problems == 10
        assert args_default.num_solutions_per_problem == 5
        assert args_default.num_final_solutions == 3
        assert args_default.num_solutions_combinational == 10
