from openai import OpenAI
import time
import json
import os
import sys

sys.path.append("./model_api")
from BaseAPI import BaseAPI


class QwenLocalAPI(BaseAPI):
    def __init__(self, model_name, generation_config={}):
        super().__init__(generation_config)
        self.model_name = model_name
        # 支持通过环境变量配置 base_url 和 api_key，默认使用本地 7008 端口
        port = os.environ.get("QWEN_PORT", 7008)
        base_url = f"http://localhost:{port}/v1"
        api_key =  "forward"
        self.client = OpenAI(
            base_url=base_url,
            api_key=api_key,
        )
        self.sys_prompt = self.without_strict_jsonformat_sys_prompt

    def response(self, messages, tools):
        if not tools:
            tools = None
        for _ in range(10):
            try:
                completion = self.client.chat.completions.create(
                    model=self.model_name,
                    tools=tools,
                    messages=messages,
                    **self.generation_config
                )
                if completion is None or completion.choices is None:
                    continue
                return completion
            except Exception as e:
                error_str = str(e)
                # 检查是否是 content_filter 错误（Azure OpenAI / 360 API 的内容过滤）
                if 'content_filter' in error_str or 'content_management' in error_str or 'ResponsibleAIPolicyViolation' in error_str or 'jailbreak' in error_str:
                    print(f"⚠️ Content filter triggered: {error_str[:200]}...")
                    return "CONTENT_FILTERED"
                print(e)
                time.sleep(1)

    def generate_response(self, messages, tools):
        completion = self.response(messages, tools)

        if completion is None:
            return None
        
        # 处理 content_filter 错误：返回一个标准的拒绝响应（符合 OpenAI assistant 消息格式）
        if completion == "CONTENT_FILTERED":
            refuse_response = (
                "I'm sorry, but I cannot assist with this request. "
                "The content has been flagged by the safety system as potentially harmful or inappropriate. "
                "Please modify your request to comply with safety guidelines."
            )
            return {"type": "content", "content": refuse_response}

        ## tool call part
        if completion.choices[0].message.tool_calls is not None and len(completion.choices[0].message.tool_calls) > 0:
            tool_call = completion.choices[0].message.tool_calls[0]
            tool_call_id = tool_call.id
            tool_name = tool_call.function.name
            if tool_call.function.arguments:
                arguments = json.loads(tool_call.function.arguments)
            else:
                arguments = {}
            return {
                "type": "tool",
                "tool_call_id": tool_call_id,
                "tool_name": tool_name,
                "arguments": arguments,
            }

        ## normal content part
        else:
            content = completion.choices[0].message.content
            return {"type": "content", "content": content}
