from typing import Any, Callable, Literal, Optional, Union
from pathlib import Path
from uuid import UUID, uuid4
from pydantic import BaseModel, Field
from enum import Enum
import os
import typing
from contextvars import ContextVar

# 尝试导入 requests_unixsocket，如果失败则使用标准的 requests
# 这允许在支持 Unix-socket 的环境中使用回调
try:
    import requests_unixsocket as requests
except ImportError:
    import requests

from agent.base import BaseAgent

# 定义上下文变量，用于追踪当前的角色
CURRENT_ROLE: ContextVar["BaseAgent"] = ContextVar("BaseAgent", default=None)

class BlockType(str, Enum):
    """不同块类型的枚举。"""

    TERMINAL = "Terminal"
    TASK = "Task"
    BROWSER = "Browser"
    BROWSER_RT = "Browser-RT"
    EDITOR = "Editor"
    GALLERY = "Gallery"
    NOTEBOOK = "Notebook"
    DOCS = "Docs"
    THOUGHT = "Thought"
    
# 定义流结束时使用的标记
END_MARKER_NAME = "end_marker"
END_MARKER_VALUE = "\x18\x19\x1B\x18\n"

class ResourceReporter(BaseModel):
    """用于资源报告的基类。"""

    block: BlockType = Field(description="报告资源的块类型")
    uuid: UUID = Field(default_factory=uuid4, description="资源的唯一标识符")
    callback_url: str = Field('', description="报告应发送到的 URL")

    def report(self, value: Any, name: str, extra: Optional[dict] = None):
        """同步报告资源观察数据。

        Args:
            value: 要报告的数据。
            name: 数据的类型名称。
            extra: 附加的额外信息。
        """
        return self._report(value, name, extra)

    @classmethod
    def set_report_fn(cls, fn: Callable):
        """设置同步报告函数。

        Args:
            fn: 用于同步报告的可调用函数。例如:

                >>> def _report(self, value: Any, name: str, extra: Optional[dict] = None):
                ...     print(value, name)
                
        """
        cls._report = fn

    def _report(self, value: Any, name: str, extra: Optional[dict] = None):
        """内部实现的同步报告方法。"""
        if not self.callback_url:
            return

        data = self._format_data(value, name, extra)
        # 使用 requests 发送 POST 请求
        resp = requests.post(self.callback_url, json=data)
        resp.raise_for_status()
        return resp.text

    def _format_data(self, value, name, extra):
        """格式化要发送的数据。"""
        data = self.model_dump(mode="json", exclude={"callback_url"})
        if isinstance(value, BaseModel):
            value = value.model_dump(mode="json")
        elif isinstance(value, Path):
            value = str(value)

        if name == "path":
            value = os.path.abspath(value)
            
        data["value"] = value
        data["name"] = name
        
        # 获取当前上下文中的角色名称
        role = CURRENT_ROLE.get()
        data["role"] = role.name if role else "unknown"
        
        if extra:
            data["extra"] = extra
        return data

    def __enter__(self):
        """进入同步流式回调上下文。"""
        return self

    def __exit__(self, *args, **kwargs):
        """退出同步流式回调上下文，并发送结束标记。"""
        self.report(None, END_MARKER_NAME)


class FileReporter(ResourceReporter):
    """文件资源回调，用于报告完整的文件路径。"""

    def report(
        self,
        value: Union[Path, dict, Any],
        name: Literal["path", "meta", "content"] = "path",
        extra: Optional[dict] = None,
    ):
        """同步报告文件资源。"""
        return super().report(value, name, extra)


class NotebookReporter(FileReporter):
    """专用于 Notebook 的报告器。
    
    这个类等同于 FileReporter(block=BlockType.NOTEBOOK)，
    它继承自 FileReporter 并将 block 类型默认设置为 NOTEBOOK。
    """
    block: BlockType = Field(default=BlockType.NOTEBOOK, description="块类型固定为 Notebook")