"""Transformative 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_api_client import LLMAPIClient
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:
    """Transformative Creative Reasoning model that orchestrates rule mutation and utilizes Exploratory Creative Reasoning."""
    
    def __init__(self, task_config: TaskConfig, backbone_llm_name: str, num_new_rule_sets: int, 
                 num_analogous_problems: int, num_solutions_per_problem: int, num_exploratory_ideas: int, num_final_solutions: int = 3, num_solutions_combinational: int = 10, num_thoughts_per_step: int = 5, search_depth: int = 3):
        """Initialize the transformative reasoning model.
        
        Args:
            task_config: TaskConfig object containing task information
            backbone_llm_name: Name of the backbone LLM to use
            num_new_rule_sets: Number of new rule sets to generate
            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
            num_final_solutions: Number of final solutions to select
            num_solutions_combinational: Number of new solutions to synthesize by the combinational reasoning
            num_thoughts_per_step: Number of thoughts per step (for consistency with other algorithms)
            search_depth: Search depth (for consistency with other algorithms)
            
        Raises:
            ValueError: If any parameter is invalid
        """
        # Validate inputs
        if not isinstance(task_config, TaskConfig):
            raise ValueError("task_config must be a TaskConfig object")
        
        if not backbone_llm_name or not backbone_llm_name.strip():
            raise ValueError("backbone_llm_name must be a non-empty string")
        
        if not isinstance(num_new_rule_sets, int) or num_new_rule_sets <= 0:
            raise ValueError("num_new_rule_sets must be a positive integer")
        
        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_exploratory_ideas, int) or num_exploratory_ideas <= 0:
            raise ValueError("num_exploratory_ideas 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")
        
        if not isinstance(num_thoughts_per_step, int) or num_thoughts_per_step <= 0:
            raise ValueError("num_thoughts_per_step must be a positive integer")
        
        if not isinstance(search_depth, int) or search_depth <= 0:
            raise ValueError("search_depth must be a positive integer")
        
        # Store parameters
        self.task_config = task_config
        self.backbone_llm_name = backbone_llm_name
        self.num_new_rule_sets = num_new_rule_sets
        self.num_analogous_problems = num_analogous_problems
        self.num_solutions_per_problem = num_solutions_per_problem
        self.num_exploratory_ideas = num_exploratory_ideas
        self.num_final_solutions = num_final_solutions
        self.num_solutions_combinational = num_solutions_combinational
        self.num_thoughts_per_step = num_thoughts_per_step
        self.search_depth = search_depth
        
        # Initialize LLM client and logging
        self.llm_client = LLMAPIClient()
        self.intermediate_logs: List[Tuple[str, List[Dict]]] = []
        
        # Instantiate CombinationalReasoningModel for reusing evaluation and formatting methods
        self.combinational_reasoning_instance = CombinationalReasoningModel(
            task_config=self.task_config,
            backbone_llm_name=self.backbone_llm_name,
            num_analogous_problems=self.num_analogous_problems,
            num_solutions_per_problem=self.num_solutions_per_problem,
            num_final_solutions=self.num_final_solutions,
            num_solutions_combinational=self.num_solutions_combinational
        )
    
    def _expose_rules_and_assumptions(self) -> List[str]:
        """Identify all explicit and hidden rules and assumptions for the target problem.
        
        Returns:
            List of identified rules and assumptions
            
        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
        prompt = f"""Given the following task description, identify ALL explicit and hidden rules, constraints, and assumptions that govern this problem.

Task Description:
{self.task_config.task_description}

Please analyze this task and identify:
1. Explicit rules and constraints that are clearly stated
2. Hidden assumptions and implicit rules that are not explicitly mentioned but are necessary for the problem to make sense
3. Domain-specific constraints that might not be obvious
4. Any underlying principles or guidelines that would affect how this problem should be approached

Return your response as a JSON array of strings, where each string is a distinct rule, constraint, or assumption.

Example format:
["Rule/assumption 1", "Rule/assumption 2", "Rule/assumption 3", ...]

Be thorough and comprehensive in your analysis."""

        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
            rules_and_assumptions = []
            for item in parsed_output:
                if isinstance(item, str):
                    rules_and_assumptions.append(item)
                else:
                    # Convert non-string items to strings
                    rules_and_assumptions.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(("Expose Rules and Assumptions", [llm_call_log_entry]))
            
            return rules_and_assumptions
            
        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(("Expose Rules and Assumptions", [error_log_entry]))
            
            # Re-raise the exception
            raise RuntimeError(f"Failed to expose rules and assumptions: {e}")
    
    def _find_analogous_rules(self, original_rule_set: List[str]) -> Dict[str, List[str]]:
        """Find analogous rules from diverse domains for each identified rule.
        
        Args:
            original_rule_set: List of original rules and assumptions
            
        Returns:
            Dictionary mapping each original rule to a list of analogous rules from different domains
            
        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
        rules_text = "\n".join([f"- {rule}" for rule in original_rule_set])
        
        prompt = f"""Given the following original rules and assumptions from the target problem, find analogous rules from diverse domains for each rule.

Original Rules and Assumptions:
{rules_text}

For each original rule, find 3-5 analogous rules from completely different domains that serve similar functional purposes. Consider domains like:
- Biology and natural systems
- Computer science and algorithms
- Architecture and engineering
- Economics and business
- Psychology and human behavior
- Physics and mechanics
- Art and design
- Mathematics and logic
- And any other relevant domains

The analogous rules should:
1. Serve a similar functional role as the original rule
2. Come from domains that are quite different from the original problem domain
3. Be specific and actionable rules, not just general principles
4. Be diverse - don't use the same domain for multiple analogous rules

Return your response as a JSON object where:
- Keys are the original rules (exact strings from the input)
- Values are arrays of analogous rules from different domains

Example format:
{{
    "Original rule 1": ["Analogous rule from biology", "Analogous rule from computer science", "Analogous rule from architecture"],
    "Original rule 2": ["Analogous rule from economics", "Analogous rule from psychology", "Analogous rule from physics"]
}}"""

        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 dictionary
            if not isinstance(parsed_output, dict):
                raise ValueError("LLM response is not a dictionary")
            
            # Ensure all values are lists of strings
            analogous_rules_map = {}
            for original_rule, analogous_rules in parsed_output.items():
                if isinstance(analogous_rules, list):
                    # Convert all items to strings
                    analogous_rules_map[original_rule] = [str(rule) for rule in analogous_rules]
                else:
                    # If not a list, wrap in a list and convert to string
                    analogous_rules_map[original_rule] = [str(analogous_rules)]
            
            # 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(("Find Analogous Rules", [llm_call_log_entry]))
            
            return analogous_rules_map
            
        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(("Find Analogous Rules", [error_log_entry]))
            
            # Re-raise the exception
            raise RuntimeError(f"Failed to find analogous rules: {e}")
    
    def _generate_new_rule_sets(self, original_rule_set: List[str], analogous_rules_map: Dict[str, List[str]]) -> List[List[str]]:
        """Generate multiple new rule sets by applying mutation operations using analogous rules.
        
        Args:
            original_rule_set: List of original rules and assumptions
            analogous_rules_map: Dictionary mapping original rules to analogous rules from different domains
            
        Returns:
            List of new rule sets, where each rule set is a list of rules
            
        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
        original_rules_text = "\n".join([f"- {rule}" for rule in original_rule_set])
        
        analogous_rules_text = ""
        for original_rule, analogous_rules in analogous_rules_map.items():
            analogous_rules_text += f"\nOriginal: {original_rule}\n"
            analogous_rules_text += "Analogous:\n"
            for analogous_rule in analogous_rules:
                analogous_rules_text += f"  - {analogous_rule}\n"
        
        prompt = f"""Given the original rule set and analogous rules from diverse domains, generate {self.num_new_rule_sets} new rule sets by applying mutation operations.

Original Rule Set:
{original_rules_text}

Analogous Rules from Diverse Domains:
{analogous_rules_text}

For each new rule set, apply one or more of these mutation operations:
1. DROP: Remove one or more original rules
2. VARY: Replace one or more original rules with analogous rules from different domains
3. ADD: Add new rules from the analogous rules to the original set

CRITICAL CONSTRAINT: Make sure new rule sets allow feasible solutions. The mutated rules should still make sense together and not create impossible constraints.

Each new rule set should:
- Be distinct from the original and from other generated rule sets
- Maintain logical consistency within the rule set
- Allow for feasible solutions to be generated
- Incorporate insights from diverse domains through the analogous rules

Return your response as a JSON array, where each element is a JSON array of rule strings representing a new rule set.

Example format:
[
    ["Modified rule 1", "Modified rule 2", "New rule from biology"],
    ["Original rule 1", "Analogous rule from computer science", "Dropped rule 3"],
    ["Completely new rule set with mixed domains"]
]

Generate exactly {self.num_new_rule_sets} new rule sets."""

        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 are lists of strings
            new_rule_sets = []
            for rule_set in parsed_output:
                if isinstance(rule_set, list):
                    # Convert all items to strings
                    new_rule_sets.append([str(rule) for rule in rule_set])
                else:
                    # If not a list, wrap in a list and convert to string
                    new_rule_sets.append([str(rule_set)])
            
            # Pad or truncate to match num_new_rule_sets
            while len(new_rule_sets) < self.num_new_rule_sets:
                # Add a placeholder rule set if we don't have enough
                new_rule_sets.append(["Generated placeholder rule set"])
            
            # Truncate if we have too many
            new_rule_sets = new_rule_sets[:self.num_new_rule_sets]
            
            # 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(("Generate New Rule Sets", [llm_call_log_entry]))
            
            return new_rule_sets
            
        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(("Generate New Rule Sets", [error_log_entry]))
            
            # Re-raise the exception
            raise RuntimeError(f"Failed to generate new rule sets: {e}")
    
    def run(self) -> Tuple[str, List[Tuple[str, List[Dict]]]]:
        """Run the transformative creative reasoning algorithm.
        
        Returns:
            Tuple of (final solution text, comprehensive intermediate logs)
            
        Raises:
            RuntimeError: If the algorithm fails to generate solutions
        """
        try:
            # Step 1: Expose rules and assumptions
            print("Step 1: Exposing rules and assumptions")
            original_rule_set = self._expose_rules_and_assumptions()
            
            # Step 2: Find analogous rules from diverse domains
            print("Step 2: Finding analogous rules from diverse domains")
            analogous_rules_map = self._find_analogous_rules(original_rule_set)
            
            # Step 3: Generate new rule sets through mutation
            print("Step 3: Generating new rule sets through mutation")
            new_rule_sets = self._generate_new_rule_sets(original_rule_set, analogous_rules_map)
            
            # Step 4: Run Exploratory Creative Reasoning for each new rule set
            all_generated_solutions: List[str] = []
            all_intermediate_exploratory_logs: List[Tuple[str, List[Dict]]] = []
            
            # Import the exploratory reasoning model
            from algorithms.exploratory_creative_reasoning.main import reasoning_model as ExploratoryReasoningModel
            
            for i, new_rule_set in enumerate(new_rule_sets):
                print(f"New Rule ID: {i}")
                # Create modified task config with new rule set
                # Format the new rule set for clarity.
                new_rules_formatted = "\n".join([f"- {rule}" for rule in new_rule_set])

                # --- This is the new, more robust introductory text ---
                introductory_text = (
                    "--- \n"
                    "**A Creative Reframing of the Problem:**\n\n"
                    "Your primary goal is to solve the task described above. Your final solution **must** be fully compliant with all of the original task's hard constraints (e.g., no new infrastructure, safety protocols must be met).\n\n"
                    "However, to push beyond conventional solutions, you are to approach this problem through a new creative lens. Think of the following 'analogous' rules not as replacements, but as a set of principles to guide your thinking toward more innovative strategies. Your final solution should be a novel policy that is valid in the original problem space but is clearly inspired by the logic and structure of these new rules."
                    "\n\n**New Analogous Rules for Creative Inspiration:**"
                )


                # Craft the final, more effective task description.
                modified_task_description = (
                    f"{self.task_config.task_description}\n\n"
                    f"{introductory_text}\n"
                    f"{new_rules_formatted}"
                )
                modified_task_config = TaskConfig(
                    feasibility_check_points=self.task_config.feasibility_check_points,
                    task_description=modified_task_description,
                    known_solutions=self.task_config.known_solutions
                )
                
                # Instantiate and run exploratory reasoning model
                exploratory_model = ExploratoryReasoningModel(
                    task_config=modified_task_config,
                    backbone_llm_name=self.backbone_llm_name,
                    num_analogous_problems=self.num_analogous_problems,
                    num_solutions_per_problem=self.num_solutions_per_problem,
                    num_exploratory_ideas=self.num_exploratory_ideas,
                    num_final_solutions=self.num_final_solutions,
                    num_solutions_combinational=self.num_solutions_combinational
                )
                
                # Run the exploratory model
                exploratory_solutions_str, exploratory_logs = exploratory_model.run()
                
                # Extract individual solutions from the exploratory results
                # For now, we'll treat the entire string as one solution
                # In a more sophisticated implementation, we might parse individual solutions
                all_generated_solutions.append(exploratory_solutions_str)
                
                # Add exploratory logs with a prefix to identify the rule set
                for log_entry in exploratory_logs:
                    all_intermediate_exploratory_logs.append((f"Rule Set {i+1}: {log_entry[0]}", log_entry[1]))
            
            # Step 5: Evaluate and rank final solutions using CombinationalReasoningModel
            ranked_solutions = self.combinational_reasoning_instance._evaluate_and_rank_solutions(all_generated_solutions)
            
            # Step 6: Format final solutions using CombinationalReasoningModel
            final_solutions_text = self.combinational_reasoning_instance._format_final_solutions(ranked_solutions)
            
            # Step 7: Aggregate logs from CombinationalReasoningModel instance
            combinational_logs = []
            for log_entry in self.combinational_reasoning_instance.intermediate_logs:
                combinational_logs.append((f"Transformative: {log_entry[0]}", log_entry[1]))
            
            # Step 8: Combine all intermediate logs
            combined_intermediate_logs = self.intermediate_logs + all_intermediate_exploratory_logs + combinational_logs
            
            return final_solutions_text, combined_intermediate_logs
            
        except Exception as e:
            raise RuntimeError(f"Error during transformative creative reasoning: {e}")
