"""Main entry point for the Creative Reasoning project."""

import argparse
import importlib
import sys
import os
import json
import uuid
import csv
import shutil
import pandas as pd
from datetime import datetime
from pathlib import Path
from typing import Tuple, List

# Add the src directory to Python path for imports
src_dir = Path(__file__).parent
if str(src_dir) not in sys.path:
    sys.path.insert(0, str(src_dir))

# Load environment variables
try:
    from dotenv import load_dotenv
    load_dotenv()
except ImportError:
    # Fallback: manually load .env file
    env_path = Path(__file__).parent.parent / ".env"
    if env_path.exists():
        with open(env_path, 'r') as f:
            for line in f:
                line = line.strip()
                if line and not line.startswith('#') and '=' in line:
                    key, value = line.split('=', 1)
                    os.environ[key] = value

from src.evaluators.run_evaluation import run_evaluation
from src.utils.results_logger import log_results
from src.data_models.task_config import TaskConfig


def load_task_description(task_name: str) -> str:
    """Load task description from the tasks directory.
    
    Args:
        task_name: Name of the task file (without .txt extension)
        
    Returns:
        Task description as a string
        
    Raises:
        FileNotFoundError: If the task file doesn't exist
    """
    task_file_path = Path(__file__).parent / "tasks" / f"{task_name}.txt"
    
    if not task_file_path.exists():
        raise FileNotFoundError(f"Task file not found: {task_file_path}")
    
    try:
        with open(task_file_path, 'r', encoding='utf-8') as f:
            return f.read().strip()
    except IOError as e:
        raise IOError(f"Failed to read task file {task_file_path}: {e}")


def load_feasibility_check_points(task_name: str) -> list[str]:
    """Load feasibility check points from the feasibility_check_points directory.
    
    Args:
        task_name: Name of the task (without .txt extension)
        
    Returns:
        List of feasibility check points
        
    Raises:
        FileNotFoundError: If the feasibility check points file doesn't exist
    """
    check_points_file_path = Path(__file__).parent / "feasibility_check_points" / f"{task_name}.txt"
    
    if not check_points_file_path.exists():
        raise FileNotFoundError(f"Feasibility check points file not found: {check_points_file_path}")
    
    try:
        with open(check_points_file_path, 'r', encoding='utf-8') as f:
            # Read lines and filter out empty lines
            check_points = [line.strip() for line in f.readlines() if line.strip()]
            return check_points
    except IOError as e:
        raise IOError(f"Failed to read feasibility check points file {check_points_file_path}: {e}")


def load_known_solutions(task_name: str) -> list[str]:
    """Load known solutions from the known_solutions directory.
    
    Args:
        task_name: Name of the task (without .txt extension)
        
    Returns:
        List of known solutions
        
    Raises:
        FileNotFoundError: If the known solutions file doesn't exist
    """
    solutions_file_path = Path(__file__).parent / "known_solutions" / f"{task_name}.txt"
    
    if not solutions_file_path.exists():
        raise FileNotFoundError(f"Known solutions file not found: {solutions_file_path}")
    
    try:
        with open(solutions_file_path, 'r', encoding='utf-8') as f:
            # Read lines and filter out empty lines
            solutions = [line.strip() for line in f.readlines() if line.strip()]
            return solutions
    except IOError as e:
        raise IOError(f"Failed to read known solutions file {solutions_file_path}: {e}")


def load_known_solutions_concept(task_name: str) -> list[str]:
    """Load known solution concepts from the known_solutions_concept directory.
    
    Args:
        task_name: Name of the task (without .txt extension)
        
    Returns:
        List of known solution concepts
        
    Raises:
        FileNotFoundError: If the known solutions concept file doesn't exist
        IOError: If the file cannot be read
    """
    concepts_file_path = Path(__file__).parent / "known_solutions_concept" / f"{task_name}.txt"
    
    if not concepts_file_path.exists():
        raise FileNotFoundError(f"Known solutions concept file not found: {concepts_file_path}")
    
    try:
        with open(concepts_file_path, 'r', encoding='utf-8') as f:
            # Read lines and filter out empty lines and comments
            concepts = [line.strip() for line in f.readlines() if line.strip() and not line.strip().startswith('#')]
            return concepts
    except IOError as e:
        raise IOError(f"Failed to read known solutions concept file {concepts_file_path}: {e}")


def load_calibration_anchors(task_name: str) -> list[str]:
    """Load calibration anchors from the calibration_anchors directory.
    
    Args:
        task_name: Name of the task (without .txt extension)
        
    Returns:
        List of calibration anchor entries
        
    Raises:
        FileNotFoundError: If the calibration anchors file doesn't exist
        IOError: If the file cannot be read
    """
    anchors_file_path = Path(__file__).parent / "calibration_anchors" / f"{task_name}.txt"
    
    if not anchors_file_path.exists():
        raise FileNotFoundError(f"Calibration anchors file not found: {anchors_file_path}")
    
    try:
        with open(anchors_file_path, 'r', encoding='utf-8') as f:
            # Read lines and filter out empty lines
            anchors = [line.strip() for line in f.readlines() if line.strip()]
            return anchors
    except IOError as e:
        raise IOError(f"Failed to read calibration anchors file {anchors_file_path}: {e}")


def load_optimal_solutions(task_name: str) -> Tuple[List[str], List[str]]:
    """Load optimal solutions from the optimal_solutions directory.
    
    Args:
        task_name: Name of the task (without .csv extension)
        
    Returns:
        Tuple of (solution_descriptions, concept_descriptions) as lists of strings
        
    Raises:
        FileNotFoundError: If the optimal solutions CSV file doesn't exist
        IOError: If the file cannot be read or parsed
        ValueError: If required columns are missing from the CSV
    """
    optimal_solutions_file_path = Path(__file__).parent / "optimal_solutions" / f"{task_name}.csv"
    
    if not optimal_solutions_file_path.exists():
        raise FileNotFoundError(f"Optimal solutions file not found: {optimal_solutions_file_path}")
    
    try:
        solution_descriptions = []
        concept_descriptions = []
        
        with open(optimal_solutions_file_path, 'r', encoding='utf-8') as f:
            reader = csv.DictReader(f)
            
            # Check if required columns exist
            if 'solution description' not in reader.fieldnames:
                raise ValueError("Required column 'solution description' not found in CSV")
            if 'concept level description' not in reader.fieldnames:
                raise ValueError("Required column 'concept level description' not found in CSV")
            
            # Read data from CSV
            for row in reader:
                solution_desc = row.get('solution description', '').strip()
                concept_desc = row.get('concept level description', '').strip()
                
                if solution_desc:  # Only add non-empty descriptions
                    solution_descriptions.append(solution_desc)
                if concept_desc:  # Only add non-empty descriptions
                    concept_descriptions.append(concept_desc)
        
        return solution_descriptions, concept_descriptions
        
    except IOError as e:
        raise IOError(f"Failed to read optimal solutions file {optimal_solutions_file_path}: {e}")
    except csv.Error as e:
        raise IOError(f"Failed to parse CSV file {optimal_solutions_file_path}: {e}")


def save_intermediate_logs(intermediate_logs: list, run_id: str) -> str:
    """Save intermediate logs to a JSON file.
    
    Args:
        intermediate_logs: List of (prompt, response_text) tuples
        run_id: Unique identifier for this run
        
    Returns:
        Path to the saved JSON file
        
    Raises:
        IOError: If there's an error writing to the file
    """
    # Define the results directory and file path
    results_dir = Path(__file__).parent.parent / "results"
    log_filename = f"{run_id}_intermediate_log.json"
    log_file_path = results_dir / log_filename
    
    # 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}")
    
    try:
        # Convert intermediate logs to JSON-serializable format
        serializable_logs = []
        for prompt, response_text in intermediate_logs:
            serializable_logs.append({
                "prompt": prompt,
                "response": response_text
            })
        
        # Write to JSON file
        with open(log_file_path, 'w', encoding='utf-8') as f:
            json.dump(serializable_logs, f, indent=2, ensure_ascii=False)
        
        return log_filename
        
    except IOError as e:
        raise IOError(f"Failed to write intermediate log file {log_file_path}: {e}")


def load_and_run_algorithm(algorithm_name: str, task_config: TaskConfig, backbone_llm_name: str, num_analogous_problems: int, num_solutions_per_problem: int, num_exploratory_ideas: int = 50, num_new_rule_sets: int = 3, num_final_solutions: int = 3, num_solutions_combinational: int = 10, num_thoughts_per_step: int = 5, search_depth: int = 3) -> tuple:
    """Dynamically load and run the specified algorithm.
    
    Args:
        algorithm_name: Name of the algorithm to load
        task_config: TaskConfig object containing task information
        backbone_llm_name: Name of the backbone LLM
        num_analogous_problems: Number of analogous problems to find
        num_solutions_per_problem: Number of solutions per analogous problem
        num_exploratory_ideas: Number of exploratory ideas to generate (for exploratory_creative_reasoning)
        num_new_rule_sets: Number of new rule sets to generate (for transformative_creative_reasoning)
        num_final_solutions: Number of final solutions to generate or select
        num_solutions_combinational: Number of new solutions to synthesize (for combinational_creative_reasoning)
        num_thoughts_per_step: Number of thoughts per step (for tot and egot algorithms)
        search_depth: Search depth (for tot and egot algorithms)
        
    Returns:
        Tuple of (solution text, intermediate logs)
        
    Raises:
        ImportError: If the algorithm module cannot be imported
        AttributeError: If the reasoning_model class or run method is missing
        Exception: Any exception raised by the algorithm execution
    """
    try:
        # Dynamically import the algorithm module
        module_path = f"algorithms.{algorithm_name}.main"
        algorithm_module = importlib.import_module(module_path)
        
        # Get the reasoning_model class
        reasoning_model_class = getattr(algorithm_module, 'reasoning_model')
        
        # Instantiate the model with conditional parameter passing
        if algorithm_name == 'exploratory_creative_reasoning':
            model = reasoning_model_class(task_config, backbone_llm_name, num_analogous_problems, num_solutions_per_problem, num_exploratory_ideas, num_final_solutions, num_solutions_combinational)
        elif algorithm_name == 'transformative_creative_reasoning':
            model = reasoning_model_class(task_config, backbone_llm_name, num_new_rule_sets, num_analogous_problems, num_solutions_per_problem, num_exploratory_ideas, num_final_solutions, num_solutions_combinational)
        elif algorithm_name == 'egot':
            # EGoT requires all parameters including num_thoughts_per_step and search_depth
            model = reasoning_model_class(task_config, backbone_llm_name, 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)
        elif algorithm_name == 'commercialized_reasoning_model' or algorithm_name == 'chain_of_thoughts' or algorithm_name == 'tot':
            # These algorithms now accept all 10 standardized parameters
            model = reasoning_model_class(task_config, backbone_llm_name, 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)
        else:
            model = reasoning_model_class(task_config, backbone_llm_name, num_analogous_problems, num_solutions_per_problem, num_final_solutions, num_solutions_combinational)
        
        # Run the model and get solution with intermediate logs
        solution_text, intermediate_logs = model.run()
        
        if not isinstance(solution_text, str):
            raise ValueError("Algorithm must return a string solution")
        
        return solution_text, intermediate_logs
        
    except ImportError as e:
        raise ImportError(f"Failed to import algorithm module '{algorithm_name}': {e}")
    except AttributeError as e:
        raise AttributeError(f"Algorithm module '{algorithm_name}' missing required 'reasoning_model' class: {e}")
    except Exception as e:
        raise Exception(f"Algorithm execution failed: {e}")


def run_evaluation_only_mode():
    """Run evaluation-only mode to re-evaluate existing solutions in results.csv."""
    print("Creative Reasoning Project - Evaluation Only Mode")
    print("=" * 50)
    
    # Define paths
    results_dir = Path(__file__).parent.parent / "results"
    results_csv_path = results_dir / "results.csv"
    
    # Check if results.csv exists
    if not results_csv_path.exists():
        print(f"Error: results.csv not found at {results_csv_path}")
        print("Please run the normal workflow first to generate some results.")
        sys.exit(1)
    
    try:
        # Step 1: Create backup with timestamp
        timestamp = datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
        backup_path = results_dir / f"results_before_evaluation_{timestamp}.csv"
        print(f"Creating backup: {backup_path}")
        shutil.copyfile(results_csv_path, backup_path)
        print(f"Backup created successfully")
        
        # Step 2: Load results.csv into DataFrame
        print("Loading results.csv...")
        df = pd.read_csv(results_csv_path)
        print(f"Loaded {len(df)} rows from results.csv")
        
        if len(df) == 0:
            print("No data found in results.csv. Nothing to re-evaluate.")
            return
        
        # Step 3: Process each row
        print("Starting re-evaluation process...")
        updated_rows = []
        
        for index, row in df.iterrows():
            print(f"\nProcessing row {index + 1}/{len(df)}: {row['task_name']} - {row['algorithm_name']}")
            
            try:
                # Extract data from row
                task_name = row['task_name']
                solution = row['solution']
                algorithm_name = row['algorithm_name']
                
                # Extract run_id from intermediate_log_filename if available
                run_id = str(uuid.uuid4())  # Generate new run_id for re-evaluation
                intermediate_log_filename = row.get('intermediate_log_filename', '')
                if intermediate_log_filename and 'eval_' in intermediate_log_filename:
                    # Extract run_id from filename like "intermediate_logs/eval_7663c613-ba41-4091-b5d1-00a5a5b7fcab_sol_1_c70c013f.json"
                    try:
                        filename_part = intermediate_log_filename.split('/')[-1]  # Get just the filename
                        if filename_part.startswith('eval_'):
                            run_id = filename_part.split('_')[1]  # Extract the UUID part
                    except (IndexError, AttributeError):
                        pass  # Keep the generated run_id
                
                original_solution_id = row.get('original_solution_id', 'default')
                backbone_llm_name = row.get('backbone_llm_name', 'gpt-4o')
                
                # Extract hyperparameters from row
                num_analogous_problems = row.get('num_analogous_problems', 10)
                num_solutions_per_problem = row.get('num_solutions_per_problem', 5)
                num_exploratory_ideas = row.get('num_exploratory_ideas', 50)
                num_new_rule_sets = row.get('num_new_rule_sets', 3)
                num_final_solutions = row.get('num_final_solutions', 3)
                num_solutions_combinational = row.get('num_solutions_combinational', 20)
                num_thoughts_per_step = row.get('num_thoughts_per_step', 5)
                search_depth = row.get('search_depth', 3)
                
                # Step 4: Dynamically load TaskConfig components
                print(f"  Loading TaskConfig for {task_name}...")
                task_description = load_task_description(task_name)
                feasibility_check_points = load_feasibility_check_points(task_name)
                known_solutions = load_known_solutions(task_name)
                
                # Load known solutions concept with fallback
                try:
                    known_solutions_concept = load_known_solutions_concept(task_name)
                except FileNotFoundError:
                    print(f"  Warning: Known solutions concept file not found for {task_name}, using empty list")
                    known_solutions_concept = []
                
                calibration_anchors = load_calibration_anchors(task_name)
                
                # Load optimal solutions with fallback
                try:
                    optimal_solutions_description, optimal_solutions_concept = load_optimal_solutions(task_name)
                except FileNotFoundError:
                    print(f"  Warning: Optimal solutions file not found for {task_name}, using empty lists")
                    optimal_solutions_description = []
                    optimal_solutions_concept = []
                
                # Create TaskConfig object
                task_config = TaskConfig(
                    feasibility_check_points=feasibility_check_points,
                    task_description=task_description,
                    known_solutions=known_solutions,
                    known_solutions_concept=known_solutions_concept,
                    calibration_anchors=calibration_anchors,
                    optimal_solutions_description=optimal_solutions_description,
                    optimal_solutions_concept=optimal_solutions_concept
                )
                
                # Step 5: Prepare pre-extracted solutions with themes for efficiency
                pre_extracted_solutions_with_themes = [{
                    'solution_text': solution,
                    'original_solution_id': original_solution_id,
                    'novelty_theme': row.get('novelty_theme', '')
                }]
                
                # Step 6: Run evaluation
                print(f"  Running evaluation for solution...")
                evaluation_results = run_evaluation(
                    solution_text=solution,
                    task_config=task_config,
                    run_id=run_id,
                    num_final_solutions=1,
                    pre_extracted_solutions_with_themes=pre_extracted_solutions_with_themes
                )
                
                # Step 7: Update rows with new evaluation results
                current_timestamp = datetime.now().isoformat()
                
                for result in evaluation_results:
                    updated_row = row.copy()
                    updated_row['datetime'] = current_timestamp
                    updated_row['feasibility_score'] = result.feasibility_score
                    updated_row['utility_score'] = result.utility_score
                    updated_row['novelty_score'] = result.novelty_score
                    updated_row['creativity_score'] = result.creativity_score
                    updated_row['feasibility_reasoning'] = result.feasibility_reasoning
                    updated_row['utility_reasoning'] = result.utility_reasoning
                    updated_row['novelty_theme'] = result.novelty_theme
                    updated_row['intermediate_log_filename'] = result.intermediate_log_filename
                    updated_row['original_solution_id'] = result.original_solution_id
                    updated_row['run_id'] = run_id
                    
                    updated_rows.append(updated_row)
                
                print(f"  ✅ Evaluation completed - {len(evaluation_results)} solutions evaluated")
                
            except Exception as e:
                print(f"  ❌ Error processing row {index + 1}: {e}")
                # Keep original row if evaluation fails
                updated_rows.append(row)
                continue
        
        # Step 8: Save updated results
        if updated_rows:
            print(f"\nSaving updated results to {results_csv_path}...")
            updated_df = pd.DataFrame(updated_rows)
            updated_df.to_csv(results_csv_path, index=False)
            print(f"✅ Results saved successfully - {len(updated_df)} rows updated")
        else:
            print("No rows were successfully processed.")
        
        print("\n" + "=" * 50)
        print("Evaluation-only mode completed successfully!")
        
    except FileNotFoundError as e:
        print(f"Error: {e}")
        sys.exit(1)
    except Exception as e:
        print(f"Unexpected error during re-evaluation: {e}")
        sys.exit(1)


def main():
    """Main function to run the Creative Reasoning project."""
    parser = argparse.ArgumentParser(description="Creative Reasoning Project - Main Entry Point")
    parser.add_argument("--rerun-evaluation", action="store_true", help="Re-evaluate existing solutions in results.csv")
    parser.add_argument("--task-name", help="Name of the task to solve")
    parser.add_argument("--algorithm-name", help="Name of the algorithm to use")
    parser.add_argument("--backbone-llm-name", help="Name of the backbone LLM")
    parser.add_argument("--num-analogous-problems", type=int, default=10, help="Number of analogous problems to find")
    parser.add_argument("--num-solutions-per-problem", type=int, default=5, help="Number of solutions per analogous problem")
    parser.add_argument("--num-exploratory-ideas", type=int, default=50, help="Number of exploratory ideas to generate for exploratory creative reasoning")
    parser.add_argument("--num-new-rule-sets", type=int, default=3, help="Number of new rule sets to generate for transformative creative reasoning")
    parser.add_argument("--num-final-solutions", type=int, default=3, help="Number of final solutions to generate or select")
    parser.add_argument("--num-solutions-combinational", type=int, default=20, help="Number of new solutions to synthesize by the Combinational Creative Reasoning algorithm")
    parser.add_argument("--num-thoughts-per-step", type=int, default=5, help="Number of thoughts per step (for tot and egot algorithms)")
    parser.add_argument("--search-depth", type=int, default=3, help="Search depth (for tot and egot algorithms)")
    
    args = parser.parse_args()
    
    # Handle evaluation-only mode
    if args.rerun_evaluation:
        return run_evaluation_only_mode()
    
    # Validate required arguments for normal mode
    if not args.task_name:
        print("Error: --task-name is required for normal execution mode")
        sys.exit(1)
    if not args.algorithm_name:
        print("Error: --algorithm-name is required for normal execution mode")
        sys.exit(1)
    if not args.backbone_llm_name:
        print("Error: --backbone-llm-name is required for normal execution mode")
        sys.exit(1)
    
    try:
        print(f"Creative Reasoning Project - Starting workflow")
        print(f"Task: {args.task_name}")
        print(f"Algorithm: {args.algorithm_name}")
        print(f"Backbone LLM: {args.backbone_llm_name}")
        print("-" * 50)
        
        # Step 1: Load task configuration
        print("Loading task configuration...")
        task_description = load_task_description(args.task_name)
        feasibility_check_points = load_feasibility_check_points(args.task_name)
        known_solutions = load_known_solutions(args.task_name)
        
        # Load known solutions concept with fallback to empty list if file doesn't exist
        try:
            known_solutions_concept = load_known_solutions_concept(args.task_name)
        except FileNotFoundError:
            print(f"Warning: Known solutions concept file not found for {args.task_name}, using empty list")
            known_solutions_concept = []
        
        calibration_anchors = load_calibration_anchors(args.task_name)
        
        # Load optimal solutions with fallback to empty lists if file doesn't exist
        try:
            optimal_solutions_description, optimal_solutions_concept = load_optimal_solutions(args.task_name)
        except FileNotFoundError:
            print(f"Warning: Optimal solutions file not found for {args.task_name}, using empty lists")
            optimal_solutions_description = []
            optimal_solutions_concept = []
        
        # Create TaskConfig object
        task_config = TaskConfig(
            feasibility_check_points=feasibility_check_points,
            task_description=task_description,
            known_solutions=known_solutions,
            known_solutions_concept=known_solutions_concept,
            calibration_anchors=calibration_anchors,
            optimal_solutions_description=optimal_solutions_description,
            optimal_solutions_concept=optimal_solutions_concept
        )
        print(f"Task configuration loaded successfully")
        print(f"  - Task description: {len(task_description)} characters")
        print(f"  - Feasibility check points: {len(feasibility_check_points)}")
        print(f"  - Known solutions: {len(known_solutions)}")
        print(f"  - Known solutions concept: {len(known_solutions_concept)}")
        print(f"  - Calibration Anchors: {len(calibration_anchors)}")
        print(f"  - Optimal solutions description: {len(optimal_solutions_description)}")
        print(f"  - Optimal solutions concept: {len(optimal_solutions_concept)}")
        
        # Step 2: Run algorithm
        print("Executing algorithm...")
        solution_text, intermediate_logs = load_and_run_algorithm(
            args.algorithm_name, 
            task_config, 
            args.backbone_llm_name,
            args.num_analogous_problems,
            args.num_solutions_per_problem,
            args.num_exploratory_ideas,
            args.num_new_rule_sets,
            args.num_final_solutions,
            args.num_solutions_combinational,
            args.num_thoughts_per_step,
            args.search_depth
        )
        print(f"Solution generated successfully ({len(solution_text)} characters)")
        
        # Step 2.5: Generate run ID and save intermediate logs (except for EGoT)
        run_id = str(uuid.uuid4())
        if args.algorithm_name == 'egot':
            print("Skipping intermediate logs for EGoT algorithm...")
            intermediate_log_filename = None
        else:
            print("Saving intermediate logs...")
            intermediate_log_filename = save_intermediate_logs(intermediate_logs, run_id)
            print(f"Intermediate logs saved to: {intermediate_log_filename}")
        
        # Step 3: Run evaluation
        print("Running evaluation...")
        evaluation_results = run_evaluation(solution_text, task_config, run_id, args.num_final_solutions)
        print(f"Evaluation completed successfully - {len(evaluation_results)} solutions evaluated")
        
        # Step 4: Log results
        print("Logging results...")
        timestamp = datetime.now().isoformat()
        
        # Log each evaluation result separately
        for result in evaluation_results:
            # For EGoT, use None for intermediate_log_filename regardless of result
            log_intermediate_filename = None if args.algorithm_name == 'egot' else result.intermediate_log_filename
            print(f"Logging result: {result.individual_solution_text}")
            print(f"Feasibility score: {result.feasibility_score}")
            print(f"Utility score: {result.utility_score}")
            print(f"Novelty theme: {result.novelty_theme}")
            print(f"Novelty score: {result.novelty_score}")
            print(f"Creativity score: {result.creativity_score}")
            log_results(
                datetime=timestamp,
                algorithm_name=args.algorithm_name,
                task_name=args.task_name,
                solution=result.individual_solution_text,
                feasibility_score=result.feasibility_score,
                utility_score=result.utility_score,
                novelty_score=result.novelty_score,
                creativity_score=result.creativity_score,
                original_solution_id=result.original_solution_id,
                intermediate_log_filename=log_intermediate_filename,
                backbone_llm_name=args.backbone_llm_name,
                feasibility_reasoning=result.feasibility_reasoning,
                utility_reasoning=result.utility_reasoning,
                novelty_theme=result.novelty_theme,
                num_analogous_problems=args.num_analogous_problems,
                num_solutions_per_problem=args.num_solutions_per_problem,
                num_exploratory_ideas=args.num_exploratory_ideas,
                num_new_rule_sets=args.num_new_rule_sets,
                num_final_solutions=args.num_final_solutions,
                num_solutions_combinational=args.num_solutions_combinational,
                num_thoughts_per_step=args.num_thoughts_per_step,
                search_depth=args.search_depth
            )
        
        print("Results logged successfully")
        print("-" * 50)
        print("Workflow completed successfully!")
        
    except FileNotFoundError as e:
        print(f"Error: {e}")
        sys.exit(1)
    except ImportError as e:
        print(f"Error: {e}")
        sys.exit(1)
    except AttributeError as e:
        print(f"Error: {e}")
        sys.exit(1)
    except Exception as e:
        print(f"Unexpected error: {e}")
        sys.exit(1)


if __name__ == "__main__":
    main()
