import os
import json
import time
from utils import save_json
from typing import Dict, List, Optional, Tuple

from openai import OpenAI, APIError, RateLimitError
from configs import logger

def generate_with_openai_sync(client: OpenAI, model_name: str, conversation: List[Dict],
                              functions: List[Dict], temperature: Optional[float] = None,
                              top_p: Optional[float] = None, max_tokens: Optional[int] = None) -> Tuple[Dict, Dict]:
    """Internal function for non-streaming generation"""
    # logger.info(f"Conversation: {conversation}")
    try:
        if functions:
            response = client.chat.completions.create(
                model=model_name,
                messages=conversation,
                temperature=temperature,
                top_p=top_p,
                max_tokens=max_tokens,
                tools=[{"type": "function", "function": f} for f in functions],
                tool_choice="auto",
                stream=False
            )
        else:
            logger.info(f"Generating without functions for model: {model_name}")
            response = client.chat.completions.create(
                model=model_name,
                messages=conversation,
                temperature=temperature,
                top_p=top_p,
                max_tokens=max_tokens,
                stream=False
            )

        usage = {
            'prompt_tokens': response.usage.prompt_tokens if hasattr(response.usage, 'prompt_tokens') else None,
            'completion_tokens': response.usage.completion_tokens if hasattr(response.usage, 'completion_tokens') else None
        }

        choice = response.choices[0]
        assistant_text = choice.message.content or ""
        # if isinstance(assistant_text, str) and len(assistant_text) == 0:
        #     raise ValueError("Received empty response from OpenAI API.")

        tool_calls = []

        # logger.info(f"Assistant Text: {assistant_text}")
        if hasattr(choice.message, 'tool_calls') and choice.message.tool_calls:
            for tc in choice.message.tool_calls:
                if tc.type == 'function':
                    tool_call = {
                        "id": tc.id,
                        "type": "function",
                        "function": {
                            "name": tc.function.name,
                            "arguments": tc.function.arguments or "{}"
                        }
                    }
                    # Ensure arguments is valid JSON
                    try:
                        json.loads(tool_call["function"]["arguments"])
                    except json.JSONDecodeError:
                        tool_call["function"]["arguments"] = "{}"
                    tool_calls.append(tool_call)
        # logger.info(f"Tool Calls: {tool_calls}")
        response_dict = {"content": assistant_text}
        if tool_calls:
            response_dict["tool_calls"] = tool_calls
        return response_dict, usage

    except (APIError, RateLimitError) as e:
        # return {"content": f"OpenAI API error: {str(e)}"}
        return {"content": f"OpenAI API error: {str(e)}", "status": "error"}, {}
    except ValueError as ve:
        return {"content": f"ValueError: {str(ve)}", "status": "error"}, {}
    except Exception as ex:
        return {"content": f"Unexpected error: {str(ex)}", "status": "error"}, {}

def generate_with_openai(conversation: List[Dict], model_cfg: Dict, all_functions: List[Dict]) -> Tuple[Dict, Dict]:
    """
    Generate text using OpenAI's API.
    
    Args:
        conversation: The conversation history
        model_cfg: Configuration for the model
        all_functions: Available functions for the model to call
        
    Returns:
        Dict containing assistant_text and tool_calls
    """
    retry_num = 5
    api_key = model_cfg.get("api_key") or os.getenv("OPENAI_API_KEY")

    model_name = model_cfg["model_name"]
    temperature = model_cfg.get("temperature", None)
    top_p = model_cfg.get("top_p", None)
    max_tokens = model_cfg.get("max_tokens", None)

    while retry_num > 0:
        logger.info(f"Calling OpenAI API, model: {model_name}, temperature: {temperature}, top_p: {top_p}, max_tokens: {max_tokens}")
        retry_num -= 1
        client = OpenAI(api_key=api_key, base_url=model_cfg.get("base_url"))

        response_dict, usage = generate_with_openai_sync(
            client, model_name, conversation, all_functions,
            temperature, top_p, max_tokens
        )
        if response_dict.get("status") == "error":
            time.sleep(2)
            logger.info(f"Error: {response_dict.get('content')}. Retrying ... {retry_num} attempts left.")
            if retry_num == 3:
                logger.info("Saving conversation to debug_conversation.json")
                save_json("debug_msg.json", {'conversation': conversation, 'model_cfg': model_cfg, 'all_functions': all_functions})
        else:
            logger.info(f"OpenAI API call successful.")
            break
    return response_dict, usage
