"""Combinational Creative Reasoning algorithm implementation."""

import json
import re
import uuid
from datetime import datetime
from typing import List, Dict, Tuple
from src.data_models.task_config import TaskConfig
from src.utils.llm_api_client import LLMAPIClient
from src.utils.llm_response_parser import extract_json_from_response


class reasoning_model:
    """Combinational Creative Reasoning model for creative reasoning tasks."""
    
    def __init__(self, task_config: TaskConfig, backbone_llm_name: str, num_analogous_problems: int, num_solutions_per_problem: int, num_final_solutions: int = 3, num_solutions_combinational: int = 10):
        """Initialize the 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_final_solutions: Number of final solutions to return
            num_solutions_combinational: Number of new solutions to synthesize
        """
        self.task_config = task_config
        self.backbone_llm_name = backbone_llm_name
        self.num_analogous_problems = num_analogous_problems
        self.num_solutions_per_problem = num_solutions_per_problem
        self.num_final_solutions = num_final_solutions
        self.num_solutions_combinational = num_solutions_combinational
        
        # Initialize LLM API client
        self.llm_client = LLMAPIClient()
        
        # Initialize intermediate logs collection
        self.intermediate_logs: List[Tuple[str, List[Dict]]] = []
        
        # Validate inputs
        if not isinstance(task_config, TaskConfig):
            raise ValueError("task_config must be a TaskConfig object")
        
        if not isinstance(backbone_llm_name, str) or not backbone_llm_name.strip():
            raise ValueError("backbone_llm_name must be a non-empty string")
        
        if not isinstance(num_analogous_problems, int) or num_analogous_problems <= 0:
            raise ValueError("num_analogous_problems must be a positive integer")
        
        if not isinstance(num_solutions_per_problem, int) or num_solutions_per_problem <= 0:
            raise ValueError("num_solutions_per_problem must be a positive integer")
        
        if not isinstance(num_solutions_combinational, int) or num_solutions_combinational <= 0:
            raise ValueError("num_solutions_combinational must be a positive integer")
    
    def run(self) -> Tuple[str, List[Tuple[str, List[Dict]]]]:
        """Run the combinational 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 = self._find_analogous_problems()
            
            # Step 2: Find solutions for each analogous problem
            print("Finding solutions for each analogous problem")
            solutions_per_problem = self._find_solutions_for_problems(analogous_problems)
            
            # Step 3: Decompose all solutions into ideas
            print("Decomposing all solutions into ideas")
            all_ideas = self._decompose_solutions_into_ideas(solutions_per_problem)
            
            # Step 4: Identify impactful ideas from known solutions
            print("Identifying impactful ideas from known solutions")
            impactful_ideas = self._identify_impactful_ideas()
            
            # Step 5: Generate new solutions through creative recombination
            print("Generating new solutions through creative recombination")
            new_solutions = self._generate_new_solutions(impactful_ideas, all_ideas)
            
            # Step 6: Evaluate and rank new solutions
            print("Evaluating and ranking new solutions")
            ranked_solutions = self._evaluate_and_rank_solutions(new_solutions)
            
            # Return top solutions as concatenated string
            print("Formatting final solutions")
            final_solution = self._format_final_solutions(ranked_solutions)
            return final_solution, self.intermediate_logs
            
        except Exception as e:
            raise RuntimeError(f"Error during solution generation: {e}")
    
    def _find_analogous_problems(self) -> List[str]:
        """Find analogous problems that are surface-level different but role-level similar.
        
        Returns:
            List of analogous problem descriptions
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        prompt = f"""
        You are a creative strategist specializing in finding deep, structural analogies from distant domains. Your task is to find {self.num_analogous_problems} problems that are structurally similar to the given task but come from completely different conceptual domains.

**Task:**
{self.task_config.task_description}

---

**Instructions & Thought Process:**
To ensure true creativity, you must follow this process:

1. Your first step is to explicitly state the domain-agnostic principle you have distilled from the main task provided above. Do this before generating any analogies.

2. Second, find analogies to this abstract principle from conceptually distant domains, explicitly avoiding the task's original domain and any closely related fields.

**Crucial Rule:** As a strict rule, if the analogous problem you think of is too similar in its surface features to the original task, you must discard it. For example, if the task is about managing vehicle traffic, an analogy about managing elevator traffic is too close. You must find more abstract analogies.

Return your response as a single JSON array of strings.

Here is an example of the method to follow, using a different, unrelated task for illustration:
* **Hypothetical Task:** Managing access to a single, high-demand telescope for two different research teams.
* **Domain-Agnostic Principle:** A system for arbitrating access to a single, indivisible resource between two competing user groups with high demand.
* **Bad Analogy (Too Close):** "Managing a single university lecture hall for two different classes."
* **Good Analogy (Distant):** "How a 'talking stick' is used in a tribal council to ensure only one person speaks at a time, granting exclusive access to the 'auditory space'."

**Example format for the final output:**
["Problem 1 description from a distant domain", "Problem 2 description from a distant domain", ...]

Return the JSON array now:
        """

        try:
            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(response)
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Collect intermediate log
            self.intermediate_logs.append(("Find Analogous Problems", [llm_call_log_entry]))
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": response if 'response' in locals() else "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Collect error log
            self.intermediate_logs.append(("Find Analogous Problems", [error_log_entry]))
            
            # Fallback: if JSON parsing fails completely, create default problems
            print(f"Warning: JSON parsing failed with error: {e}. Using fallback problems.")
            parsed_output = [f"Analogous problem {i+1}" for i in range(self.num_analogous_problems)]
        
        # Validate and process the parsed output with fallback handling
        if not isinstance(parsed_output, list):
            # Fallback: if response is not a list, create a default list
            print(f"Warning: Expected list but got {type(parsed_output).__name__}. Using fallback list.")
            parsed_output = [f"Analogous problem {i+1}" for i in range(self.num_analogous_problems)]
        
        # Ensure we have the right number of problems
        if len(parsed_output) != self.num_analogous_problems:
            # If we got fewer, pad with placeholders; if more, truncate
            if len(parsed_output) < self.num_analogous_problems:
                parsed_output.extend([f"Additional analogous problem {i+1}" for i in range(self.num_analogous_problems - len(parsed_output))])
            else:
                parsed_output = parsed_output[:self.num_analogous_problems]
        
        return parsed_output
    
    def _find_solutions_for_problems(self, analogous_problems: List[str]) -> Dict[str, List[str]]:
        """Find solutions for each analogous problem in a single LLM call.
        
        Args:
            analogous_problems: List of analogous problem descriptions
            
        Returns:
            Dictionary mapping problem descriptions to their solutions
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        problems_text = "\n".join([f"{i+1}. {problem}" for i, problem in enumerate(analogous_problems)])
        
        prompt = f"""For each of the following {len(analogous_problems)} problems, find {self.num_solutions_per_problem} different solutions:

{problems_text}

Return your response as a JSON object where:
- Keys are the problem numbers (1, 2, 3, etc.)
- Values are arrays of solution strings

For each analogous problem provided, generate solutions that could only emerge from its unique context. Instead of applying generic principles (like "first-come, first-served"), derive mechanisms from the specific properties, constraints, and dynamics inherent to that specific domain.

Example format:
{{
  "1": ["Solution 1 for problem 1", "Solution 2 for problem 1", ...],
  "2": ["Solution 1 for problem 2", "Solution 2 for problem 2", ...],
  ...
}}"""

        try:
            # Use higher max_tokens for finding solutions to prevent truncation
            response = self.llm_client.call_llm_model(
                prompt=prompt,
                model_name=self.backbone_llm_name,
                temperature=0.7,
                max_tokens=8192  # Significantly increased for generating many solutions per problem
            )
            
            # Parse the response
            parsed_output = extract_json_from_response(response)
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Collect intermediate log
            self.intermediate_logs.append(("Find Solutions for Problems", [llm_call_log_entry]))
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": response if 'response' in locals() else "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Collect error log
            self.intermediate_logs.append(("Find Solutions for Problems", [error_log_entry]))
            
            # Fallback: if JSON parsing fails completely, create default solutions
            print(f"Warning: JSON parsing failed with error: {e}. Using fallback solutions.")
            result = {}
            for i, problem in enumerate(analogous_problems):
                result[problem] = [f"Solution {j+1} for {problem}" for j in range(self.num_solutions_per_problem)]
            return result
        
        # Validate and process the parsed output with fallback handling
        if not isinstance(parsed_output, dict):
            # Fallback: if response is not a dictionary, create a default structure
            print(f"Warning: Expected dictionary but got {type(parsed_output).__name__}. Using fallback structure.")
            result = {}
            for i, problem in enumerate(analogous_problems):
                result[problem] = [f"Solution {j+1} for {problem}" for j in range(self.num_solutions_per_problem)]
            return result
        
        # Convert to problem description -> solutions mapping
        result = {}
        for i, problem in enumerate(analogous_problems):
            key = str(i + 1)
            if key in parsed_output and isinstance(parsed_output[key], list):
                result[problem] = parsed_output[key][:self.num_solutions_per_problem]
            else:
                result[problem] = [f"Solution {j+1} for {problem}" for j in range(self.num_solutions_per_problem)]
        
        return result
    
    def _decompose_solutions_into_ideas(self, solutions_per_problem: Dict[str, List[str]]) -> List[str]:
        """Decompose all solutions into ideas in a single LLM call.
        
        Args:
            solutions_per_problem: Dictionary mapping problems to their solutions
            
        Returns:
            List of decomposed ideas
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        # Flatten all solutions into a single text
        all_solutions = []
        for problem, solutions in solutions_per_problem.items():
            all_solutions.append(f"Problem: {problem}")
            for i, solution in enumerate(solutions):
                all_solutions.append(f"  Solution {i+1}: {solution}")
        
        solutions_text = "\n".join(all_solutions)
        
        prompt = f"""
        
        You are a strategic analyst. Your task is to analyze the following solutions and extract the specific, transferable **mechanisms** or **strategic principles** they contain.

**Solutions to Analyze:**
{solutions_text}

---

**Instructions:**
1.  For each solution, identify the core **actionable mechanism** it proposes.
2.  Describe this mechanism as a concise, self-contained principle that could be applied to a different problem.
3.  **Crucially, avoid generic themes.** Focus on the "how" of the solution, not just the "what."

Return your response as a single JSON array of strings, where each string is a distinct, actionable principle.

**Example of what to do:**
* **Input Solution:** "Implement a priority system for elevators based on the number of waiting passengers on each floor."
* **Correct Output:** "A mechanism that dynamically assigns priority based on real-time queue length or demand."

**Example of what to avoid (too generic):**
* **Input Solution:** "Implement a priority system for elevators..."
* **Incorrect Output:** "A priority system."

**Example format:**
["Mechanism or principle 1", "Mechanism or principle 2", "Mechanism or principle 3", ...]

Return the JSON array now:"""

        try:
            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(response)
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Collect intermediate log
            self.intermediate_logs.append(("Decompose Solutions into Ideas", [llm_call_log_entry]))
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": response if 'response' in locals() else "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Collect error log
            self.intermediate_logs.append(("Decompose Solutions into Ideas", [error_log_entry]))
            
            # Fallback: if JSON parsing fails completely, create default ideas
            print(f"Warning: JSON parsing failed with error: {e}. Using fallback ideas.")
            parsed_output = [f"Creative idea {i+1}" for i in range(10)]  # Default 10 ideas
        
        # Validate and process the parsed output with fallback handling
        if not isinstance(parsed_output, list):
            # Fallback: if response is not a list, create a default list
            print(f"Warning: Expected list but got {type(parsed_output).__name__}. Using fallback ideas.")
            parsed_output = [f"Creative idea {i+1}" for i in range(10)]  # Default 10 ideas
        
        # Remove duplicates and ensure we have a reasonable number of ideas
        unique_ideas = list(dict.fromkeys(parsed_output))  # Preserves order while removing duplicates
        return unique_ideas
    
    def _identify_impactful_ideas(self) -> Dict[str, List[str]]:
        """Identify impactful ideas from known solutions in a single LLM call.
        
        Returns:
            Dictionary mapping known solutions to their impactful ideas
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        known_solutions_text = "\n".join([f"{i+1}. {solution}" for i, solution in enumerate(self.task_config.known_solutions)])
        
        prompt = f"""Analyze the following known solutions and decompose each into ideas (smaller pieces). Then, for each solution, choose the ideas that have the most impact on solving the problem and the task described below.

Known solutions:
{known_solutions_text}

Task: {self.task_config.task_description}

Return your response as a JSON object where:
- Keys are the solution numbers (1, 2, 3, etc.)
- Values are arrays of impactful idea strings

Example format:
{{
  "1": ["Impactful idea 1", "Impactful idea 2", ...],
  "2": ["Impactful idea 1", "Impactful idea 2", ...],
  ...
}}"""

        try:
            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(response)
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Collect intermediate log
            self.intermediate_logs.append(("Identify Impactful Ideas", [llm_call_log_entry]))
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": response if 'response' in locals() else "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Collect error log
            self.intermediate_logs.append(("Identify Impactful Ideas", [error_log_entry]))
            
            # Fallback: if JSON parsing fails completely, create default ideas
            print(f"Warning: JSON parsing failed with error: {e}. Using fallback ideas.")
            result = {}
            for i, solution in enumerate(self.task_config.known_solutions):
                result[solution] = [f"Impactful idea for solution {i+1}"]
            return result
        
        # Validate and process the parsed output with fallback handling
        if not isinstance(parsed_output, dict):
            # Fallback: if response is not a dictionary, create a default structure
            print(f"Warning: Expected dictionary but got {type(parsed_output).__name__}. Using fallback structure.")
            result = {}
            for i, solution in enumerate(self.task_config.known_solutions):
                result[solution] = [f"Impactful idea for solution {i+1}"]
            return result
        
        # Convert to solution -> impactful ideas mapping
        result = {}
        for i, solution in enumerate(self.task_config.known_solutions):
            key = str(i + 1)
            if key in parsed_output and isinstance(parsed_output[key], list):
                result[solution] = parsed_output[key]
            else:
                result[solution] = [f"Impactful idea for solution {i+1}"]
        
        return result
    
    def _generate_new_solutions(self, impactful_ideas: Dict[str, List[str]], all_ideas: List[str]) -> List[str]:
        """Generate new solutions through creative recombination in a single LLM call.
        
        Args:
            impactful_ideas: Dictionary mapping known solutions to their impactful ideas
            all_ideas: List of all ideas from analogous problems
            
        Returns:
            List of new creative solutions
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        # Use the specified number of solutions to generate
        num_solutions = self.num_solutions_combinational
        
        # Format impactful ideas
        impactful_ideas_text = "\n".join([f"Solution: {solution}\n  Ideas: {', '.join(ideas)}" for solution, ideas in impactful_ideas.items()])
        
        # Format all ideas from analogous problems
        all_ideas_text = "\n".join([f"- {idea}" for idea in all_ideas])
        
        prompt = f"""
You are a creative strategist. Your task is to generate {num_solutions} new, creative solutions by strictly following a three-step synthesis process for each one.

**Task:**
{self.task_config.task_description}

---

**Pool A: Impactful Ideas from Known Solutions**
{impactful_ideas_text}

**Pool B: Ideas from Analogous Problems**
{all_ideas_text}

---

**Instructions for Generating Each New Solution:**
You must generate a set of {num_solutions} unique and creative solutions. For each solution, you will internally follow this three-step thinking process:
1.  **Step 1: Identify a Pair.** First, select one core idea from Pool A (e.g., "Process vehicles in batches").
2.  **Step 2: Find an Analogy.** Second, find a functionally similar but surface-level different idea from Pool B (e.g., "Reservation system for peak hours").
3.  **Step 3: Synthesize.** Finally, combine these two ideas into a single, new solution that applies the more advanced concept from the analogy to the original problem. The final solution must be a self-contained paragraph.

The new solution must introduce a novel operational logic that doesn't exist in either parent concept, creating an approach that is conceptually distinct from a simple combination.
To create a true hybrid, you must explicitly map the core mechanics of the analogical idea onto the given problem. Your final solution's rules should directly mirror the unique operational logic of the source analogy, not just its high-level theme.

After completing this internal thought process for all {num_solutions} solutions, you will format your final output.

Crucially, ensure that each of the {num_solutions} solutions you generate is based on a fundamentally different strategic approach. Do not generate multiple solutions that are just variations of the same core idea.

Present each solution as a single, self-contained paragraph that functions as an operational blueprint. Focus on describing the core functional mechanism by identifying the essential components, their inputs and outputs, and how they interact to produce the final result. The level of detail must be sufficient to make the process unambiguous and reproducible in principle, but should not be a granular, step-by-step implementation guide. Omit all introductory framing, meta-commentary, or discussion of advantages and disadvantages.

**Your entire response must be a single JSON array of strings, where each string is the final 'synthesized_solution' from Step 3.**

**Example of the internal thinking process leading to one solution:**
* **Step 1 (Idea from Pool A):** "Process vehicles in batches"
* **Step 2 (Idea from Pool B):** "Slot-based system for reserved landing and takeoff times"
* **Step 3 (Synthesized Solution):** "Establish a dynamic reservation system where travelers can pre-book crossing slots in scheduled convoys. On-demand traffic without reservations is then allowed to cross in the dynamically calculated gaps between these scheduled convoys, optimizing for both predictability and efficiency."

**Example format for the final output:**
["New solution 1", "New solution 2", "New solution 3", ...]

Return ONLY the JSON array of the final synthesized solutions now:

"""

        try:
            # Use higher max_tokens for solution generation to prevent truncation
            response = self.llm_client.call_llm_model(
                prompt=prompt,
                model_name=self.backbone_llm_name,
                temperature=0.7,
                max_tokens=8192  # Significantly increased for generating 50 detailed solutions
            )
            
            # Parse the response
            parsed_output = extract_json_from_response(response)
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Collect intermediate log
            self.intermediate_logs.append(("Generate New Solutions", [llm_call_log_entry]))
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": response if 'response' in locals() else "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Collect error log
            self.intermediate_logs.append(("Generate New Solutions", [error_log_entry]))
            
            # Fallback: if JSON parsing fails completely, try to extract solutions from raw response
            print(f"Warning: JSON parsing failed with error: {e}. Attempting to extract solutions from raw response.")
            
            # Try to extract individual solutions from the raw response
            if 'response' in locals() and response:
                import re
                # Look for patterns that suggest individual solutions in the response
                # The response might contain multiple solution descriptions
                solution_patterns = [
                    r'"([^"]*bridge[^"]*)"',  # Solutions mentioning bridge
                    r'"([^"]*system[^"]*)"',  # Solutions mentioning system  
                    r'"([^"]*protocol[^"]*)"',  # Solutions mentioning protocol
                    r'"([^"]*dynamic[^"]*)"',  # Solutions mentioning dynamic
                    r'"([^"]*batch[^"]*)"',  # Solutions mentioning batch
                    r'"([^"]*allocation[^"]*)"',  # Solutions mentioning allocation
                    r'"([^"]*scheduling[^"]*)"',  # Solutions mentioning scheduling
                ]
                
                extracted_solutions = []
                for pattern in solution_patterns:
                    matches = re.findall(pattern, response, re.IGNORECASE)
                    extracted_solutions.extend(matches)
                
                # Remove duplicates and filter by length
                unique_solutions = list(dict.fromkeys(extracted_solutions))
                valid_solutions = [s for s in unique_solutions if len(s) > 20 and len(s) < 500]
                
                if valid_solutions:
                    print(f"Successfully extracted {len(valid_solutions)} solutions from raw response.")
                    parsed_output = valid_solutions[:num_solutions]
                else:
                    print("Could not extract valid solutions from raw response. Using default solutions.")
                    parsed_output = [f"Creative solution {i+1}" for i in range(num_solutions)]
            else:
                print("No raw response available. Using default solutions.")
                parsed_output = [f"Creative solution {i+1}" for i in range(num_solutions)]
        
        # Validate and process the parsed output with fallback handling
        if not isinstance(parsed_output, list):
            # Fallback: if response is not a list, create a default list
            print(f"Warning: Expected list but got {type(parsed_output).__name__}. Using fallback solutions.")
            parsed_output = [f"Creative solution {i+1}" for i in range(num_solutions)]
        
        # Ensure we have the right number of solutions
        if len(parsed_output) < num_solutions:
            parsed_output.extend([f"Additional creative solution {i+1}" for i in range(num_solutions - len(parsed_output))])
        else:
            parsed_output = parsed_output[:num_solutions]
        
        return parsed_output
    
    def _evaluate_and_rank_solutions(self, new_solutions: List[str]) -> List[tuple]:
        """Evaluate and rank new solutions based on feasibility, utility, and novelty.
        
        Args:
            new_solutions: List of new solutions to evaluate
            
        Returns:
            List of tuples (solution, score) sorted by score in descending order
        """
        # Generate unique identifiers for logging
        llm_call_id = str(uuid.uuid4())
        timestamp = datetime.now().isoformat()
        
        solutions_text = "\n".join([f"{i+1}. {solution}" for i, solution in enumerate(new_solutions)])
        
        prompt = f"""Evaluate the following new solutions based on feasibility, utility, and novelty to the task described below. This is for internal self-reflection only - do not use external evaluation tools.

Task: {self.task_config.task_description}

Solutions to evaluate:
{solutions_text}

For each solution, provide a score from 0-100 based on:
- Feasibility (0-30): How practical and implementable is this solution based on feasibility check points?
- Feasibility Check Points:
{self.task_config.feasibility_check_points}

- Utility (0-30): How well does this solution address the problem?
- Novelty (0-40): How creative and original is this solution compared to known solutions?
- Known Solutions:
{self.task_config.known_solutions}

Return your response as a JSON object where:
- Keys are the solution numbers (1, 2, 3, etc.)
- Values are the total scores (0-100)

Example format:
{{
  "1": 85,
  "2": 72,
  "3": 91,
  ...
}}"""

        try:
            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(response)
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "prompt": prompt,
                "raw_response": response,
                "parsed_output": parsed_output,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id
            }
            
            # Collect intermediate log
            self.intermediate_logs.append(("Evaluate and Rank Solutions", [llm_call_log_entry]))
            
        except Exception as e:
            # Construct error log entry
            error_log_entry = {
                "prompt": prompt,
                "raw_response": response if 'response' in locals() else "",
                "parsed_output": None,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "llm_call_id": llm_call_id,
                "error": str(e)
            }
            
            # Collect error log
            self.intermediate_logs.append(("Evaluate and Rank Solutions", [error_log_entry]))
            
            # Fallback: if JSON parsing fails completely, create default scores
            print(f"Warning: JSON parsing failed with error: {e}. Using default scores.")
            ranked_solutions = [(solution, 50.0) for solution in new_solutions]
            return ranked_solutions
        
        # Validate and process the parsed output with fallback handling
        if not isinstance(parsed_output, dict):
            # Fallback: if response is not a dictionary, create default scores
            print(f"Warning: Expected dictionary but got {type(parsed_output).__name__}. Using default scores.")
            ranked_solutions = [(solution, 50.0) for solution in new_solutions]
            return ranked_solutions
        
        # Create list of (solution, score) tuples
        ranked_solutions = []
        for i, solution in enumerate(new_solutions):
            key = str(i + 1)
            score = parsed_output.get(key, 50)  # Default score of 50 if not found
            if not isinstance(score, (int, float)):
                score = 50
            ranked_solutions.append((solution, float(score)))
        
        # Sort by score in descending order
        ranked_solutions.sort(key=lambda x: x[1], reverse=True)
        return ranked_solutions
    
    def _format_final_solutions(self, ranked_solutions: List[tuple]) -> str:
        """Format the top-ranked solutions into a final string.
        
        Args:
            ranked_solutions: List of (solution, score) tuples sorted by score
            
        Returns:
            Formatted string containing the top solutions
        """
        # Get top solutions based on num_final_solutions
        top_solutions = ranked_solutions[:self.num_final_solutions]
        
        # Format the output
        result_lines = []
        for i, (solution, score) in enumerate(top_solutions, 1):
            result_lines.append(f"Solution {i} (Score: {score:.1f}):")
            result_lines.append(solution)
            result_lines.append("")  # Empty line for readability
        
        return "\n".join(result_lines)
