"""Unit tests for results logger."""

import pytest
import csv
import tempfile
import os
from pathlib import Path
from unittest.mock import patch, mock_open
from src.utils.results_logger import log_results


class TestResultsLogger:
    """Test cases for results logger."""
    
    def test_log_results_with_intermediate_log_filename(self):
        """Test logging results with intermediate log filename."""
        with tempfile.TemporaryDirectory() as temp_dir:
            # Mock the results directory path
            with patch('src.utils.results_logger.Path') as mock_path:
                mock_results_dir = Path(temp_dir)
                mock_csv_path = mock_results_dir / "results.csv"
                mock_path.return_value.parent.parent.parent.__truediv__.return_value = mock_results_dir
                
                # Call log_results with intermediate_log_filename
                log_results(
                    datetime="2024-01-01T12:00:00",
                    algorithm_name="test_algorithm",
                    task_name="test_task",
                    solution="Test solution",
                    feasibility_score=0.8,
                    utility_score=0.9,
                    novelty_score=0.7,
                    creativity_score=0.8,
                    original_solution_id="sol_1",
                    intermediate_log_filename="test_run_intermediate_log.json"
                )
                
                # Verify CSV file was created
                assert mock_csv_path.exists()
                
                # Read and verify CSV content
                with open(mock_csv_path, 'r', newline='', encoding='utf-8') as csvfile:
                    reader = csv.DictReader(csvfile)
                    rows = list(reader)
                    
                    assert len(rows) == 1
                    row = rows[0]
                    assert row['datetime'] == "2024-01-01T12:00:00"
                    assert row['algorithm_name'] == "test_algorithm"
                    assert row['task_name'] == "test_task"
                    assert row['solution'] == "Test solution"
                    assert row['feasibility_score'] == "0.8"
                    assert row['utility_score'] == "0.9"
                    assert row['novelty_score'] == "0.7"
                    assert row['creativity_score'] == "0.8"
                    assert row['original_solution_id'] == "sol_1"
                    assert row['intermediate_log_filename'] == "test_run_intermediate_log.json"
    
    def test_log_results_without_intermediate_log_filename(self):
        """Test logging results without intermediate log filename."""
        with tempfile.TemporaryDirectory() as temp_dir:
            # Mock the results directory path
            with patch('src.utils.results_logger.Path') as mock_path:
                mock_results_dir = Path(temp_dir)
                mock_csv_path = mock_results_dir / "results.csv"
                mock_path.return_value.parent.parent.parent.__truediv__.return_value = mock_results_dir
                
                # Call log_results without intermediate_log_filename
                log_results(
                    datetime="2024-01-01T12:00:00",
                    algorithm_name="test_algorithm",
                    task_name="test_task",
                    solution="Test solution",
                    feasibility_score=0.8,
                    utility_score=0.9,
                    novelty_score=0.7,
                    creativity_score=0.8,
                    original_solution_id="sol_1"
                )
                
                # Verify CSV file was created
                assert mock_csv_path.exists()
                
                # Read and verify CSV content
                with open(mock_csv_path, 'r', newline='', encoding='utf-8') as csvfile:
                    reader = csv.DictReader(csvfile)
                    rows = list(reader)
                    
                    assert len(rows) == 1
                    row = rows[0]
                    assert row['intermediate_log_filename'] == ""
    
    def test_log_results_multiple_entries(self):
        """Test logging multiple results entries."""
        with tempfile.TemporaryDirectory() as temp_dir:
            # Mock the results directory path
            with patch('src.utils.results_logger.Path') as mock_path:
                mock_results_dir = Path(temp_dir)
                mock_csv_path = mock_results_dir / "results.csv"
                mock_path.return_value.parent.parent.parent.__truediv__.return_value = mock_results_dir
                
                # Log first entry
                log_results(
                    datetime="2024-01-01T12:00:00",
                    algorithm_name="test_algorithm",
                    task_name="test_task",
                    solution="Test solution 1",
                    feasibility_score=0.8,
                    utility_score=0.9,
                    novelty_score=0.7,
                    creativity_score=0.8,
                    original_solution_id="sol_1",
                    intermediate_log_filename="test_run_intermediate_log.json"
                )
                
                # Log second entry
                log_results(
                    datetime="2024-01-01T12:01:00",
                    algorithm_name="test_algorithm",
                    task_name="test_task",
                    solution="Test solution 2",
                    feasibility_score=0.9,
                    utility_score=0.8,
                    novelty_score=0.8,
                    creativity_score=0.9,
                    original_solution_id="sol_2",
                    intermediate_log_filename="test_run_intermediate_log.json"
                )
                
                # Verify CSV file was created
                assert mock_csv_path.exists()
                
                # Read and verify CSV content
                with open(mock_csv_path, 'r', newline='', encoding='utf-8') as csvfile:
                    reader = csv.DictReader(csvfile)
                    rows = list(reader)
                    
                    assert len(rows) == 2
                    
                    # Check first row
                    row1 = rows[0]
                    assert row1['solution'] == "Test solution 1"
                    assert row1['feasibility_score'] == "0.8"
                    assert row1['original_solution_id'] == "sol_1"
                    
                    # Check second row
                    row2 = rows[1]
                    assert row2['solution'] == "Test solution 2"
                    assert row2['feasibility_score'] == "0.9"
                    assert row2['original_solution_id'] == "sol_2"
                    
                    # Both should have the same intermediate log filename
                    assert row1['intermediate_log_filename'] == "test_run_intermediate_log.json"
                    assert row2['intermediate_log_filename'] == "test_run_intermediate_log.json"
    
    def test_log_results_with_hyperparameters(self):
        """Test logging results with algorithm hyperparameters."""
        with tempfile.TemporaryDirectory() as temp_dir:
            # Mock the results directory path
            with patch('src.utils.results_logger.Path') as mock_path:
                mock_results_dir = Path(temp_dir)
                mock_csv_path = mock_results_dir / "results.csv"
                mock_path.return_value.parent.parent.parent.__truediv__.return_value = mock_results_dir
                
                # Call log_results with hyperparameters
                log_results(
                    datetime="2024-01-01T12:00:00",
                    algorithm_name="test_algorithm",
                    task_name="test_task",
                    solution="Test solution",
                    feasibility_score=0.8,
                    utility_score=0.9,
                    novelty_score=0.7,
                    creativity_score=0.8,
                    original_solution_id="sol_1",
                    intermediate_log_filename="test_run_intermediate_log.json",
                    backbone_llm_name="gpt-4",
                    feasibility_reasoning="Test feasibility reasoning",
                    utility_reasoning="Test utility reasoning",
                    novelty_theme="Test novelty theme",
                    num_analogous_problems=15,
                    num_solutions_per_problem=8,
                    num_exploratory_ideas=75,
                    num_new_rule_sets=5,
                    num_final_solutions=4
                )
                
                # Verify CSV file was created
                assert mock_csv_path.exists()
                
                # Read and verify CSV content
                with open(mock_csv_path, 'r', newline='', encoding='utf-8') as csvfile:
                    reader = csv.DictReader(csvfile)
                    rows = list(reader)
                    
                    assert len(rows) == 1
                    row = rows[0]
                    
                    # Check original fields
                    assert row['datetime'] == "2024-01-01T12:00:00"
                    assert row['algorithm_name'] == "test_algorithm"
                    assert row['task_name'] == "test_task"
                    assert row['solution'] == "Test solution"
                    assert row['feasibility_score'] == "0.8"
                    assert row['utility_score'] == "0.9"
                    assert row['novelty_score'] == "0.7"
                    assert row['creativity_score'] == "0.8"
                    assert row['original_solution_id'] == "sol_1"
                    assert row['intermediate_log_filename'] == "test_run_intermediate_log.json"
                    assert row['backbone_llm_name'] == "gpt-4"
                    assert row['feasibility_reasoning'] == "Test feasibility reasoning"
                    assert row['utility_reasoning'] == "Test utility reasoning"
                    assert row['novelty_theme'] == "Test novelty theme"
                    
                    # Check new hyperparameter fields
                    assert row['num_analogous_problems'] == "15"
                    assert row['num_solutions_per_problem'] == "8"
                    assert row['num_exploratory_ideas'] == "75"
                    assert row['num_new_rule_sets'] == "5"
                    assert row['num_final_solutions'] == "4"
    
    def test_log_results_with_none_hyperparameters(self):
        """Test logging results with None hyperparameters (backward compatibility)."""
        with tempfile.TemporaryDirectory() as temp_dir:
            # Mock the results directory path
            with patch('src.utils.results_logger.Path') as mock_path:
                mock_results_dir = Path(temp_dir)
                mock_csv_path = mock_results_dir / "results.csv"
                mock_path.return_value.parent.parent.parent.__truediv__.return_value = mock_results_dir
                
                # Call log_results without hyperparameters (should use None defaults)
                log_results(
                    datetime="2024-01-01T12:00:00",
                    algorithm_name="test_algorithm",
                    task_name="test_task",
                    solution="Test solution",
                    feasibility_score=0.8,
                    utility_score=0.9,
                    novelty_score=0.7,
                    creativity_score=0.8,
                    original_solution_id="sol_1"
                )
                
                # Verify CSV file was created
                assert mock_csv_path.exists()
                
                # Read and verify CSV content
                with open(mock_csv_path, 'r', newline='', encoding='utf-8') as csvfile:
                    reader = csv.DictReader(csvfile)
                    rows = list(reader)
                    
                    assert len(rows) == 1
                    row = rows[0]
                    
                    # Check that hyperparameter fields are empty (None values become empty strings in CSV)
                    assert row['num_analogous_problems'] == ""
                    assert row['num_solutions_per_problem'] == ""
                    assert row['num_exploratory_ideas'] == ""
                    assert row['num_new_rule_sets'] == ""
                    assert row['num_final_solutions'] == ""