"""Results logging utility for creative reasoning experiments."""

import csv
import os
from pathlib import Path
from typing import Union


def log_results(
    datetime: str,
    algorithm_name: str,
    task_name: str,
    solution: str,
    feasibility_score: Union[float, int],
    utility_score: Union[float, int],
    novelty_score: Union[float, int],
    creativity_score: Union[float, int],
    original_solution_id: str = "default",
    intermediate_log_filename: str = None,
    backbone_llm_name: str = None,
    feasibility_reasoning: str = None,
    utility_reasoning: str = None,
    novelty_theme: str = None,
    num_analogous_problems: int = None,
    num_solutions_per_problem: int = None,
    num_exploratory_ideas: int = None,
    num_new_rule_sets: int = None,
    num_final_solutions: int = None,
    num_solutions_combinational: int = None,
    num_thoughts_per_step: int = None,
    search_depth: int = None
) -> None:
    """Log results to a CSV file.
    
    Args:
        datetime: ISO format timestamp
        algorithm_name: Name of the algorithm used
        task_name: Name of the task solved
        solution: Generated solution text
        feasibility_score: Feasibility score (0.0-1.0)
        utility_score: Utility score (0.0-1.0)
        novelty_score: Novelty score (0.0-1.0)
        creativity_score: Creativity score (0.0-1.0)
        original_solution_id: Unique identifier for the original solution
        intermediate_log_filename: Path to the intermediate log JSON file
        backbone_llm_name: Name of the backbone LLM used
        feasibility_reasoning: LLM's reasoning for the feasibility score
        utility_reasoning: LLM's reasoning for the utility score
        novelty_theme: Main theme or core idea extracted for novelty calculation
        num_analogous_problems: Number of analogous problems used
        num_solutions_per_problem: Number of solutions per analogous problem
        num_exploratory_ideas: Number of exploratory ideas generated
        num_new_rule_sets: Number of new rule sets generated
        num_final_solutions: Number of final solutions generated or selected
        num_solutions_combinational: Number of new solutions synthesized by combinational reasoning
        num_thoughts_per_step: Number of thoughts per step (for tot and egot algorithms)
        search_depth: Search depth for tree-based algorithms
        
    Raises:
        IOError: If there's an error writing to the file
    """
    # Define the results directory and file path
    results_dir = Path(__file__).parent.parent.parent / "results"
    csv_file_path = results_dir / "results.csv"
    
    # Create results directory if it doesn't exist
    try:
        results_dir.mkdir(exist_ok=True)
    except OSError as e:
        raise IOError(f"Failed to create results directory {results_dir}: {e}")
    
    # Define the CSV header
    fieldnames = [
        'datetime',
        'algorithm_name',
        'task_name',
        'solution',
        'feasibility_score',
        'utility_score',
        'novelty_score',
        'creativity_score',
        'original_solution_id',
        'intermediate_log_filename',
        'backbone_llm_name',
        'feasibility_reasoning',
        'utility_reasoning',
        'novelty_theme',
        'num_analogous_problems',
        'num_solutions_per_problem',
        'num_exploratory_ideas',
        'num_new_rule_sets',
        'num_final_solutions',
        'num_solutions_combinational',
        'num_thoughts_per_step',
        'search_depth'
    ]
    
    # Prepare the data row
    data_row = {
        'datetime': datetime,
        'algorithm_name': algorithm_name,
        'task_name': task_name,
        'solution': solution,
        'feasibility_score': feasibility_score,
        'utility_score': utility_score,
        'novelty_score': novelty_score,
        'creativity_score': creativity_score,
        'original_solution_id': original_solution_id,
        'intermediate_log_filename': intermediate_log_filename,
        'backbone_llm_name': backbone_llm_name,
        'feasibility_reasoning': feasibility_reasoning,
        'utility_reasoning': utility_reasoning,
        'novelty_theme': novelty_theme,
        'num_analogous_problems': num_analogous_problems,
        'num_solutions_per_problem': num_solutions_per_problem,
        'num_exploratory_ideas': num_exploratory_ideas,
        'num_new_rule_sets': num_new_rule_sets,
        'num_final_solutions': num_final_solutions,
        'num_solutions_combinational': num_solutions_combinational,
        'num_thoughts_per_step': num_thoughts_per_step,
        'search_depth': search_depth
    }
    
    # Check if file exists and if it has a header
    file_exists = csv_file_path.exists()
    needs_header = True
    
    if file_exists:
        # Check if the file has a header by reading the first line
        try:
            with open(csv_file_path, 'r', encoding='utf-8') as csvfile:
                first_line = csvfile.readline().strip()
                # Check if first line matches our expected header
                expected_header = ','.join(fieldnames)
                needs_header = first_line != expected_header
        except (IOError, UnicodeDecodeError):
            # If we can't read the file, assume it needs a header
            needs_header = True
    
    try:
        # Open file in append mode
        with open(csv_file_path, 'a', newline='', encoding='utf-8') as csvfile:
            writer = csv.DictWriter(csvfile, fieldnames=fieldnames)
            
            # Write header if file is new or doesn't have the correct header
            if not file_exists or needs_header:
                writer.writeheader()
            
            # Write the data row
            writer.writerow(data_row)
            
    except IOError as e:
        raise IOError(f"Failed to write to results file {csv_file_path}: {e}")
