"""Unit tests for the exploratory creative reasoning algorithm."""

import unittest
from unittest.mock import Mock, patch, MagicMock
import uuid
from datetime import datetime

from src.algorithms.exploratory_creative_reasoning.main import reasoning_model
from data_models.task_config import TaskConfig


class TestExploratoryCreativeReasoning(unittest.TestCase):
    """Test cases for the exploratory creative reasoning algorithm."""
    
    def setUp(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"]
        )
        self.backbone_llm_name = "gpt-4"
        self.num_analogous_problems = 5
        self.num_solutions_per_problem = 3
        self.num_exploratory_ideas = 10
    
    def test_init_default_parameters(self):
        """Test initialization with default parameters."""
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            num_solutions_combinational=10
        )
        
        self.assertEqual(model.num_exploratory_ideas, 50)  # Default value
        self.assertEqual(model.task_config, self.task_config)
        self.assertEqual(model.backbone_llm_name, self.backbone_llm_name)
        self.assertEqual(model.num_analogous_problems, self.num_analogous_problems)
        self.assertEqual(model.num_solutions_per_problem, self.num_solutions_per_problem)
        self.assertEqual(model.intermediate_logs, [])
        self.assertIsInstance(model.intermediate_logs, list)
    
    def test_init_custom_parameters(self):
        """Test initialization with custom parameters."""
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        
        self.assertEqual(model.num_exploratory_ideas, self.num_exploratory_ideas)
        self.assertEqual(model.task_config, self.task_config)
        self.assertEqual(model.backbone_llm_name, self.backbone_llm_name)
        self.assertEqual(model.num_analogous_problems, self.num_analogous_problems)
        self.assertEqual(model.num_solutions_per_problem, self.num_solutions_per_problem)
        self.assertEqual(model.intermediate_logs, [])
        self.assertIsInstance(model.intermediate_logs, list)
    
    def test_init_invalid_num_exploratory_ideas(self):
        """Test initialization with invalid num_exploratory_ideas."""
        with self.assertRaises(ValueError):
            reasoning_model(
                self.task_config,
                self.backbone_llm_name,
                self.num_analogous_problems,
                self.num_solutions_per_problem,
                -1,  # Invalid negative value
                num_solutions_combinational=10
            )
        
        with self.assertRaises(ValueError):
            reasoning_model(
                self.task_config,
                self.backbone_llm_name,
                self.num_analogous_problems,
                self.num_solutions_per_problem,
                0,  # Invalid zero value
                num_solutions_combinational=10
            )
    
    def test_init_invalid_num_solutions_combinational(self):
        """Test initialization with invalid num_solutions_combinational."""
        with self.assertRaises(ValueError):
            reasoning_model(
                self.task_config,
                self.backbone_llm_name,
                self.num_analogous_problems,
                self.num_solutions_per_problem,
                self.num_exploratory_ideas,
                num_solutions_combinational=-1  # Invalid negative value
            )
        
        with self.assertRaises(ValueError):
            reasoning_model(
                self.task_config,
                self.backbone_llm_name,
                self.num_analogous_problems,
                self.num_solutions_per_problem,
                self.num_exploratory_ideas,
                num_solutions_combinational=0  # Invalid zero value
            )
    
    @patch('src.algorithms.exploratory_creative_reasoning.main.uuid.uuid4')
    @patch('src.algorithms.exploratory_creative_reasoning.main.datetime')
    @patch('src.algorithms.exploratory_creative_reasoning.main.extract_json_from_response')
    def test_expand_ideas_exploratorily_success(self, mock_extract_json, mock_datetime, mock_uuid):
        """Test successful idea expansion."""
        # Setup mocks
        mock_uuid.return_value = "test-uuid-123"
        mock_datetime.now.return_value.isoformat.return_value = "2024-01-01T12:00:00"
        
        mock_llm_response = '["New idea 1", "New idea 2", "New idea 3"]'
        mock_parsed_response = ["New idea 1", "New idea 2", "New idea 3"]
        
        mock_extract_json.return_value = mock_parsed_response
        
        # Create model with mocked LLM client
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        model.llm_client = Mock()
        model.llm_client.call_openai.return_value = mock_llm_response
        
        # Test the method
        all_ideas = ["Existing idea 1", "Existing idea 2"]
        result = model._expand_ideas_exploratorily(all_ideas)
        
        # Verify LLM call
        model.llm_client.call_openai.assert_called_once()
        call_args = model.llm_client.call_openai.call_args
        
        self.assertEqual(call_args[1]['model_name'], self.backbone_llm_name)
        self.assertEqual(call_args[1]['temperature'], 0.7)
        self.assertIn(self.task_config.task_description, call_args[1]['prompt'])
        self.assertIn("Existing idea 1", call_args[1]['prompt'])
        self.assertIn("Existing idea 2", call_args[1]['prompt'])
        self.assertIn(str(self.num_exploratory_ideas), call_args[1]['prompt'])
        
        # Verify JSON extraction
        mock_extract_json.assert_called_once_with(mock_llm_response)
        
        # Verify result
        self.assertEqual(result, mock_parsed_response)
        
        # Verify logging
        self.assertEqual(len(model.intermediate_logs), 1)
        step_name, log_entries = model.intermediate_logs[0]
        self.assertEqual(step_name, "Exploratory Idea Expansion")
        self.assertEqual(len(log_entries), 1)
        
        log_entry = log_entries[0]
        self.assertEqual(log_entry['prompt'], call_args[1]['prompt'])
        self.assertEqual(log_entry['raw_response'], mock_llm_response)
        self.assertEqual(log_entry['parsed_output'], mock_parsed_response)
        self.assertEqual(log_entry['llm_model_name'], self.backbone_llm_name)
        self.assertEqual(log_entry['temperature'], 0.7)
        self.assertEqual(log_entry['timestamp'], "2024-01-01T12:00:00")
        self.assertEqual(log_entry['llm_call_id'], "test-uuid-123")
        self.assertNotIn('error', log_entry)
    
    @patch('src.algorithms.exploratory_creative_reasoning.main.uuid.uuid4')
    @patch('src.algorithms.exploratory_creative_reasoning.main.datetime')
    @patch('src.algorithms.exploratory_creative_reasoning.main.extract_json_from_response')
    def test_expand_ideas_exploratorily_llm_failure(self, mock_extract_json, mock_datetime, mock_uuid):
        """Test idea expansion with LLM call failure."""
        # Setup mocks
        mock_uuid.return_value = "test-uuid-456"
        mock_datetime.now.return_value.isoformat.return_value = "2024-01-01T12:00:00"
        
        # Create model with mocked LLM client that raises exception
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        model.llm_client = Mock()
        model.llm_client.call_openai.side_effect = Exception("LLM API error")
        
        # Test the method
        all_ideas = ["Existing idea 1"]
        
        with self.assertRaises(RuntimeError) as context:
            model._expand_ideas_exploratorily(all_ideas)
        
        self.assertIn("Failed to expand ideas exploratorily", str(context.exception))
        self.assertIn("LLM API error", str(context.exception))
        
        # Verify error logging
        self.assertEqual(len(model.intermediate_logs), 1)
        step_name, log_entries = model.intermediate_logs[0]
        self.assertEqual(step_name, "Exploratory Idea Expansion")
        self.assertEqual(len(log_entries), 1)
        
        log_entry = log_entries[0]
        self.assertIn('error', log_entry)
        self.assertEqual(log_entry['error'], "LLM API error")
        self.assertEqual(log_entry['llm_call_id'], "test-uuid-456")
        self.assertEqual(log_entry['timestamp'], "2024-01-01T12:00:00")
    
    @patch('src.algorithms.exploratory_creative_reasoning.main.uuid.uuid4')
    @patch('src.algorithms.exploratory_creative_reasoning.main.datetime')
    @patch('src.algorithms.exploratory_creative_reasoning.main.extract_json_from_response')
    def test_expand_ideas_exploratorily_parsing_failure(self, mock_extract_json, mock_datetime, mock_uuid):
        """Test idea expansion with JSON parsing failure."""
        # Setup mocks
        mock_uuid.return_value = "test-uuid-789"
        mock_datetime.now.return_value.isoformat.return_value = "2024-01-01T12:00:00"
        
        mock_llm_response = "Invalid JSON response"
        mock_extract_json.side_effect = Exception("JSON parsing error")
        
        # Create model with mocked LLM client
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        model.llm_client = Mock()
        model.llm_client.call_openai.return_value = mock_llm_response
        
        # Test the method
        all_ideas = ["Existing idea 1"]
        
        with self.assertRaises(RuntimeError) as context:
            model._expand_ideas_exploratorily(all_ideas)
        
        self.assertIn("Failed to expand ideas exploratorily", str(context.exception))
        self.assertIn("JSON parsing error", str(context.exception))
        
        # Verify error logging
        self.assertEqual(len(model.intermediate_logs), 1)
        step_name, log_entries = model.intermediate_logs[0]
        self.assertEqual(step_name, "Exploratory Idea Expansion")
        self.assertEqual(len(log_entries), 1)
        
        log_entry = log_entries[0]
        self.assertIn('error', log_entry)
        self.assertEqual(log_entry['error'], "JSON parsing error")
        self.assertEqual(log_entry['llm_call_id'], "test-uuid-789")
        self.assertEqual(log_entry['timestamp'], "2024-01-01T12:00:00")
    
    @patch('src.algorithms.exploratory_creative_reasoning.main.uuid.uuid4')
    @patch('src.algorithms.exploratory_creative_reasoning.main.datetime')
    @patch('src.algorithms.exploratory_creative_reasoning.main.extract_json_from_response')
    def test_expand_ideas_exploratorily_invalid_response_type(self, mock_extract_json, mock_datetime, mock_uuid):
        """Test idea expansion with invalid response type (not a list)."""
        # Setup mocks
        mock_uuid.return_value = "test-uuid-999"
        mock_datetime.now.return_value.isoformat.return_value = "2024-01-01T12:00:00"
        
        mock_llm_response = '{"not": "a list"}'
        mock_parsed_response = {"not": "a list"}
        
        mock_extract_json.return_value = mock_parsed_response
        
        # Create model with mocked LLM client
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        model.llm_client = Mock()
        model.llm_client.call_openai.return_value = mock_llm_response
        
        # Test the method
        all_ideas = ["Existing idea 1"]
        
        with self.assertRaises(RuntimeError) as context:
            model._expand_ideas_exploratorily(all_ideas)
        
        self.assertIn("Failed to expand ideas exploratorily", str(context.exception))
        self.assertIn("LLM response is not a list", str(context.exception))
        
        # Verify error logging
        self.assertEqual(len(model.intermediate_logs), 1)
        step_name, log_entries = model.intermediate_logs[0]
        self.assertEqual(step_name, "Exploratory Idea Expansion")
        self.assertEqual(len(log_entries), 1)
        
        log_entry = log_entries[0]
        self.assertIn('error', log_entry)
        self.assertIn("LLM response is not a list", log_entry['error'])
        self.assertEqual(log_entry['llm_call_id'], "test-uuid-999")
        self.assertEqual(log_entry['timestamp'], "2024-01-01T12:00:00")
    
    @patch('src.algorithms.exploratory_creative_reasoning.main.uuid.uuid4')
    @patch('src.algorithms.exploratory_creative_reasoning.main.datetime')
    @patch('src.algorithms.exploratory_creative_reasoning.main.extract_json_from_response')
    def test_expand_ideas_exploratorily_mixed_response_types(self, mock_extract_json, mock_datetime, mock_uuid):
        """Test idea expansion with mixed response types (some non-strings)."""
        # Setup mocks
        mock_uuid.return_value = "test-uuid-mixed"
        mock_datetime.now.return_value.isoformat.return_value = "2024-01-01T12:00:00"
        
        mock_llm_response = '["String idea", 123, {"nested": "object"}]'
        mock_parsed_response = ["String idea", 123, {"nested": "object"}]
        
        mock_extract_json.return_value = mock_parsed_response
        
        # Create model with mocked LLM client
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        model.llm_client = Mock()
        model.llm_client.call_openai.return_value = mock_llm_response
        
        # Test the method
        all_ideas = ["Existing idea 1"]
        result = model._expand_ideas_exploratorily(all_ideas)
        
        # Verify result - non-string items should be converted to strings
        expected_result = ["String idea", "123", "{'nested': 'object'}"]
        self.assertEqual(result, expected_result)
        
        # Verify logging
        self.assertEqual(len(model.intermediate_logs), 1)
        step_name, log_entries = model.intermediate_logs[0]
        self.assertEqual(step_name, "Exploratory Idea Expansion")
        self.assertEqual(len(log_entries), 1)
        
        log_entry = log_entries[0]
        self.assertEqual(log_entry['parsed_output'], mock_parsed_response)
        self.assertNotIn('error', log_entry)
    
    @patch('src.utils.llm_api_client.LLMAPIClient')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._find_analogous_problems')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._find_solutions_for_problems')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._decompose_solutions_into_ideas')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._identify_impactful_ideas')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._generate_new_solutions')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._evaluate_and_rank_solutions')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._format_final_solutions')
    def test_run_method_workflow(self, mock_format_final, mock_evaluate_rank, mock_generate_new, 
                                mock_identify_impactful, mock_decompose, mock_find_solutions, 
                                mock_find_analogous, mock_llm_client):
        """Test the complete run method workflow."""
        # Setup mock return values
        mock_find_analogous.return_value = ["Problem 1", "Problem 2"]
        mock_find_solutions.return_value = {"Problem 1": ["Sol 1", "Sol 2"], "Problem 2": ["Sol 3", "Sol 4"]}
        mock_decompose.return_value = ["Idea 1", "Idea 2", "Idea 3"]
        mock_identify_impactful.return_value = {"Solution 1": ["Impact 1"], "Solution 2": ["Impact 2"]}
        mock_generate_new.return_value = ["New Solution 1", "New Solution 2"]
        mock_evaluate_rank.return_value = [("New Solution 1", 85.0), ("New Solution 2", 72.0)]
        mock_format_final.return_value = "Final formatted solution"
        
        # Create model
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        
        # Mock the LLM client at the instance level
        mock_llm_instance = Mock()
        mock_llm_instance.call_openai.return_value = '["Mock idea 1", "Mock idea 2"]'
        model.llm_client = mock_llm_instance
        
        # Mock the _expand_ideas_exploratorily method to return specific ideas
        with patch.object(model, '_expand_ideas_exploratorily') as mock_expand:
            mock_expand.return_value = ["Exploratory Idea 1", "Exploratory Idea 2"]
            
            # Run the method
            result_solution, result_logs = model.run()
            
            # Verify the workflow sequence
            mock_find_analogous.assert_called_once()
            mock_find_solutions.assert_called_once_with(["Problem 1", "Problem 2"])
            mock_decompose.assert_called_once_with({"Problem 1": ["Sol 1", "Sol 2"], "Problem 2": ["Sol 3", "Sol 4"]})
            # The expand method should be called with the original ideas
            mock_expand.assert_called_once()
            mock_identify_impactful.assert_called_once()
            mock_generate_new.assert_called_once_with({"Solution 1": ["Impact 1"], "Solution 2": ["Impact 2"]}, 
                                                     ["Idea 1", "Idea 2", "Idea 3", "Exploratory Idea 1", "Exploratory Idea 2"])
            mock_evaluate_rank.assert_called_once_with(["New Solution 1", "New Solution 2"])
            mock_format_final.assert_called_once_with([("New Solution 1", 85.0), ("New Solution 2", 72.0)])
            
            # Verify return values
            self.assertEqual(result_solution, "Final formatted solution")
            self.assertEqual(result_logs, model.intermediate_logs)
    
    @patch('src.utils.llm_api_client.LLMAPIClient')
    @patch('src.algorithms.exploratory_creative_reasoning.main.CombinationalReasoningModel._find_analogous_problems')
    def test_run_method_exception_handling(self, mock_find_analogous, mock_llm_client):
        """Test run method exception handling."""
        # Setup mock to raise exception
        mock_find_analogous.side_effect = Exception("Test error")
        
        # Create model
        model = reasoning_model(
            self.task_config,
            self.backbone_llm_name,
            self.num_analogous_problems,
            self.num_solutions_per_problem,
            self.num_exploratory_ideas,
            num_solutions_combinational=10
        )
        
        # Mock the LLM client at the instance level
        mock_llm_instance = Mock()
        mock_llm_instance.call_openai.return_value = '["Mock idea 1", "Mock idea 2"]'
        model.llm_client = mock_llm_instance
        
        # Test exception handling
        with self.assertRaises(RuntimeError) as context:
            model.run()
        
        self.assertIn("Error during exploratory solution generation", str(context.exception))
        self.assertIn("Test error", str(context.exception))


if __name__ == '__main__':
    unittest.main()
