"""
Enhanced memory tool for Qwen Code Python implementation with better memory management and organization.
"""
import os
import json
import difflib
from typing import Dict, Any
from .base_tool import BaseTool
from .tool_types import ToolKind


# Constants for memory management
GEMINI_CONFIG_DIR = ".qwen"
DEFAULT_CONTEXT_FILENAME = "QWEN.md"
MEMORY_SECTION_HEADER = "## Qwen Added Memories"


def ensure_newline_separation(current_content: str) -> str:
    """Ensures proper newline separation before appending content."""
    if not current_content:
        return ""
    if current_content.endswith("\n\n") or current_content.endswith("\r\n\r\n"):
        return ""
    if current_content.endswith("\n") or current_content.endswith("\r\n"):
        return "\n"
    return "\n\n"


def compute_new_content(current_content: str, fact: str) -> str:
    """Computes the new content that would result from adding a memory entry."""
    processed_text = fact.strip()
    # Remove leading hyphens and spaces that might be misinterpreted as markdown list items
    processed_text = processed_text.replace(r'^(-+\s*)+', '', 1).strip()
    new_memory_item = f"- {processed_text}"

    header_index = current_content.find(MEMORY_SECTION_HEADER)

    if header_index == -1:
        # Header not found, append header and then the entry
        separator = ensure_newline_separation(current_content)
        return f"{current_content}{separator}{MEMORY_SECTION_HEADER}\n{new_memory_item}\n"
    else:
        # Header found, find where to insert the new memory entry
        start_of_section_content = header_index + len(MEMORY_SECTION_HEADER)
        end_of_section_index = current_content.find("\n## ", start_of_section_content)
        if end_of_section_index == -1:
            end_of_section_index = len(current_content)  # End of file

        before_section_marker = current_content[:start_of_section_content].rstrip()
        section_content = current_content[start_of_section_content:end_of_section_index].rstrip()
        after_section_marker = current_content[end_of_section_index:]

        section_content += f"\n{new_memory_item}"
        return f"{before_section_marker}\n{section_content.lstrip()}\n{after_section_marker}".rstrip() + "\n"


def get_memory_file_path(scope: str, working_dir: str) -> str:
    """Get the path to the memory file based on scope."""
    if scope == "global":
        return os.path.join(os.path.expanduser("~"), GEMINI_CONFIG_DIR, DEFAULT_CONTEXT_FILENAME)
    else:  # project
        return os.path.join(working_dir, DEFAULT_CONTEXT_FILENAME)


class MemoryTool(BaseTool):
    """Enhanced tool for saving information to long-term memory with better organization."""
    Name = "save_memory"

    def __init__(self, working_dir: str):
        super().__init__(
            self.Name,
            """Saves a specific piece of information or fact to your long-term memory.

Use this tool:

- When the user explicitly asks you to remember something (e.g., "Remember that I like pineapple on pizza", "Please save this: my cat's name is Whiskers").
- When the user states a clear, concise fact about themselves, their preferences, or their environment that seems important for you to retain for future interactions to provide a more personalized and effective assistance.

Do NOT use this tool:

- To remember conversational context that is only relevant for the current session.
- To save long, complex, or rambling pieces of text. The fact should be relatively short and to the point.
- If you are unsure whether the information is a fact worth remembering long-term. If in doubt, you can ask the user, "Should I remember that for you?"

## Parameters

- `fact` (string, required): The specific fact or piece of information to remember. This should be a clear, self-contained statement. For example, if the user says "My favorite color is blue", the fact would be "My favorite color is blue".
- `scope` (string, optional): Where to save the memory:
  - "global": Saves to user-level ~/.qwen/QWEN.md (shared across all projects)
  - "project": Saves to current project's QWEN.md (project-specific)
  - If not specified, the tool will ask the user where they want to save the memory.""",
            {
                "type": "object",
                "properties": {
                    "fact": {
                        "type": "string",
                        "description": "The specific fact or piece of information to remember. Should be a clear, self-contained statement."
                    },
                    "scope": {
                        "type": "string",
                        "enum": ["global", "project"],
                        "description": "Where to save the memory: \"global\" saves to user-level ~/.qwen/QWEN.md (shared across all projects), \"project\" saves to current project's QWEN.md (project-specific). If not specified, will prompt user to choose."
                    }
                },
                "required": ["fact"]
            },
            ToolKind.THINK
        )
        self.working_dir = working_dir

    def validate_params(self, params: Dict[str, Any]) -> str:
        error = super().validate_params(params)
        if error:
            return error

        fact = params["fact"]
        if not fact or not fact.strip():
            return "Parameter \"fact\" must be a non-empty string."

        scope = params.get("scope")
        if scope and scope not in ["global", "project"]:
            return "The 'scope' parameter must be either 'global' or 'project', or omitted."

        return None

    def get_description(self, params: Dict[str, Any]) -> str:
        """Get a description of the save memory operation."""
        fact = params.get("fact", "")
        scope = params.get("scope")
        
        if not scope:
            global_path = get_memory_file_path("global", self.working_dir)
            project_path = get_memory_file_path("project", self.working_dir)
            return f"CHOOSE: {global_path} (global) OR {project_path} (project)"

        memory_file_path = get_memory_file_path(scope, self.working_dir)
        return f"{memory_file_path} ({scope})"

    def execute(self, params: Dict[str, Any]) -> Dict[str, Any]:
        error = self.validate_params(params)
        if error:
            return {
                "llmContent": f"Error: Invalid parameters provided. Reason: {error}",
                "returnDisplay": f"Error: {error}"
            }

        fact = params["fact"]
        scope = params.get("scope")
        modified_by_user = params.get("modified_by_user", False)
        modified_content = params.get("modified_content")

        # If scope is not specified, return error prompting for choice
        if not scope and not modified_by_user:
            global_path = get_memory_file_path("global", self.working_dir)
            project_path = get_memory_file_path("project", self.working_dir)
            error_message = f"""Please specify where to save this memory:

Global: {global_path} (shared across all projects)
Project: {project_path} (current project only)"""
            
            return {
                "llmContent": json.dumps({
                    "success": False,
                    "error": "Please specify where to save this memory"
                }),
                "returnDisplay": error_message
            }

        # Default to global if not specified but modified by user
        if not scope:
            scope = "global"

        try:
            memory_file_path = get_memory_file_path(scope, self.working_dir)
            
            if modified_by_user and modified_content is not None:
                # User modified the content in external editor, write it directly
                memory_dir = os.path.dirname(memory_file_path)
                os.makedirs(memory_dir, exist_ok=True)
                
                with open(memory_file_path, "w", encoding="utf-8") as f:
                    f.write(modified_content)
                    
                success_message = f"Okay, I've updated the {scope} memory file with your modifications."
                return {
                    "llmContent": json.dumps({
                        "success": True,
                        "message": success_message
                    }),
                    "returnDisplay": success_message
                }
            else:
                # Use the normal memory entry logic
                memory_dir = os.path.dirname(memory_file_path)
                os.makedirs(memory_dir, exist_ok=True)

                # Read existing content
                existing_content = ""
                if os.path.exists(memory_file_path):
                    with open(memory_file_path, "r", encoding="utf-8") as f:
                        existing_content = f.read()

                # Calculate the new content that will be written to the memory file
                new_content = compute_new_content(existing_content, fact)

                # Write updated content
                with open(memory_file_path, "w", encoding="utf-8") as f:
                    f.write(new_content)

                success_message = f"Okay, I've remembered that in {scope} memory: \"{fact}\""
                return {
                    "llmContent": json.dumps({
                        "success": True,
                        "message": success_message
                    }),
                    "returnDisplay": success_message
                }

        except PermissionError:
            error_msg = f"Permission denied writing to memory file: {memory_file_path}"
            return {
                "llmContent": json.dumps({
                    "success": False,
                    "error": f"Failed to save memory. Detail: {error_msg}"
                }),
                "returnDisplay": f"Error saving memory: {error_msg}"
            }
        except Exception as e:
            error_msg = str(e)
            return {
                "llmContent": json.dumps({
                    "success": False,
                    "error": f"Failed to save memory. Detail: {error_msg}"
                }),
                "returnDisplay": f"Error saving memory: {error_msg}"
            }