import base64
import json
from typing import Any

from fileagent_libs.fileagent_mcp_host import (async_sandbox_write_file, async_sandbox_execute_command, async_sandbox_create_sandbox, async_sandbox_destroy_sandbox)

from verl.tools.base_tool import BaseTool
from verl.tools.schemas import OpenAIFunctionToolSchema, ToolResponse

FILE = tuple[str, bytes]


class FileAgentSandboxTool(BaseTool):
    def __init__(self, config: dict, tool_schema: OpenAIFunctionToolSchema):
        super().__init__(config, tool_schema)
        self._session_ids: dict[str, str] = {}

    async def create(self, instance_id: str | None = None, files: list[FILE] = list(), **kwargs) -> tuple[str, ToolResponse]:
        created_instance_id, tool_response = await super().create(instance_id, **kwargs)
        session_id = json.loads(await async_sandbox_create_sandbox().result)["session_id"]  # TODO: get rid of json.loads
        self._session_ids[created_instance_id] = session_id

        for filename, content in files:
            await async_sandbox_write_file(session_id, filename, base64.b64encode(content).decode("ascii"), binary=True)
        return created_instance_id, tool_response

    async def execute(self, instance_id: str, parameters: dict[str, Any], **kwargs) -> tuple[ToolResponse, float, dict]:
        session_id = self._session_ids[instance_id]
        match parameters["action"]:
            case "WriteFile":
                response = await async_sandbox_write_file(session_id, parameters["filename"], parameters["content"], binary=False)
            case "ExecuteCommand":
                response = await async_sandbox_execute_command(session_id, parameters["command"])
            case _:
                response = None
        if response is None:
            return ToolResponse(), 0, {}
        else:
            return ToolResponse(text=json.dumps(response.as_dict())), 0, {}  # TODO: get rid of json.dumps

    async def release(self, instance_id: str, **kwargs) -> None:
        session_id = self._session_ids.pop(instance_id)
        await async_sandbox_destroy_sandbox(session_id)
