import json
import re
import asyncio
from typing import Dict, Any, List
from core.base_agent import BaseAgent
from services.tool_service import ToolService

from pydantic import BaseModel, Field
from typing import Any, Dict, Optional

class ToolCall(BaseModel):
    name: str = Field(..., description="The name of the tool function")
    arguments: Dict[str, Any] = Field(default_factory=dict, description="The arguments to pass to the tool")


class Executor(BaseAgent):
    """
    Pipeline: Context -> LLM(Async) -> JSON -> Check -> ToolService(Async) -> Result
    """
    
    PROMPT_TOOL_DECISION = """
    You are an Executor Agent. 
    Your task is to analyze the provided Context with available Tools, then generate a valid JSON to call the appropriate tool.

    Return format is ONLY a JSON object, no other text:
    {{
        "name": "...",
        "arguments": {{ "...": "...", ... }}
    }}

    Context:
    {context}

    """
    
    call_log: List[Dict] = []

    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        self.tool_service = ToolService()


    def check_format(self, content: str) -> Dict:
        default = {"name": "function", "arguments": {"key": "value"}}
        try:
            data = json.loads(content)
            return data
        except Exception:
            return default

    async def execute(self, context: str) -> Dict[str, str]:
        prompt = self.PROMPT_TOOL_DECISION.format(context=context)
        messages = [{"role": "user", "content": prompt}]
        
        # unused by sglang reason-parser
        reasoning_content, content = await self.call(messages, ToolCall)

        fc_content = self.check_format(content)
        tool_call_results = await self.tool_service.run_stream(context, fc_content)
        
        return {
            "context": context,
            "fc_content": fc_content,
            "result": tool_call_results
        }
