"""Unit tests for data models."""

import pytest
from src.data_models.task_config import TaskConfig
from src.data_models.evaluation_result import EvaluationResult


class TestTaskConfig:
    """Test cases for TaskConfig model."""
    
    def test_valid_task_config(self):
        """Test creating a valid TaskConfig."""
        config = TaskConfig(
            feasibility_check_points=["Check 1", "Check 2"],
            task_description="Test task",
            known_solutions=["Solution 1", "Solution 2"],
            calibration_anchors=["Anchor 1", "Anchor 2"]
        )
        
        assert len(config.feasibility_check_points) == 2
        assert config.task_description == "Test task"
        assert len(config.known_solutions) == 2
        assert len(config.calibration_anchors) == 2
    
    def test_valid_task_config_without_calibration_anchors(self):
        """Test creating a valid TaskConfig without calibration_anchors (should default to empty list)."""
        config = TaskConfig(
            feasibility_check_points=["Check 1", "Check 2"],
            task_description="Test task",
            known_solutions=["Solution 1", "Solution 2"]
        )
        
        assert len(config.feasibility_check_points) == 2
        assert config.task_description == "Test task"
        assert len(config.known_solutions) == 2
        assert config.calibration_anchors == []  # Should default to empty list
    
    def test_empty_lists(self):
        """Test TaskConfig with empty lists."""
        config = TaskConfig(
            feasibility_check_points=[],
            task_description="Test task",
            known_solutions=[],
            calibration_anchors=[]
        )
        
        assert len(config.feasibility_check_points) == 0
        assert len(config.known_solutions) == 0
        assert len(config.calibration_anchors) == 0
    
    def test_calibration_anchors_field(self):
        """Test TaskConfig calibration_anchors field specifically."""
        # Test with non-empty calibration anchors
        config = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            calibration_anchors=["Score 4/10: Example 1", "Score 8/10: Example 2"]
        )
        
        assert config.calibration_anchors == ["Score 4/10: Example 1", "Score 8/10: Example 2"]
        
        # Test with empty calibration anchors
        config_empty = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            calibration_anchors=[]
        )
        
        assert config_empty.calibration_anchors == []
    
    def test_known_solutions_concept_field(self):
        """Test TaskConfig known_solutions_concept field specifically."""
        # Test with non-empty known_solutions_concept
        config = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            known_solutions_concept=["Concept 1", "Concept 2"],
            calibration_anchors=[]
        )
        
        assert config.known_solutions_concept == ["Concept 1", "Concept 2"]
        
        # Test with empty known_solutions_concept
        config_empty = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            known_solutions_concept=[],
            calibration_anchors=[]
        )
        
        assert config_empty.known_solutions_concept == []
        
        # Test without known_solutions_concept (should default to empty list)
        config_default = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            calibration_anchors=[]
        )
        
        assert config_default.known_solutions_concept == []
    
    def test_known_solutions_concept_optional_nature(self):
        """Test that known_solutions_concept is optional and defaults to empty list."""
        # Test that TaskConfig can be created without known_solutions_concept
        config = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"]
        )
        
        # Should default to empty list
        assert config.known_solutions_concept == []
        assert isinstance(config.known_solutions_concept, list)
    
    def test_optimal_solutions_description_field(self):
        """Test TaskConfig optimal_solutions_description field specifically."""
        # Test with non-empty optimal_solutions_description
        config = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            optimal_solutions_description=["Optimal Solution 1", "Optimal Solution 2"],
            calibration_anchors=[]
        )
        
        assert config.optimal_solutions_description == ["Optimal Solution 1", "Optimal Solution 2"]
        
        # Test with empty optimal_solutions_description
        config_empty = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            optimal_solutions_description=[],
            calibration_anchors=[]
        )
        
        assert config_empty.optimal_solutions_description == []
        
        # Test without optimal_solutions_description (should default to empty list)
        config_default = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            calibration_anchors=[]
        )
        
        assert config_default.optimal_solutions_description == []
    
    def test_optimal_solutions_concept_field(self):
        """Test TaskConfig optimal_solutions_concept field specifically."""
        # Test with non-empty optimal_solutions_concept
        config = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            optimal_solutions_concept=["Optimal Concept 1", "Optimal Concept 2"],
            calibration_anchors=[]
        )
        
        assert config.optimal_solutions_concept == ["Optimal Concept 1", "Optimal Concept 2"]
        
        # Test with empty optimal_solutions_concept
        config_empty = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            optimal_solutions_concept=[],
            calibration_anchors=[]
        )
        
        assert config_empty.optimal_solutions_concept == []
        
        # Test without optimal_solutions_concept (should default to empty list)
        config_default = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"],
            calibration_anchors=[]
        )
        
        assert config_default.optimal_solutions_concept == []
    
    def test_optimal_solutions_fields_optional_nature(self):
        """Test that optimal_solutions fields are optional and default to empty lists."""
        # Test that TaskConfig can be created without optimal_solutions fields
        config = TaskConfig(
            feasibility_check_points=["Check 1"],
            task_description="Test task",
            known_solutions=["Solution 1"]
        )
        
        # Should default to empty lists
        assert config.optimal_solutions_description == []
        assert config.optimal_solutions_concept == []
        assert isinstance(config.optimal_solutions_description, list)
        assert isinstance(config.optimal_solutions_concept, list)
    
    def test_all_fields_preserved_with_new_fields(self):
        """Test that existing fields are preserved when new optimal_solutions fields are added."""
        config = TaskConfig(
            feasibility_check_points=["Check 1", "Check 2"],
            task_description="Test task",
            known_solutions=["Solution 1", "Solution 2"],
            known_solutions_concept=["Concept 1", "Concept 2"],
            calibration_anchors=["Anchor 1", "Anchor 2"],
            optimal_solutions_description=["Optimal Desc 1", "Optimal Desc 2"],
            optimal_solutions_concept=["Optimal Concept 1", "Optimal Concept 2"]
        )
        
        # Verify all existing fields are preserved
        assert config.feasibility_check_points == ["Check 1", "Check 2"]
        assert config.task_description == "Test task"
        assert config.known_solutions == ["Solution 1", "Solution 2"]
        assert config.known_solutions_concept == ["Concept 1", "Concept 2"]
        assert config.calibration_anchors == ["Anchor 1", "Anchor 2"]
        
        # Verify new fields are present
        assert config.optimal_solutions_description == ["Optimal Desc 1", "Optimal Desc 2"]
        assert config.optimal_solutions_concept == ["Optimal Concept 1", "Optimal Concept 2"]


class TestEvaluationResult:
    """Test cases for EvaluationResult model."""
    
    def test_valid_evaluation_result(self):
        """Test creating a valid EvaluationResult."""
        result = EvaluationResult(
            original_solution_id="sol_1_abc123",
            individual_solution_text="This is an individual solution.",
            feasibility_score=0.8,
            utility_score=0.9,
            novelty_score=0.7,
            creativity_score=0.8
        )
        
        assert result.original_solution_id == "sol_1_abc123"
        assert result.individual_solution_text == "This is an individual solution."
        assert result.feasibility_score == 0.8
        assert result.utility_score == 0.9
        assert result.novelty_score == 0.7
        assert result.creativity_score == 0.8
    
    def test_score_boundaries(self):
        """Test that scores are properly bounded."""
        # Test that Pydantic validation rejects out-of-bounds scores
        with pytest.raises(ValueError):
            EvaluationResult(
                original_solution_id="sol_1_abc123",
                individual_solution_text="Test solution",
                feasibility_score=1.5,  # Should be rejected (> 1.0)
                utility_score=0.5,
                novelty_score=0.5,
                creativity_score=0.5
            )
        
        with pytest.raises(ValueError):
            EvaluationResult(
                original_solution_id="sol_1_abc123",
                individual_solution_text="Test solution",
                feasibility_score=0.5,
                utility_score=-0.5,     # Should be rejected (< 0.0)
                novelty_score=0.5,
                creativity_score=0.5
            )
        
        # Test that valid scores are accepted
        result = EvaluationResult(
            original_solution_id="sol_1_abc123",
            individual_solution_text="Test solution",
            feasibility_score=1.0,  # Valid boundary value
            utility_score=0.0,      # Valid boundary value
            novelty_score=0.5,
            creativity_score=0.5
        )
        
        assert result.feasibility_score == 1.0
        assert result.utility_score == 0.0
        assert result.novelty_score == 0.5
        assert result.creativity_score == 0.5
    
    def test_intermediate_log_filename_field(self):
        """Test the intermediate_log_filename field."""
        # Test with intermediate_log_filename provided
        result = EvaluationResult(
            original_solution_id="sol_1_abc123",
            individual_solution_text="Test solution",
            feasibility_score=0.8,
            utility_score=0.9,
            novelty_score=0.7,
            creativity_score=0.8,
            intermediate_log_filename="test_run_intermediate_log.json"
        )
        
        assert result.intermediate_log_filename == "test_run_intermediate_log.json"
        
        # Test without intermediate_log_filename (should default to None)
        result = EvaluationResult(
            original_solution_id="sol_1_abc123",
            individual_solution_text="Test solution",
            feasibility_score=0.8,
            utility_score=0.9,
            novelty_score=0.7,
            creativity_score=0.8
        )
        
        assert result.intermediate_log_filename is None
    
    def test_missing_individual_solution_text(self):
        """Test that EvaluationResult instantiation fails when individual_solution_text is omitted."""
        with pytest.raises(ValueError):
            EvaluationResult(
                original_solution_id="sol_1_abc123",
                feasibility_score=0.8,
                utility_score=0.9,
                novelty_score=0.7,
                creativity_score=0.8
            )