"""
Core module for Qwen Code Python implementation.
"""
import os
import json
import sys
import glob
from typing import List, Dict, Any, Union
from openai import OpenAI
from datetime import datetime
from tqdm import tqdm
from dataclasses import dataclass
import shutil

sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))

from utils import SessionLogger, parse_json_response_with_codeblock, llm_generation
from tools import get_all_tools
from .system_prompt import (
    get_core_system_prompt, 
    get_compression_prompt, 
    get_planning_prompt, 
    get_validation_prompt, 
    get_summary_prompt
)
from .tool_registry import ToolRegistry
from .templates import TEMPLATES, safe_copy_template, build_chosen_template
from .compress_output import compress_output

RETRY_LIMIT = 5

@dataclass
class AgentConfig:
    instruction: str
    model: str
    working_dir: str
    log_dir: str
    max_history_length: int = 20
    max_iterations: int = 50
    overwrite: bool = False
    max_tokens: int = 8192
    compression_ratio: float = 0.5
    max_validation_num: int = 5
    max_summary_retry: int = 5
    backend_ratio: float = 0.5
    clear_backend: bool = False
    template_names: str | None = None


class WebGenAgent2V1:
    """The first version of FullStack-Agent"""
    
    def __init__(self, config: AgentConfig):
        self.config = config
        self.llm_generation = llm_generation
        self.history: List[Dict[str, Any]] = []
        self.tool_call_history: List[Dict[str, Any]] = []  # New attribute to track all tool calls
        self.max_history_length = self.config.max_history_length
        self.max_backend_iterations = round(self.config.max_iterations * self.config.backend_ratio)
        # Create a new session logger for this agent instance
        self.session_logger = SessionLogger(self.config.log_dir)
        self.validation_num = 0
        self.backend_summary = ""
        self.is_frontend = False
        self.is_pure_frontend = False

        self.initialize_tool_registry()

    def initialize_tool_registry(self):
        self.tools = get_all_tools(self.config.working_dir, self.config.log_dir)
        self.registry = ToolRegistry()
        for tool in self.tools:
            self.registry.register_tool(tool)

    def import_template(self, chosen_template_name: str = None):
        """Import a template based on the instruction and copy it to the working directory."""
        if chosen_template_name is None:
            # Create a prompt to choose the template
            template_descriptions = "Backend Templates:\n" + "\n".join([
                f"- {k}: {v}" 
                for k, v in TEMPLATES["template_descriptions"]["backend_templates"].items()
            ])
            template_descriptions += "\n\nFrontend Templates:\n" + "\n".join([
                f"- {k}: {v}" 
                for k, v in TEMPLATES["template_descriptions"]["frontend_templates"].items()
            ])

            prompt = f"""Based on the user's instruction, choose the most appropriate frontend and backend template from the available options.

Instruction: {self.config.instruction}

Available templates:
{template_descriptions}

Respond with a JSON object containing the name of the template you choose and whether the project is pure frontend. The output should strictly follow the format:
```json
{{
    "frontend_template_name": [frontend template name (string)],
    "backend_template_name": [backend template name (string)],
}}
```"""

            messages = [
                {"role": "system", "content": "You are an expert at choosing the most appropriate web development template based on user requirements. Respond with only the name of the template you choose."},
                {"role": "user", "content": prompt}
            ]
            
            response = self.llm_generation(messages, model=self.config.model)

            request_data = {
                "model": self.config.model,
                "messages": messages.copy(),
                "is_pure_frontend": self.is_pure_frontend
            }
            self.session_logger.log_llm_request_response(0, "import_template", request_data, response)

            try:
                response_text = response.get("content", "").strip()
                response_dict = parse_json_response_with_codeblock(response_text)
                chosen_template_name = response_dict.get("frontend_template_name", "") + "__" + response_dict.get("backend_template_name", "")
                # self.is_pure_frontend = response_dict.get("is_pure_frontend", False)
                self.is_pure_frontend = False
            except:
                chosen_template_name = ""
                self.is_pure_frontend = False
        
        # Find the chosen template
        chosen_template, chosen_template_name = build_chosen_template(chosen_template_name)
                    
        self.session_logger.log_message(f"Chosen template: {chosen_template['name']}")

        # Copy template contents to working directory
        frontend_name, backend_name = chosen_template["name"].split("__")
        frontend_template_source = os.path.join(TEMPLATES["root_dir"], frontend_name)
        backend_template_source = os.path.join(TEMPLATES["root_dir"], backend_name)
        common_files_dir = os.path.join(TEMPLATES["root_dir"], "common_files")
        copy_success = safe_copy_template(frontend_template_source, backend_template_source, common_files_dir, chosen_template, self.config.working_dir, max_retries=5)
        if not copy_success:
            raise RuntimeError(f"Failed to copy frontend template {frontend_name} to {self.config.working_dir}")

        return chosen_template

    def get_plans(self) -> Union[Dict[str, Any], None]:
        """Get the backend and frontend plans based on the instruction."""
        planning_prompt = get_planning_prompt(self.config.instruction, self.is_pure_frontend)
        messages = [
            {"role": "system", "content": "You are a senior full-stack software architect."},
            {"role": "user", "content": planning_prompt}
        ]
        
        for i in range(RETRY_LIMIT):
            self.session_logger.log_message(f"Planning attempt {i+1}")
            try:
                response = self.llm_generation(messages, model=self.config.model, max_tokens=self.config.max_tokens)

                request_data = {
                    "model": self.config.model,
                    "messages": messages.copy(),
                    "is_pure_frontend": self.is_pure_frontend
                }
                self.session_logger.log_llm_request_response(0, "get_plans", request_data, response)

                content = response.get("content", "")
                # Attempt to parse the response as JSON
                plans = parse_json_response_with_codeblock(content)
                if plans is None:
                    self.session_logger.log_message("Failed to parse planning response as JSON")
                if (self.is_pure_frontend and "backendPlan" not in plans and "frontendPlan" in plans) or (not self.is_pure_frontend and "backendPlan" in plans and "frontendPlan" in plans):
                    return plans
                else:
                    self.session_logger.log_message("Planning response does not contain both 'backendPlan' and 'frontendPlan'")
            except json.JSONDecodeError as e:
                self.session_logger.log_message(f"Failed to parse planning response as JSON: {str(e)}")
            except Exception as e:
                self.session_logger.log_message(f"Error during planning: {str(e)}")
        else:
            self.session_logger.log_message("Exceeded maximum planning attempts")
            raise RuntimeError(f"Failed to obtain valid plans after {RETRY_LIMIT} attempts")

    def initialize_template_and_plan(self):
        """Initialize the working directory with the chosen template."""
        # load previous chosen template if it exists
        chosen_template_name = None
        template_file = os.path.join(self.config.log_dir, "chosen_template.json")
        plan_file = os.path.join(self.config.log_dir, "plans.json")
        if os.path.exists(template_file):
            try:
                with open(template_file, "r", encoding="utf-8") as f:
                    chosen_template = json.load(f)
                chosen_template_name = chosen_template['chosen_template_name']
                self.session_logger.log_message(f"Loaded previously chosen template: {chosen_template_name}")
            except Exception as e:
                self.session_logger.log_message(f"Failed to load chosen template from log: {str(e)}")
        self.chosen_template = self.import_template(chosen_template_name)
        with open(template_file, "w", encoding="utf-8") as f:
            json.dump({"chosen_template_name": self.chosen_template["name"]}, f, indent=4)
        
        # load previous plans if they exist
        if os.path.exists(plan_file):
            try:
                with open(plan_file, "r", encoding="utf-8") as f:
                    self.plans = json.load(f)
                self.session_logger.log_message("Loaded previously saved plans")
            except Exception as e:
                self.session_logger.log_message(f"Failed to load plans from log: {str(e)}")
                self.plans = self.get_plans()
        else:
            self.plans = self.get_plans()
        with open(plan_file, "w", encoding="utf-8") as f:
            json.dump(self.plans, f, indent=4)

    def initialize_agent(self) -> int:
        """Initialize the client with system context. If previous logs exist, restart from the last step."""
        # Check if we should restart from a previous run
        log_dir = self.config.log_dir
        start_step = 0
        if not self.config.overwrite and os.path.exists(log_dir):
            # Look for existing tool response files to determine the last completed step
            tool_response_files = glob.glob(os.path.join(log_dir, "*_tool_response.json"))
            if tool_response_files:
                # Extract step numbers and sort them in descending order
                step_numbers = []
                for file_path in tool_response_files:
                    filename = os.path.basename(file_path)
                    try:
                        step_num = int(filename.split("_")[0])
                        step_numbers.append(step_num)
                    except ValueError:
                        continue
                
                if step_numbers:
                    step_numbers.sort(reverse=True)  # Sort in descending order
                    
                    # Try to load history from the most recent logs, going backwards until one succeeds
                    for step_num in step_numbers:
                        self.session_logger.log_message(f"Attempting to restart from step {step_num}")
                        log_file = os.path.join(log_dir, f"{step_num}_tool_response.json")
                        try:
                            with open(log_file, "r", encoding="utf-8") as f:
                                log_data = json.load(f)
                                # Restore history from the logged messages
                                if "request" in log_data and "messages" in log_data["request"]:
                                    # Start with the base messages from the request
                                    self.history = log_data["request"]["messages"] + log_data.get("response", [])
                                    start_step = step_num + 1  # Next step should be one more than the loaded step
                                    self.session_logger.log_message(f"Loaded history with {len(self.history)} messages from step {step_num}")
                                    self.session_logger.log_message(f"Restoring chosen template: {log_data.get('chosen_template_name', 'N/A')}")
                                    
                                    # Load tool_call_history if it exists
                                    if "tool_call_history" in log_data:
                                        self.tool_call_history = log_data["tool_call_history"]
                                        # Execute tool calls that would affect the environment state
                                        self._restore_environment_state()
                                    
                                    self.session_logger.log_message(f"Restarting from step {start_step}")
                                    self.validation_num = log_data["request"].get("validation_num", 0)
                                    self.is_frontend = log_data["request"].get("is_frontend", False)
                                    self.is_pure_frontend = log_data["request"].get("is_pure_frontend", False)
                                    break  # Successfully loaded, exit the loop
                        except (json.JSONDecodeError, FileNotFoundError, KeyError) as e:
                            # Log the error and continue to the next log file
                            self.session_logger.log_message(f"Failed to load log file {log_file}: {str(e)}")
                            continue
                    else:
                        # If we get here, all log loading attempts failed
                        self.session_logger.log_message("Failed to load any previous logs, starting from scratch")
        
        # If we're not restarting, initialize with system context
        if not self.history:
            # Initialize backend first
            system_prompt = get_core_system_prompt(self.config.working_dir) + f"\n\n--- Template Information ---\n\n{self.chosen_template['common_instruction']}"
            if not self.is_pure_frontend:
                user_prompt = f"--- User Instruction ---\n\n{self.config.instruction}\n\n--- Backend Plan ---\n\n{json.dumps(self.plans['backendPlan'], indent=2)}\n\n--- Backend Information ---\n\n{self.chosen_template['backend_instruction']}\n\nImplement the backend part of the project based on the User Instruction and the Backend Plan. You should **only** modify the backend part of the project."
            else:
                user_prompt = f"--- User Instruction ---\n\n{self.config.instruction}\n\n--- Frontend Plan ---\n\n{json.dumps(self.plans['frontendPlan'], indent=2)}\n\n--- Frontend Information ---\n\n{self.chosen_template['frontend_instruction']}\n\nThis is a pure frontend project. Implement the frontend part of the project based on the User Instruction and the Frontend Plan. You should **only** modify the frontend part of the project. Do NOT create, modify, or reference the backend."
            
            self.history = [
                {"role": "system", "content": system_prompt},
                {"role": "assistant", "content": "Got it. Thanks for the context!"},
                {"role": "user", "content": user_prompt}
            ]
        
        return start_step
        
    def _restore_environment_state(self):
        """Execute tool calls that would affect the environment state to restore it."""
        # List of tools that affect the environment state
        state_affecting_tools = {"write_file", "replace", "run_shell_command"}
        
        self.session_logger.log_message(f"Restoring environment state with {len(self.tool_call_history)} tool calls")
        
        for i, tool_call in enumerate(self.tool_call_history):
            tool_name = tool_call.get("function", {}).get("name")
            if tool_name in state_affecting_tools:
                try:
                    tool_args = json.loads(tool_call["function"]["arguments"])
                    self.session_logger.log_message(f"Executing state-restoring tool call {i}: {tool_name}")
                    # Execute the tool call to restore the environment state
                    self.registry.execute_tool(tool_name, tool_args)
                except Exception as e:
                    self.session_logger.log_message(f"Failed to execute state-restoring tool call {i}: {str(e)}")
                    # Continue with the next tool call even if one fails
                    continue
        
        self.session_logger.log_message("Environment state restoration completed")
        
    def get_history(self) -> List[Dict[str, Any]]:
        """Get the conversation history."""
        return self.history
        
    def set_history(self, history: List[Dict[str, Any]]):
        """Set the conversation history."""
        self.history = history
        
    def _compress_history(self, force=False, compression_ratio=None, step_idx=0):
        if compression_ratio is None:
            compression_ratio = self.config.compression_ratio
        """Compress history using the compression prompt when it exceeds maximum length."""
        if len(self.history) > self.max_history_length or force:
            # Group messages to ensure assistant messages with tool calls and their responses stay together
            grouped_history = []
            i = 3  # Start from index 3 to skip the initial context
            while i < len(self.history):
                # If current message is an assistant message with tool calls
                if (self.history[i].get("role") == "assistant" and 
                    "tool_calls" in self.history[i] and 
                    self.history[i]["tool_calls"]):
                    # Group this assistant message with its tool responses
                    group = [self.history[i]]
                    i += 1
                    # Add all subsequent tool responses
                    while i < len(self.history) and self.history[i].get("role") == "tool":
                        group.append(self.history[i])
                        i += 1
                    grouped_history.append(group)
                else:
                    # Single message group
                    grouped_history.append([self.history[i]])
                    i += 1

            if len(grouped_history) <= self.max_history_length and not force:
                return
            # Split into compressed_history and remaining_history
            compression_threshold = int(len(grouped_history) * compression_ratio)
            if compression_threshold > 0:
                compressed_groups = grouped_history[:-compression_threshold]
                remaining_groups = grouped_history[-compression_threshold:]
            else:
                compressed_groups = grouped_history
                remaining_groups = []

            # Flatten the groups
            compressed_history = []
            for group in compressed_groups:
                compressed_history.extend(group)
                
            remaining_history = []
            for group in remaining_groups:
                remaining_history.extend(group)
            
            # If there's nothing to compress, just trim the history
            if not compressed_history:
                # Keep the initial context and the remaining history
                self.history = self.history[:3] + remaining_history
                return

            try:
                # Prepare the compression prompt with the compressible history
                compression_prompt = get_compression_prompt()
                
                # Create a message to send to the LLM for compression
                compression_messages = [
                    {"role": "system", "content": compression_prompt},
                    {"role": "user", "content": json.dumps(compressed_history)}
                ]
                
                # Send request to LLM to compress the history
                compressed_response = llm_generation(
                    compression_messages, 
                    model=self.config.model,
                    max_tokens=self.config.max_tokens
                )

                request_data = {
                    "model": self.config.model,
                    "messages": compression_messages.copy(),
                    "is_pure_frontend": self.is_pure_frontend
                }
                self.session_logger.log_llm_request_response(step_idx, "compress_history", request_data, compressed_response)
                
                # Extract the compressed content
                compressed_content = compressed_response.get("content", "")
                
                # Create a compressed history message
                compressed_history_message = {
                    "role": "system",
                    "content": f"<COMPRESSED_HISTORY>{compressed_content}</COMPRESSED_HISTORY>"
                }
                
                # Form the new history: initial context + compressed result + remaining history
                self.history = self.history[:3] + [compressed_history_message] + remaining_history
                
            except Exception as e:
                # If compression fails, fall back to simple trimming
                # Keep initial context and the last N-3 messages
                remaining_groups = grouped_history[- compression_threshold:]
                remaining_history = []
                for group in remaining_groups:
                    remaining_history.extend(group)
                self.history = self.history[:3] + remaining_history
            
    def step(self, step_idx: int) -> Dict[str, Any]:
        """Send a message to the LLM, process the response, update history, and log response."""
        self._compress_history(step_idx=step_idx)

        # Prepare request for logging
        request_data = {
            "model": self.config.model,
            "messages": self.history.copy(),
        }
        tool_schemas = self.registry.get_tool_schemas()
        request_data["tool_schemas"] = tool_schemas
            
        try:
            # Send request to LLM
            response = self.llm_generation(self.history, model=self.config.model, tools=tool_schemas, max_tokens=self.config.max_tokens)
            self.session_logger.log_llm_request_response(step_idx, "llm_response", request_data, response, chosen_template_name=self.chosen_template["name"])
            self.history.append(response)

            # Execute the tool calls
            request_data = {
                "model": self.config.model,
                "validation_num": self.validation_num,
                "is_frontend": self.is_frontend,
                "messages": self.history.copy(),
                "is_pure_frontend": self.is_pure_frontend
            }
            tool_responses = []
            tool_calls = response.get("tool_calls", None)

            if tool_calls is None or len(tool_calls) == 0:
                content_text = response.get("content", "")
                if content_text.strip().startswith("Summary:"):
                    if self.history[-2].get("role", "") == "user":
                        # if the first message after continue prompt is summary, finish
                        return {"type": "finished", "summary": content_text}
                    else:
                        self.history = self.history[:-1]
                        return {"type": "to_be_continued"}
                else:
                    return {"type": "to_be_continued"}
            # tool_result = self.registry.execute_tool("run_shell_command", {"command": "npm -v", "is_input": False})
            # print(tool_result)
            # tool_result = self.registry.execute_tool("run_shell_command", {"command": "node --version", "is_input": False})
            # print(tool_result)
            for tool_call in tool_calls:
                if "<|channel|>" in tool_call["function"]["name"]:
                    tool_call["function"]["name"] = tool_call["function"]["name"].split("<|channel|>")[0]
                self.tool_call_history.append(tool_call)
                self.session_logger.log_message(f'executing {tool_call["id"]}: {tool_call["function"]["name"]}...', self.is_frontend)
                try:
                    tool_args = json.loads(tool_call["function"]["arguments"])
                    tool_result = self.registry.execute_tool(tool_call["function"]["name"], tool_args)
                except json.JSONDecodeError as e:
                    # Handle incomplete or invalid JSON in tool arguments
                    error_msg = f"Invalid JSON in tool arguments: {str(e)}"
                    self.session_logger.log_message(f'error in {tool_call["id"]}: {error_msg}', self.is_frontend)
                    tool_result = {
                        "llmContent": error_msg,
                        "returnDisplay": error_msg
                    }
                except Exception as e:
                    # Handle other errors in tool execution
                    error_msg = f"Error executing tool: {str(e)}"
                    self.session_logger.log_message(f'error in {tool_call["id"]}: {error_msg}', self.is_frontend)
                    tool_result = {
                        "llmContent": error_msg,
                        "returnDisplay": error_msg
                    }
                
                return_display = tool_result.get("returnDisplay", "")
                if return_display is None:
                    return_display = ""
                self.session_logger.log_message(f'returned {tool_call["id"]}: {str(return_display)[:500]}', self.is_frontend)
                content = compress_output(str(tool_result["llmContent"]), self.config.model, tool_call["function"]["name"], self.session_logger, step_idx)
                tool_response = {
                    "role": "tool",
                    "tool_call_id": tool_call["id"],
                    "content": content
                }
                if "new_db_entries" in tool_result:
                    tool_response["new_db_entries"] = tool_result["new_db_entries"]
                self.history.append(tool_response)
                tool_responses.append(tool_response)
            self.session_logger.log_llm_request_response(step_idx, "tool_response", request_data, tool_responses, tool_call_history=self.tool_call_history, chosen_template_name=self.chosen_template["name"])
            
            return {"type": "success"}
                
        except Exception as e:
            # Log error
            self.session_logger.log_llm_request_response(step_idx, "error", request_data, error=str(e))
            self.session_logger.log_message(f"error: {str(e)}", self.is_frontend)
            if "Please reduce the length of the messages or completion." in str(e) or "'max_tokens' or 'max_completion_tokens' is too large" in str(e):
                self._compress_history(force=True, step_idx=step_idx)
            return {"type": "error", "value": str(e)}

    def clear_history(self):
        """Clear the conversation history."""
        self.history = []
        self.tool_call_history = []

    def run_agent(self) -> bool:
        start_step = self.initialize_agent()
        for step_idx in tqdm(range(start_step, self.config.max_iterations)):
            result = self.step(step_idx)
            if result["type"] == "to_be_continued":
                if self.validation_num < self.config.max_validation_num:
                    self.history.append({
                        "role": "user",
                        "content": get_validation_prompt(self.is_frontend, self.config.instruction)
                    })
                    self.validation_num += 1
                elif self.validation_num < self.config.max_validation_num + self.config.max_summary_retry:
                    self.history.append({
                        "role": "user",
                        "content": get_summary_prompt(self.is_frontend)
                    })
                    self.validation_num += 1
                elif not self.is_frontend:
                    user_prompt = f"--- User Instruction ---\n\n{self.config.instruction}\n\nThe backend has already been implemented above.\n\n--- Frontend Plan ---\n\n{json.dumps(self.plans['frontendPlan'], indent=2)}\n\n--- Frontend Information ---\n\n{self.chosen_template['frontend_instruction']}\n\nImplement the frontend part of the project based on the User Instruction and the Frontend Plan. The backend APIs have already been implemented. You should **only** modify the frontend part of the project if possible. Do NOT modify the backend unless **absolutely necessary** and change as little as possible if you have to modify it."
                    self.validation_num = 0
                    if self.config.clear_backend:
                        self.history = self.history[:2] + [{
                            "role": "user",
                            "content": user_prompt
                        }]
                    else:
                        self.history.append({
                            "role": "user",
                            "content": user_prompt
                        })
                    self.is_frontend = True
            elif result["type"] == "finished":
                if self.is_frontend:
                    return True
                else:
                    user_prompt = f"--- User Instruction ---\n\n{self.config.instruction}\n\nThe backend has already been implemented above.\n\n--- Frontend Plan ---\n\n{json.dumps(self.plans['frontendPlan'], indent=2)}\n\n--- Frontend Information ---\n\n{self.chosen_template['frontend_instruction']}\n\nImplement the frontend part of the project based on the User Instruction and the Frontend Plan. The backend APIs have already been implemented. You should **only** modify the frontend part of the project if possible. Do NOT modify the backend unless **absolutely necessary** and change as little as possible if you have to modify it."
                    self.validation_num = 0
                    if self.config.clear_backend:
                        self.history = self.history[:2] + [{
                            "role": "user",
                            "content": user_prompt
                        }]
                    else:
                        self.history.append({
                            "role": "user",
                            "content": user_prompt
                        })
                    self.is_frontend = True
            elif (not self.is_frontend) and step_idx >= self.max_backend_iterations:    # backend and over max_backend_iterations
                if self.validation_num < self.config.max_validation_num + self.config.max_summary_retry:
                    self.history.append({
                        "role": "user",
                        "content": get_summary_prompt(self.is_frontend)
                    })
                    if self.validation_num < self.config.max_validation_num: # if validation not done, then move to summary
                        self.validation_num = self.config.max_validation_num
                    else: # else increase summary retry number
                        self.validation_num += 1
                else: # if validation and summary quota all over, start frontend dev
                    user_prompt = f"--- User Instruction ---\n\n{self.config.instruction}\n\nThe backend has already been implemented above.\n\n--- Frontend Plan ---\n\n{json.dumps(self.plans['frontendPlan'], indent=2)}\n\n--- Frontend Information ---\n\n{self.chosen_template['frontend_instruction']}\n\nImplement the frontend part of the project based on the User Instruction and the Frontend Plan. The backend APIs have already been implemented. You should **only** modify the frontend part of the project if possible. Do NOT modify the backend unless **absolutely necessary** and change as little as possible if you have to modify it."
                    self.validation_num = 0
                    if self.config.clear_backend:
                        self.history = self.history[:2] + [{
                            "role": "user",
                            "content": user_prompt
                        }]
                    else:
                        self.history.append({
                            "role": "user",
                            "content": user_prompt
                        })
                    self.is_frontend = True
        return False

    def run(self):
        self.initialize_template_and_plan()

        # Run backend
        finished = self.run_agent()

        if finished:
            with open(os.path.join(self.config.log_dir, "finished.json"), "w", encoding="utf-8") as f:
                json.dump({"is_finished": True}, f, indent=4)
        

