"""Exploratory Creative Reasoning algorithm implementation."""

import uuid
from datetime import datetime
from typing import List, Dict, Tuple
from src.data_models.task_config import TaskConfig
from src.utils.llm_response_parser import extract_json_from_response
from src.algorithms.combinational_creative_reasoning.main import reasoning_model as CombinationalReasoningModel


class reasoning_model(CombinationalReasoningModel):
    """Exploratory Creative Reasoning model that extends combinational reasoning with idea expansion."""
    
    def __init__(self, task_config: TaskConfig, backbone_llm_name: str, num_analogous_problems: int, num_solutions_per_problem: int, num_exploratory_ideas: int = 50, num_final_solutions: int = 3, num_solutions_combinational: int = 10):
        """Initialize the exploratory reasoning model.
        
        Args:
            task_config: TaskConfig object containing task information
            backbone_llm_name: Name of the backbone LLM to use
            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 (default: 50)
            num_final_solutions: Number of final solutions to return
            num_solutions_combinational: Number of new solutions to synthesize by the combinational reasoning
        """
        # Call parent constructor
        super().__init__(task_config, backbone_llm_name, num_analogous_problems, num_solutions_per_problem, num_final_solutions, num_solutions_combinational)
        
        # Store exploratory ideas parameter
        self.num_exploratory_ideas = num_exploratory_ideas
        
        # Update intermediate_logs type hint to match project spec
        self.intermediate_logs: List[Tuple[str, List[Dict]]] = []
        
        # Validate num_exploratory_ideas
        if not isinstance(num_exploratory_ideas, int) or num_exploratory_ideas <= 0:
            raise ValueError("num_exploratory_ideas must be a positive integer")
    
    def run(self) -> Tuple[str, List[Tuple[str, List[Dict]]]]:
        """Run the exploratory creative reasoning algorithm.
        
        Returns:
            Tuple of (generated solution text, intermediate logs)
            
        Raises:
            RuntimeError: If the algorithm fails to generate solutions
        """
        try:
            # Step 1: Find analogous problems
            print("Finding analogous problems")
            analogous_problems = super()._find_analogous_problems()
            
            # Step 2: Find solutions for each analogous problem
            print("Finding solutions for each analogous problem")
            solutions_per_problem = super()._find_solutions_for_problems(analogous_problems)
            
            # Step 3: Decompose all solutions into ideas
            print("Decomposing all solutions into ideas")
            all_ideas = super()._decompose_solutions_into_ideas(solutions_per_problem)
            
            # Step 4: Expand ideas exploratorily (NEW STEP)
            print("Expanding ideas exploratorily")
            new_exploratory_ideas = self._expand_ideas_exploratorily(all_ideas)
            
            # Step 5: Extend all_ideas with new exploratory ideas
            print("Extending all_ideas with new exploratory ideas")
            all_ideas.extend(new_exploratory_ideas)
            
            # Step 6: Identify impactful ideas from known solutions
            print("Identifying impactful ideas from known solutions")
            impactful_ideas = super()._identify_impactful_ideas()
            
            # Step 7: Generate new solutions through creative recombination
            print("Generating new solutions through creative recombination")
            new_solutions = super()._generate_new_solutions(impactful_ideas, all_ideas)
            
            # Step 8: Evaluate and rank new solutions
            print("Evaluating and ranking new solutions")
            ranked_solutions = super()._evaluate_and_rank_solutions(new_solutions)
            
            # Step 9: Return top solutions as concatenated string
            print("Formatting final solutions")
            final_solution = super()._format_final_solutions(ranked_solutions)
            return final_solution, self.intermediate_logs
            
        except Exception as e:
            raise RuntimeError(f"Error during exploratory solution generation: {e}")
    
    def _expand_ideas_exploratorily(self, all_ideas: List[str]) -> List[str]:
        """Generate new ideas that are far in surface level but similar in role/functionality.
        
        Args:
            all_ideas: List of existing ideas to expand upon
            
        Returns:
            List of new exploratory ideas
            
        Raises:
            RuntimeError: If LLM call fails or response parsing fails
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        # Construct the prompt
        ideas_text = "\n".join([f"- {idea}" for idea in all_ideas])
        
        prompt = f"""Given the following task and existing ideas, generate {self.num_exploratory_ideas} new ideas that are far in surface level (i.e., domain) but have similar role/functionality.

Task: {self.task_config.task_description}

Existing ideas:
{ideas_text}

For each new idea, think of concepts from completely different domains that serve similar functional roles. For example, if the task is about transportation, consider ideas from biology, computer science, architecture, etc. that solve similar underlying problems.

Return your response as a JSON array of strings, where each string is a new exploratory idea.

Example format:
["New exploratory idea 1", "New exploratory idea 2", "New exploratory idea 3", ...]"""

        try:
            # Make LLM call
            raw_response = self.llm_client.call_llm_model(
                prompt=prompt,
                model_name=self.backbone_llm_name,
                temperature=0.7
            )
            
            # Parse the response
            parsed_output = extract_json_from_response(raw_response)
            
            # Validate that the response is a list
            if not isinstance(parsed_output, list):
                raise ValueError("LLM response is not a list")
            
            # Ensure all items in the list are strings
            new_ideas = []
            for item in parsed_output:
                if isinstance(item, str):
                    new_ideas.append(item)
                else:
                    # Convert non-string items to strings
                    new_ideas.append(str(item))
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": raw_response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Append to intermediate logs
            self.intermediate_logs.append(("Exploratory Idea Expansion", [llm_call_log_entry]))
            
            return new_ideas
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Append error to intermediate logs
            self.intermediate_logs.append(("Exploratory Idea Expansion", [error_log_entry]))
            
            # Re-raise the exception without fallback
            raise RuntimeError(f"Failed to expand ideas exploratorily: {e}")
