import logging
import os
from typing import Any, Optional, Union
from uuid import uuid4

from verl.utils.rollout_trace import rollout_trace_op
from verl.tools.base_tool import BaseTool
from verl.tools.schemas import OpenAIFunctionToolSchema, ToolResponse
from recipe.fileagent.agent_loop import ImageToolResponse
from recipe.fileagent.utils.metric_utils import build_tool_metric
from recipe.fileagent.utils.vision_utils import controlled_smart_resize
import json
from typing import Any, Callable, Coroutine
from typing import Optional, Union
from uuid import uuid4
from fileagent_libs.fileagent_mcp_host import ToolResponse as FileAgentToolResponse
from fileagent_libs.fileagent_mcp_host import map_tools

from verl.tools.base_tool import BaseTool
from verl.tools.schemas import OpenAIFunctionToolSchema, ToolResponse
from recipe.fileagent.utils.metric_utils import build_tool_metric


logger = logging.getLogger(__name__)
logger.setLevel(os.getenv("VERL_LOGGING_LEVEL", "WARN"))


class SearchMultimediaTool(BaseTool):
    def __init__(self, config: dict, tool_schema: OpenAIFunctionToolSchema):
        super().__init__(config, tool_schema)
        self.function: Callable[..., Coroutine[Any, Any, FileAgentToolResponse]] = map_tools(self.name)
        self.custom_config = self.config.get("custom", {})
        self.max_attempts = self.custom_config.pop("max_attempts", 1)
        self._instance_dict = {}
        print(f"FileAgentStatelessTool: name: {self.name}, custom_config: {self.custom_config}, max_attempts: {self.max_attempts}")
    
    async def create(self, instance_id: Optional[str] = None, **kwargs) -> tuple[str, ToolResponse]:
        if instance_id is None:
            instance_id = str(uuid4())
        images = kwargs["create_kwargs"]["images"]
        model_input_sizes = kwargs["create_kwargs"]["model_input_sizes"]
        if images is None:
            raise ValueError("images must be provided when creating the image zoom in tool.")
        if model_input_sizes is None:
            raise ValueError("model_input_sizes must be provided when creating the image zoom in tool.")
        self._instance_dict[instance_id] = {
            "response": "",
            "images": images,
            "model_input_sizes": model_input_sizes,
        }
        return instance_id, ToolResponse()

    @rollout_trace_op
    async def execute(self, instance_id: str, parameters: dict[str, Any], **kwargs) -> tuple[Union[ImageToolResponse, ToolResponse], float, dict]:
        index = parameters.get("image_id",1)
        if "image_id" in parameters:
            parameters.pop("image_id")
        combined: dict[str, Any] = {}
        combined.update(self.custom_config)
        try:
            index=int(index)-1
            import copy
            image = copy.deepcopy(self._instance_dict[instance_id]["images"][index])
        except:
            metric = build_tool_metric(tool_name=self.name, succeeded=False)
            return ToolResponse(text=f"Invalid image id:{index}"), 0, metric

        height,width=image.size
        rh=height
        rw=width
        if height>1024:
            rh=1024
        if width>1024:
            rw=1024
        if height!=rh or width!=rw:
            image=image.resize((rw,rh))
        import base64
        import io
        buffer = io.BytesIO()
        image.save(buffer, format="PNG")
        img_b64 = base64.b64encode(buffer.getvalue()).decode()
        parameters["query"]=img_b64
        combined.update(parameters)

        for _ in range(self.max_attempts):
            response: FileAgentToolResponse = await self.function(**combined)
            if response.status.is_succeeded():
                break

        # Tool Metric
        metric = build_tool_metric(
            tool_name=self.name,
            succeeded=response.status.is_succeeded(),
        )

        if not response.status.is_succeeded():
            print(f"FileAgentStatelessTool Error: name: {self.name}, params: {combined}, response: {json.dumps(response.as_dict())}")

        return ToolResponse(text=json.dumps(response.as_dict())), 0, metric  # TODO:get rid of json.dumps
