import os
from typing import List, Dict, Any, Optional
from dataclasses import dataclass, asdict
from openai import OpenAI
from dotenv import load_dotenv

load_dotenv()


class memory_refiner:
    def __init__(
            self,
            provider: str = "openai",
            api_key: str = os.getenv("memory_llm_api_key"),
            model: str = os.getenv('memory_llm_model'),
            base_url: str = os.getenv("memory_llm_base_url")
    ):
        self.provider = provider
        self.api_key = api_key
        self.base_url = base_url
        self.model = model

        # Initialize client
        self.client = OpenAI(api_key=self.api_key, base_url=self.base_url)

    def memory_refinement(self, task_goal: str, experience: Dict[str, Any], is_success: bool) -> Dict[str, Any]:
        query = experience['high_level_goal']
        trajectory = experience['low_level_instructions']

        if is_success:
            prompt = f"""
            The following trajectory was successful in addressing the query.
            Distill the key reasoning steps and strategies into a few memory items.
            Each memory item should have a title, a short description, and content.
            Format the output as a JSON string representing a list of dictionaries. For example:
            [
                {{
                    "title": "Example Title",
                    "description": "A short description.",
                    "content": "The detailed reasoning steps."
                }}
            ]

            Query: {query}

            Trajectory:
            {trajectory}
            """
        else:
            prompt = f"""
            The following trajectory failed to address the query.
            Analyze the failure and distill the lessons learned into a few memory items.
            Each memory item should have a title, a short description, and content describing the pitfall and how to avoid it.
            Format the output as a JSON string representing a list of dictionaries. For example:
            {{
                    "title": "Example Pitfall",
                    "description": "A short description of the error.",
                    "content": "A detailed explanation of the mistake and how to avoid it in the future."
            }}

            Query: {query}

            Trajectory:
            {trajectory}
            """


        response = self._call_llm(prompt)


        planning = f"""
        # Role: Task Planner
        ## Instruction
        Please generate specific and executable task planning steps based on the task objectives and historical experience summary.
        """

        result = self._call_llm(planning)


        return result

    def _clean_json_response(self, response: str) -> str:
        """Clean JSON response from LLM"""
        cleaned = response.strip()

        # Extract content after <answer> tag if present
        answer_start = cleaned.find('<answer>')
        if answer_start != -1:
            cleaned = cleaned[answer_start + 8:]
            answer_end = cleaned.find('</answer>')
            if answer_end != -1:
                cleaned = cleaned[:answer_end]

        # Remove markdown code blocks
        if cleaned.startswith("```"):
            parts = cleaned.split("```")
            if len(parts) >= 2:
                cleaned = parts[1]
                if cleaned.startswith("json"):
                    cleaned = cleaned[4:]

        return cleaned.strip()

    def _call_llm(self, prompt: str, max_tokens: int = 2500) -> str:
        """Call LLM API"""
        try:
            response = self.client.chat.completions.create(
                model=self.model,
                messages=[
                    {
                        "role": "system",
                        "content": "You are an expert at analyzing GUI interaction sequences and summarizing user goals. You provide concise, accurate descriptions of what users are trying to accomplish."
                    },
                    {
                        "role": "user",
                        "content": prompt
                    }
                ],
                max_tokens=max_tokens,
                temperature=0.1
            )
            return response.choices[0].message.content
        except Exception as e:
            print(f"OpenAI API call error: {e}")
            return ""