"""Chain of thoughts algorithm implementation for creative reasoning."""

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


class reasoning_model:
    """Chain of thoughts 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_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):
        """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 (ignored for chain_of_thoughts)
            num_solutions_per_problem: Number of solutions per problem (ignored for chain_of_thoughts)
            num_exploratory_ideas: Number of exploratory ideas (ignored for chain_of_thoughts)
            num_new_rule_sets: Number of new rule sets (ignored for chain_of_thoughts)
            num_final_solutions: Number of final solutions to generate
            num_solutions_combinational: Number of combinational solutions (ignored for chain_of_thoughts)
            num_thoughts_per_step: Number of thoughts per step (ignored for chain_of_thoughts)
            search_depth: Search depth (ignored for chain_of_thoughts)
        """
        # 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_final_solutions, int) or num_final_solutions <= 0:
            raise ValueError("num_final_solutions must be a positive integer")
        
        # Store all parameters as instance attributes for consistency
        self.task_description_text = task_config.task_description
        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_exploratory_ideas = num_exploratory_ideas
        self.num_new_rule_sets = num_new_rule_sets
        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 API client
        self.llm_client = LLMAPIClient()
        
        # Initialize intermediate logs collection with detailed structure
        self.intermediate_logs: List[Tuple[str, List[Dict]]] = []
    
    def run(self) -> Tuple[str, List[Tuple[str, List[Dict]]]]:
        """Run the reasoning model to generate a solution with chain-of-thought reasoning.
        
        Returns:
            Tuple of (generated solution text, intermediate logs)
            
        Raises:
            RuntimeError: If the model fails to generate a solution
        """
        try:
            # Generate unique identifiers for logging
            llm_call_id = str(uuid.uuid4())
            timestamp = datetime.datetime.now(datetime.timezone.utc).isoformat()
            
            # Construct the chain-of-thought prompt
            prompt = f"Think step-by-step. First, outline your reasoning, then provide the final solution.\n\n{self.task_description_text}\n\nPlease provide {self.num_final_solutions} distinct creative solutions."
            
            # Make LLM call with specified parameters
            solution_text = self.llm_client.call_llm_model(
                prompt=prompt,
                model_name=self.backbone_llm_name,
                temperature=0.7
            )
            
            if not solution_text or not solution_text.strip():
                raise RuntimeError("Failed to generate solution - empty response from LLM")
            
            # Construct detailed log entry
            llm_call_log_entry = {
                "llm_call_id": llm_call_id,
                "prompt": prompt,
                "raw_response": solution_text,
                "llm_model_name": self.backbone_llm_name,
                "temperature": 0.7,
                "timestamp": timestamp,
                "parsed_output": None  # No complex parsing for single call
            }
            
            # Append to intermediate logs
            self.intermediate_logs.append(("Chain-of-Thought Generation", [llm_call_log_entry]))
                
            return solution_text, self.intermediate_logs
            
        except Exception as e:
            raise RuntimeError(f"Error during solution generation: {e}")