from openai import OpenAI
import time
import json
import os
import sys
sys.path.append('./model_api')
from BaseAPI import BaseAPI

class OpenaiAPI(BaseAPI):
    def __init__(self, model_name, generation_config={}):
        super().__init__(generation_config)
        self.model_name = model_name
        # 支持通过环境变量配置 base_url 和 api_key，默认使用本地 7008 端口
        base_url = os.getenv("OPENAI_API_BASE_URL", "http://localhost:8111/v1")
        api_key = os.getenv("OPENAI_API_KEY", "forward")
        self.client = OpenAI(base_url=base_url, api_key=api_key)
        # 360ai API 404
        self.client = OpenAI(base_url="YOUR_360_API_URL", api_key="YOUR_API_KEY")
        
        self.sys_prompt = self.without_strict_jsonformat_sys_prompt
        # self.sys_prompt = self.sys_prompt_with_failure_modes_without_strict_jsonformat
        # self.sys_prompt = self.sys_prompt_with_simple_failure_modes_without_strict_jsonformat

    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
                )
                # print(f'completion: {completion}')
                if completion is None or completion.choices is None:
                    continue
                return completion
            except Exception as e:
                error_str = str(e)
                # 检查是否是 content_filter 错误（Azure OpenAI 的内容过滤）
                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]}...")
                    # 返回一个特殊的 completion 对象，表示被内容过滤拒绝
                    return "CONTENT_FILTERED"
                print(e)
                time.sleep(1)

    def generate_response(self, messages, tools):
        completion = self.response(messages, tools)
        # print(f'messages: {messages}\ncompletion: {completion}')

        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
        # print(f'completion: {completion}')
        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}
