import subprocess
import json
from pathlib import Path
from nyuctf_multiagent.challenge import CTFChallenge

from .tools import ToolCall, ToolResult, ALLTOOLS
from .logging import logger
import concurrent.futures
import multiprocessing
from multiprocessing import get_context
# at top-level of the file
def tool_target(tool, parsed_args, queue):
    try:
        result = tool.call(**parsed_args)
        queue.put(result)
    except Exception as e:
        queue.put({"stderr": str(e), "returncode": 1})
class CTFEnvironment:
    """Manages the docker env for the agent, and the challenge container."""
    def __init__(self, challenge: CTFChallenge, container_image: str, network: str, toolset: str="default"):
        self.challenge = challenge
        self.container_image = container_image
        self.network = network
        self.tools = {}
        for tool in ALLTOOLS:
            tool_instance = tool(self)
            self.tools[tool.NAME] = tool_instance

        # The SubmitFlagTool can set this to indicated if flag is found
        self.solved = False
        # The GiveupTool can set this to give up the challenge
        self.giveup = False

    def get_toolset(self, toolset):
        """Return a set of initialized tools"""
        logger.debug_message(f"Requested toolset: {toolset}")
        logger.debug_message(f"Available tools: {list(self.tools.keys())}")
        
        result_tools = {}
        for name in toolset:
            if name in self.tools:
                result_tools[name] = self.tools[name]
                logger.debug_message(f"Added tool: {name}")
            else:
                logger.debug_message(f"WARNING: Tool '{name}' not found in available tools!")
        
        logger.debug_message(f"Final toolset: {list(result_tools.keys())}")
        return result_tools

    def setup(self):
        self.start_docker()
        for tool in self.tools.values():
            tool.setup()
        # # Copy files
        # for file in self.challenge.files:
        #     hostpath = self.challenge.challenge_dir / file
        #     self.copy_into_container(hostpath, f"ctf_files/{file}")

    def teardown(self, exc_type, exc_value, traceback):
        # Tear down the tools first so they can clean up
        for tool in self.tools.values():
            tool.teardown(exc_type, exc_value, traceback)
        self.stop_docker()

    def start_docker(self):
        logger.print(f"Starting environment container {self.container_image}...", force=True)
        cmd = ["docker", "run", "-d", "--rm", 
               "--network", self.network,"--add-host", "host.docker.internal:host-gateway", "--platform", "linux/amd64",
               self.container_image]
        output = subprocess.run(cmd, check=True, capture_output=True, text=True)
        self.container = output.stdout.strip()
        logger.debug_message(f"...started {self.container}")

    def copy_into_container(self, hostpath, filename):
        if Path(filename).is_absolute():
            containerpath = Path(filename)
        else:
            containerpath = self.container_home / filename
            # Make parent path (only locals)
            cmd = ["docker", "exec", self.container, "mkdir", "-p", str(containerpath.parent)]
            subprocess.run(cmd, capture_output=True)
        # Copy file
        logger.debug_message(f"Copying file {hostpath} into container {self.container} at {containerpath}")
        cmd = ["docker", "cp", "-aq", str(hostpath), f"{self.container}:{containerpath}"]
        subprocess.run(cmd, capture_output=True)
        return containerpath

    def stop_docker(self):
        logger.print(f"Stopping environment container {self.container_image} {self.container}...", force=True)
        subprocess.run(["docker", "stop", self.container], check=True, capture_output=True)




    def run_tool(self, tool_call):
        tool = self.tools[tool_call.name]
        queue = multiprocessing.Queue()

        process = multiprocessing.Process(target=tool_target, args=(tool, tool_call.parsed_arguments, queue))
        process.start()
        process.join(timeout=115)
        if process.is_alive():
            process.terminate()
            process.join()
            res = {
                "stdout": "Timeout of 2 minutes...",
                "stderr": "Timeout of 2 minutes...",
                "returncode": 1,
                "timed_out": True
            }
        else:
            if not queue.empty():
                res = queue.get()
                print(res)
                self.giveup = res.get('giveup', False)
            else:
                res = {
                    "stdout": "",
                    "stderr": "No output received.",
                    "returncode": 1,
                    "timed_out": False
                }

        # Extract cost from tool result if available
        tool_cost = 0.0
        if isinstance(res, dict) and "search_cost" in res:
            tool_cost = res["search_cost"]
        
        return ToolResult(name=tool_call.name, id=tool_call.id, result=res, cost=tool_cost)
    @property
    def container_home(self):
        return Path("/home/ctfplayer")

