# -*- coding: utf-8 -*-
"""
tool_utils.py
- 只保留两个工具：code（python执行），terminal（shell执行）
- 工具在内部对接 sandbox_livehouse v2：run_code / sandbox_execute_command
- 每个线程有独立的 session_id（由 api_infer 在任务开始处 set_session_id）
"""

import json
import threading
from typing import Any, Dict
import os
import requests
from fileagent_libs.fileagent_mcp_host import (
    run_code,
    sandbox_execute_command,
    read_link 
)
from firecrawl import FirecrawlApp
from config import (
    FIRECRAWL_API_KEY,
    SERPAPI_API_KEY
)

firecrawl_app = FirecrawlApp(api_key=FIRECRAWL_API_KEY)
# 线程本地上下文：存每个任务的 session_id
_TLS = threading.local()

def set_session_id(session_id: str) -> None:
    _TLS.session_id = session_id

def get_session_id() -> str:
    return getattr(_TLS, "session_id", "")

def clear_session_id() -> None:
    if hasattr(_TLS, "session_id"):
        delattr(_TLS, "session_id")

def search_api(query: str, return_n: int = 10) -> dict:
    """
    调用SerpAPI作为搜索工具，执行网络搜索查询
    
    Args:
        query (str): 搜索查询关键词或问题
        return_n (int, optional): 返回结果数量，默认为10
        
    Returns:
        dict: 包含搜索结果的字典
            - data: 搜索结果数据或错误信息
            - succeed: 布尔值，表示是否成功
    """
    # 构建搜索API的URL（请在环境变量中配置 SEARCH_TOOL_BASE_URL）
    base_url = os.getenv("SEARCH_TOOL_BASE_URL", "")
    if not base_url:
        return {
            "data": "[search_api] SEARCH_TOOL_BASE_URL is not set",
            "succeed": False,
        }
    params = {
        "query": query,
        "num": return_n,
        "start": 0,
        "engine": "google",
        "location": "cn"
    }
    
    try:
        # 发送GET请求
        response = requests.get(base_url, params=params)
        
        # 检查请求是否成功
        if response.status_code == 200:
            data = response.json()
            return {
                "data": data,
                "succeed": True
            }
        else:
            error_msg = f"请求失败，状态码: {response.status_code}"
            print(error_msg)
            return {
                "data": error_msg,
                "succeed": False
            }
            
    except Exception as e:
        error_msg = f"SerpAPI调用失败，错误信息: {e}"
        print(error_msg)
        return {
            'data': error_msg,
            'succeed': False
        }


def link_reader_tool(url: str) -> dict:
    """
    调用网页阅读工具读取指定URL网页内容，返回markdown格式的网页内容
    
    Args:
        url (str): 需要读取内容的网页URL地址
        
    Returns:
        dict: 包含网页内容的字典
            - markdown: markdown格式的网页内容
            - succeed: 布尔值，表示是否成功
            - error_message: 错误信息（仅在失败时）
    """
    try:
        # 使用Firecrawl抓取网页内容
        link_result = read_link(url)
        if link_result.status.is_succeeded():
            return {
                "data": link_result.result,
                'succeed': True
            }
        else:
            error_msg = f"LinkReader工具调用失败，状态码: {link_result.status}"
            return {
                'error_message': error_msg,
                'succeed': False
            }
            
    except Exception as e:
        error_msg = f"LinkReader工具调用异常: {e}"
        return {
            'error_message': error_msg,
            'succeed': False
        }

# --- 规范化返回 ---
def _normalize_run_code(resp: Any) -> Dict[str, Any]:
    """
    兼容多种返回风格：
    - 对象：有 .code/.stdout/.stderr
    - 字典：{'code':..,'stdout':..,'stderr':..}
    - 包含 .result JSON 字符串的对象
    """
    # 先尝试 resp.result -> JSON
    try:
        if hasattr(resp, "result"):
            j = json.loads(resp.result)
            if isinstance(j, dict) and {"code", "stdout", "stderr"} <= j.keys():
                return {"code": int(j.get("code", 1)),
                        "stdout": str(j.get("stdout", "")),
                        "stderr": str(j.get("stderr", ""))}
    except Exception:
        pass

    # 直接对象属性
    if hasattr(resp, "__dict__"):
        code = int(getattr(resp, "code", getattr(resp, "returncode", 1)))
        stdout = str(getattr(resp, "stdout", ""))
        stderr = str(getattr(resp, "stderr", ""))
        return {"code": code, "stdout": stdout, "stderr": stderr}

    # 字典
    if isinstance(resp, dict):
        return {"code": int(resp.get("code", 1)),
                "stdout": str(resp.get("stdout", "")),
                "stderr": str(resp.get("stderr", ""))}

    # 未知
    return {"code": 1, "stdout": "", "stderr": f"[run_code] unexpected type: {type(resp).__name__}"}


def _normalize_shell(resp: Any) -> Dict[str, Any]:
    """
    兼容：
    - 字典：{'code':..,'stdout':..,'stderr':..}
    - 对象.as_dict()['return'] -> dict
    """
    # as_dict()['return']
    try:
        if hasattr(resp, "as_dict"):
            val = resp.as_dict().get("return")
            if isinstance(val, str):
                try:
                    val = json.loads(val)
                except Exception:
                    # 不是json
                    return {"code": 1, "stdout": "", "stderr": str(val)}
            if isinstance(val, dict):
                return {"code": int(val.get("code", 1)),
                        "stdout": str(val.get("stdout", "")),
                        "stderr": str(val.get("stderr", ""))}
    except Exception:
        pass

    # 直接 dict
    if isinstance(resp, dict):
        return {"code": int(resp.get("code", 1)),
                "stdout": str(resp.get("stdout", "")),
                "stderr": str(resp.get("stderr", ""))}

    return {"code": 1, "stdout": "", "stderr": f"[sandbox_execute_command] unexpected type: {type(resp).__name__}"}


# --- 分发器：只支持 code / terminal ---
def tool_dispatcher(tool_call: dict) -> str:
    """
    统一分发入口（返回字符串；建议为 JSON 字符串）
    支持：
      - name: "code"       arguments: {"code": "<python code>"}
      - name: "terminal"   arguments: {"command": "<shell command>"}
    """
    name = tool_call.get("name")
    args = tool_call.get("arguments", {})

    if isinstance(args, str):
        try:
            args = json.loads(args) if args.strip() else {}
        except Exception:
            args = {"_raw": args}

    session_id = get_session_id()  # 每题独立沙盒
    if not session_id:
        return json.dumps({"error": "[dispatcher] missing session_id"}, ensure_ascii=False)

    try:
        if name in ("code", "ExecuteCode", "python"):
            code_text = args.get("code", "")
            resp = run_code(code_text, language="python", session_id=session_id)
            norm = _normalize_run_code(resp)
            return json.dumps(norm, ensure_ascii=False)

        elif name in ("terminal", "ExecuteShell", "shell"):
            cmd = args.get("command", "")
            resp = sandbox_execute_command(session_id=session_id, command=cmd)
            norm = _normalize_shell(resp)
            return json.dumps(norm, ensure_ascii=False)
        
        elif name == "search_api":
            return _handle_search_api(args)
        
        elif name == "link_reader_tool":
            return _handle_link_reader_tool(args)

        else:
            return f"[dispatcher] ❌ 未知工具: {name}"

    except Exception as e:
        return json.dumps({"code": 1, "stdout": "", "stderr": f"[dispatcher exception] {type(e).__name__}: {e}"}, ensure_ascii=False)



def _handle_search_api(tool_args: dict) -> str:
    """
    处理search_api工具调用
    
    Args:
        tool_args (dict): 工具参数
        
    Returns:
        str: 搜索结果内容
    """
    query = tool_args.get("query", "")
    print(f"[search_api] 查询: {query}")
    
    result = search_api(query=query)
    
    if result.get("succeed", True):
        content = json.dumps(result["data"], ensure_ascii=False, indent=2)
        return content
    else:
        return f"[search_api] 失败: {result.get('data')}"


def _handle_link_reader_tool(tool_args: dict) -> str:
    """
    处理link_reader_tool工具调用
    
    Args:
        tool_args (dict): 工具参数
        
    Returns:
        str: 网页内容或错误信息
    """
    url = tool_args.get("url", "")
    print(f"[link_reader_tool] URL: {url}")
    
    result = link_reader_tool(url=url)
    
    if result.get("succeed", True):
        markdown_content = result.get("markdown", "")
        if markdown_content:
            return markdown_content
        else:
            return "[link_reader_tool] 返回空的markdown内容"
    else:
        return result.get("error_message", "[link_reader_tool] 未知错误")
