import os
from datetime import datetime
from src.logger import setup_logger

# Setup logger
hf_model = os.getenv("AGENT_LLM_MODEL", "default_model")
run_id = os.getenv("RUN_ID", "default_run") 

logger = setup_logger(
    __name__,
    level="INFO",
    log_subdir=hf_model,
    log_filename=f"run_{run_id}.log"
)

class Memory:
    def __init__(self, max_length=10000):
        """
        Initialize the memory module.
        """
        self.max_length = max_length
        self.memory = []

    def add(self, step, command, result_dict):
        """
        Add a new entry to the memory.
        """
        if result_dict == "finished":
            return_code = 0
            success = True
            entry = {
                "status": "finished",
                "return_code": return_code,
                "timestamp": datetime.now().isoformat()
            }
        else:
            return_code = result_dict.get("return_code", -1)
            success = return_code == 0
            entry = {
                "step": step,
                "command": command,
                "status": "success" if success else "failed",
                "result": result_dict["stdout"] if success else result_dict["stderr"],
                "return_code": return_code,
                "timestamp": datetime.now().isoformat()
            }
        self.memory.append(entry)
        self._truncate_if_needed()

    def _truncate_if_needed(self):
        """
        Control memory size, keep the first entry, and summarize the long output of specific type commands.
        """
        def is_install_command(cmd):
            return any(keyword in cmd for keyword in ['pip install', 'apt install', 'conda install', 'pip3 install'])

        def is_transfer_command(cmd):
            return any(keyword in cmd for keyword in ['rsync'])

        def simplify_entry(entry, type="install"):
            if entry['status'] == 'success':
                if type == "install":
                    entry['result'] = "All dependencies were successfully installed."
                elif type == "transfer":
                    entry['result'] = "All files were successfully transferred."
            else:
                if type == "install":
                    entry['result'] = "Dependency installation failed"
                elif type == "transfer":
                    entry['result'] = "Files failed to be transferred"

        while self._current_length() > self.max_length:
            simplified = False
            for i in range(1, len(self.memory)): 
                entry = self.memory[i]
                if is_install_command(entry['command']) and len(entry['result']) > 200:
                    simplify_entry(entry, type="install")
                    logger.info(f"Memory Size Exceeded -- Simplified Entry with Command {entry.get('command')}")
                    simplified = True
                    break
                elif is_transfer_command(entry['command']) and len(entry['result']) > 200:
                    simplify_entry(entry, type="transfer")
                    logger.info(f"Memory Size Exceeded -- Simplified Entry with Command {entry.get('command')}")
                    simplified = True
                    break 
            if not simplified:
                if len(self.memory) > 1:
                    logger.info("Memory Exceeded Max Length -- Truncate from Second Entry")
                    self.memory.pop(1)
                else:
                    break

    def _current_length(self):
        """
        Get the current total length of the memory.
        """
        return sum(len(str(entry)) for entry in self.memory)

    def get_memory(self):
        """
        Get all structured memory.
        """
        return self.memory

    def clear(self):
        """
        Clear the memory.
        """
        self.memory = []

    def __str__(self):
        """
        Return a formatted string representation (for debugging).
        """
        return "\n\n".join(
            f"Step {entry.get('step', '')}\nTime: {entry.get('timestamp', '')}\nCommand: {entry.get('command', '')}\nResult: {entry.get('result', '')}"
            for entry in self.memory
        )
