"""Integration tests for the main workflow."""

import pytest
from unittest.mock import Mock, patch, MagicMock
from pathlib import Path
import tempfile
import shutil

from src.main import (
    load_task_description,
    load_feasibility_check_points,
    load_known_solutions,
    load_known_solutions_concept,
    load_calibration_anchors,
    main
)
from src.data_models.task_config import TaskConfig


class TestMainIntegration:
    """Integration tests for main workflow."""
    
    def test_load_task_description_success(self, tmp_path):
        """Test successful task description loading."""
        # Create temporary task file
        task_file = tmp_path / "tasks" / "bridge.txt"
        task_file.parent.mkdir()
        task_file.write_text("Test task description")
        
        with patch('src.main.Path') as mock_path:
            # Mock the Path(__file__).parent to return our temp directory
            mock_path.return_value.parent = tmp_path
            
            description = load_task_description("bridge")
            assert description == "Test task description"
    
    def test_load_feasibility_check_points_success(self, tmp_path):
        """Test successful feasibility check points loading."""
        # Create temporary check points file
        check_points_file = tmp_path / "feasibility_check_points" / "bridge.txt"
        check_points_file.parent.mkdir()
        check_points_file.write_text("Check point 1\nCheck point 2\n\nCheck point 3")
        
        with patch('src.main.Path') as mock_path:
            mock_path.return_value.parent = tmp_path
            
            check_points = load_feasibility_check_points("bridge")
            assert len(check_points) == 3
            assert "Check point 1" in check_points
            assert "Check point 2" in check_points
            assert "Check point 3" in check_points
    
    def test_load_known_solutions_success(self, tmp_path):
        """Test successful known solutions loading."""
        # Create temporary solutions file
        solutions_file = tmp_path / "known_solutions" / "bridge.txt"
        solutions_file.parent.mkdir()
        solutions_file.write_text("Solution 1\nSolution 2\n\nSolution 3")
        
        with patch('src.main.Path') as mock_path:
            mock_path.return_value.parent = tmp_path
            
            solutions = load_known_solutions("bridge")
            assert len(solutions) == 3
            assert "Solution 1" in solutions
            assert "Solution 2" in solutions
            assert "Solution 3" in solutions
    
    def test_load_known_solutions_concept_success(self, tmp_path):
        """Test successful known solutions concept loading."""
        # Create temporary concepts file
        concepts_file = tmp_path / "known_solutions_concept" / "bridge.txt"
        concepts_file.parent.mkdir()
        concepts_file.write_text("# This is a comment\nConcept 1\nConcept 2\n\n# Another comment\nConcept 3")
        
        with patch('src.main.Path') as mock_path:
            mock_path.return_value.parent = tmp_path
            
            concepts = load_known_solutions_concept("bridge")
            assert len(concepts) == 3
            assert "Concept 1" in concepts
            assert "Concept 2" in concepts
            assert "Concept 3" in concepts
            # Comments should be filtered out
            assert "# This is a comment" not in concepts
            assert "# Another comment" not in concepts
    
    def test_load_calibration_anchors_success(self, tmp_path):
        """Test successful calibration anchors loading."""
        # Create temporary calibration anchors file
        anchors_file = tmp_path / "calibration_anchors" / "bridge.txt"
        anchors_file.parent.mkdir()
        anchors_file.write_text("* Score 4/10: Example 1\n* Score 8/10: Example 2\n\n* Score 6/10: Example 3")
        
        with patch('src.main.Path') as mock_path:
            mock_path.return_value.parent = tmp_path
            
            anchors = load_calibration_anchors("bridge")
            assert len(anchors) == 3
            assert "* Score 4/10: Example 1" in anchors
            assert "* Score 8/10: Example 2" in anchors
            assert "* Score 6/10: Example 3" in anchors
    
    def test_file_not_found_errors(self, tmp_path):
        """Test file not found errors."""
        with patch('src.main.Path') as mock_path:
            mock_path.return_value.parent = tmp_path
            
            # Test task description file not found
            with pytest.raises(FileNotFoundError, match="Task file not found"):
                load_task_description("nonexistent")
            
            # Test feasibility check points file not found
            with pytest.raises(FileNotFoundError, match="Feasibility check points file not found"):
                load_feasibility_check_points("nonexistent")
            
            # Test known solutions file not found
            with pytest.raises(FileNotFoundError, match="Known solutions file not found"):
                load_known_solutions("nonexistent")
            
            # Test known solutions concept file not found
            with pytest.raises(FileNotFoundError, match="Known solutions concept file not found"):
                load_known_solutions_concept("nonexistent")
    
    @patch('src.main.load_task_description')
    @patch('src.main.load_feasibility_check_points')
    @patch('src.main.load_known_solutions')
    @patch('src.main.load_known_solutions_concept')
    @patch('src.main.load_calibration_anchors')
    @patch('src.main.load_and_run_algorithm')
    @patch('src.main.run_evaluation')
    @patch('src.main.log_results')
    def test_main_workflow_success(
        self, 
        mock_log_results, 
        mock_run_evaluation, 
        mock_load_algorithm, 
        mock_load_calibration_anchors,
        mock_load_solutions_concept,
        mock_load_solutions, 
        mock_load_check_points, 
        mock_load_task_description
    ):
        """Test successful main workflow execution."""
        # Mock all the loading functions
        mock_load_task_description.return_value = "Test task description"
        mock_load_check_points.return_value = ["Check 1", "Check 2"]
        mock_load_solutions.return_value = ["Solution 1", "Solution 2"]
        mock_load_solutions_concept.return_value = ["Concept 1", "Concept 2"]
        mock_load_calibration_anchors.return_value = ["Score 4/10: Example 1", "Score 8/10: Example 2"]
        
        # Mock algorithm execution
        mock_load_algorithm.return_value = ("Generated solution text", [("prompt", "response")])
        
        # Mock evaluation results
        from src.data_models.evaluation_result import EvaluationResult
        mock_evaluation_results = [
            EvaluationResult(
                original_solution_id="sol_1_abc123",
                individual_solution_text="Generated solution text",
                feasibility_score=0.8,
                utility_score=0.9,
                novelty_score=0.7,
                creativity_score=0.79,
                intermediate_log_filename="test_run_intermediate_log.json",
                feasibility_reasoning="Test feasibility reasoning",
                utility_reasoning="Test utility reasoning",
                novelty_theme="Test novelty theme"
            )
        ]
        mock_run_evaluation.return_value = mock_evaluation_results
        
        # Mock argument parsing
        with patch('src.main.argparse.ArgumentParser.parse_args') as mock_parse_args:
            mock_args = Mock()
            mock_args.task_name = "bridge"
            mock_args.algorithm_name = "o1"
            mock_args.backbone_llm_name = "gemini-2.5-pro"
            mock_args.num_analogous_problems = 12
            mock_args.num_solutions_per_problem = 6
            mock_args.num_exploratory_ideas = 60
            mock_args.num_new_rule_sets = 4
            mock_args.num_final_solutions = 5
            mock_parse_args.return_value = mock_args
            
            # Run main function
            main()
            
            # Verify all functions were called correctly
            mock_load_task_description.assert_called_once_with("bridge")
            mock_load_check_points.assert_called_once_with("bridge")
            mock_load_solutions.assert_called_once_with("bridge")
            mock_load_solutions_concept.assert_called_once_with("bridge")
            # Check that load_and_run_algorithm was called with correct parameters
            call_args = mock_load_algorithm.call_args
            assert call_args[0][0] == "o1"  # algorithm_name
            assert call_args[0][2] == "gemini-2.5-pro"  # backbone_llm_name
            assert call_args[0][3] == 12  # num_analogous_problems
            assert call_args[0][4] == 6  # num_solutions_per_problem
            assert call_args[0][5] == 60  # num_exploratory_ideas
            assert call_args[0][6] == 4  # num_new_rule_sets
            assert call_args[0][7] == 5  # num_final_solutions
            mock_run_evaluation.assert_called_once()
            
            # Verify logging was called for each result with hyperparameters
            assert mock_log_results.call_count == 1
            # Check that log_results was called with hyperparameters
            call_args = mock_log_results.call_args
            assert call_args[1]['num_analogous_problems'] == 12
            assert call_args[1]['num_solutions_per_problem'] == 6
            assert call_args[1]['num_exploratory_ideas'] == 60
            assert call_args[1]['num_new_rule_sets'] == 4
            assert call_args[1]['num_final_solutions'] == 5
    
    @patch('src.main.load_task_description')
    @patch('src.main.load_feasibility_check_points')
    @patch('src.main.load_known_solutions')
    @patch('src.main.load_known_solutions_concept')
    @patch('src.main.load_and_run_algorithm')
    @patch('src.main.run_evaluation')
    @patch('src.main.save_intermediate_logs')
    @patch('src.main.log_results')
    def test_main_workflow_csv_output_with_hyperparameters(
        self, 
        mock_log_results,
        mock_save_logs,
        mock_run_evaluation, 
        mock_load_algorithm, 
        mock_load_solutions_concept,
        mock_load_solutions, 
        mock_load_check_points, 
        mock_load_task_description,
        tmp_path
    ):
        """Test that main workflow correctly writes hyperparameters to CSV."""
        import csv
        from pathlib import Path
        
        # Mock all the loading functions
        mock_load_task_description.return_value = "Test task description"
        mock_load_check_points.return_value = ["Check 1", "Check 2"]
        mock_load_solutions.return_value = ["Solution 1", "Solution 2"]
        mock_load_solutions_concept.return_value = ["Concept 1", "Concept 2"]
        
        # Mock algorithm execution
        mock_load_algorithm.return_value = ("Generated solution text", [("prompt", "response")])
        
        # Mock intermediate logs saving
        mock_save_logs.return_value = "test_run_intermediate_log.json"
        
        # Mock evaluation results
        from src.data_models.evaluation_result import EvaluationResult
        mock_evaluation_results = [
            EvaluationResult(
                original_solution_id="sol_1_abc123",
                individual_solution_text="Generated solution text",
                feasibility_score=0.8,
                utility_score=0.9,
                novelty_score=0.7,
                creativity_score=0.79,
                intermediate_log_filename="test_run_intermediate_log.json",
                feasibility_reasoning="Test feasibility reasoning",
                utility_reasoning="Test utility reasoning",
                novelty_theme="Test novelty theme"
            )
        ]
        mock_run_evaluation.return_value = mock_evaluation_results
        
        # Mock argument parsing
        with patch('src.main.argparse.ArgumentParser.parse_args') as mock_parse_args:
            mock_args = Mock()
            mock_args.task_name = "bridge"
            mock_args.algorithm_name = "o1"
            mock_args.backbone_llm_name = "gemini-2.5-pro"
            mock_args.num_analogous_problems = 15
            mock_args.num_solutions_per_problem = 7
            mock_args.num_exploratory_ideas = 80
            mock_args.num_new_rule_sets = 6
            mock_args.num_final_solutions = 4
            mock_parse_args.return_value = mock_args
            
            # Run main function
            main()
            
            # Verify log_results was called with correct hyperparameters
            assert mock_log_results.call_count == 1
            call_args = mock_log_results.call_args
            
            # Check basic fields
            assert call_args[1]['algorithm_name'] == "o1"
            assert call_args[1]['task_name'] == "bridge"
            assert call_args[1]['backbone_llm_name'] == "gemini-2.5-pro"
            
            # Check hyperparameter fields
            assert call_args[1]['num_analogous_problems'] == 15
            assert call_args[1]['num_solutions_per_problem'] == 7
            assert call_args[1]['num_exploratory_ideas'] == 80
            assert call_args[1]['num_new_rule_sets'] == 6
            assert call_args[1]['num_final_solutions'] == 4
