import logging
import os
import warnings
from .model_base import BlackBoxModelBase
from openai import OpenAI
from .conversation import get_conv_template
from httpx import URL
class OpenaiModel(BlackBoxModelBase):
    def __init__(
        self,
        model_name: str,
        api_keys: str,
        generation_config=None,
        base_url: str | URL | None = None,
        *,
        role: str | None = None,
    ):
        """
        Initializes the OpenAI model with necessary parameters.
        :param str model_name: The name of the model to use.
        :param str api_keys: API keys for accessing the OpenAI service.
        :param str template_name: The name of the conversation template, defaults to 'chatgpt'.
        :param dict generation_config: Configuration settings for generation, defaults to an empty dictionary.
        :param str|URL base_url: The base URL for the OpenAI API, defaults to None.
        """
        # Persist base_url so other components (e.g., unified judge) can reliably detect local usage.
        # Keep as the original type (str|URL|None); consumers can cast to str as needed.
        self.base_url = base_url
        # Optional role tag to allow per-role determinism controls (e.g., attack/target/eval).
        self.role = (role or "").strip().lower() or None
        self.client = OpenAI(api_key=api_keys, base_url=base_url)
        self.model_name = model_name
        self.conversation = get_conv_template('chatgpt')
        self.generation_config = generation_config if generation_config is not None else {}

    def set_system_message(self, system_message: str):
        """
        Sets a system message for the conversation.
        :param str system_message: The system message to set.
        """
        self.conversation.system_message = system_message

    def generate(self, messages, clear_old_history=True, **kwargs):
        """
        Generates a response based on messages that include conversation history.
        :param list[str]|str messages: A list of messages or a single message string.
                                       User and assistant messages should alternate.
        :param bool clear_old_history: If True, clears the old conversation history before adding new messages.
        :return str: The response generated by the OpenAI model based on the conversation history.
        """
        if messages is None:
            raise ValueError(
                "OpenaiModel.generate(messages=None): caller passed None. "
                "Pass a string, list[str], or OpenAI-formatted list[dict]."
            )

        # Determinism defaults (align with Multi-turn-jailbreak UnifiedLLMClient semantics):
        # If ENABLE_DETERMINISM=true (default) and caller didn't specify, set:
        # - top_p=1, n=1, seed=123
        # (Best-effort: some local OpenAI-compatible servers may ignore seed.)
        merged_cfg = dict(self.generation_config or {})
        merged_cfg.update(kwargs or {})  # call-time kwargs override generation_config
        # Per-role override:
        # - If ENABLE_DETERMINISM_<ROLE> is set, it takes precedence.
        # - Otherwise fall back to global ENABLE_DETERMINISM (default true).
        enable_determinism = None
        if self.role:
            role_key = f"ENABLE_DETERMINISM_{self.role.upper()}"
            if os.getenv(role_key) is not None:
                enable_determinism = os.getenv(role_key, "true").lower() == "true"
        if enable_determinism is None:
            enable_determinism = os.getenv("ENABLE_DETERMINISM", "true").lower() == "true"
        if enable_determinism:
            # For fully deterministic behavior, force these values for this model instance.
            # (Attack/rewrite are handled by per-role enable flags; default attack is OFF in scripts.)
            merged_cfg["temperature"] = 0
            merged_cfg["top_p"] = 1
            merged_cfg["n"] = 1
            merged_cfg["seed"] = 123

        # If caller passed OpenAI-formatted messages, bypass template building and send as-is
        if isinstance(messages, list) and messages and isinstance(messages[0], dict) and 'role' in messages[0]:
            payload_messages = messages
            # Prepend system message if configured but missing
            if self.conversation.system_message and not any(m.get('role') == 'system' for m in payload_messages):
                payload_messages = (
                    [{'role': 'system', 'content': self.conversation.system_message}] + payload_messages
                )
            response = self.client.chat.completions.create(
                model=self.model_name,
                messages=payload_messages,
                **merged_cfg
            )
            return response.choices[0].message.content

        # Otherwise, treat input as plain text turns and build conversation via template
        if clear_old_history:
            self.conversation.messages = []
        if isinstance(messages, str):
            messages = [messages]
        for index, message in enumerate(messages):
            self.conversation.append_message(self.conversation.roles[index % 2], message)
        response = self.client.chat.completions.create(
            model=self.model_name,
            messages=self.conversation.to_openai_api_messages(),
            **merged_cfg
        )
        return response.choices[0].message.content

    def batch_generate(self, conversations, **kwargs):
        """
        Generates responses for multiple conversations in a batch.
        :param list[list[str]]|list[str] conversations: A list of conversations, each as a list of messages.
        :return list[str]: A list of responses for each conversation.
        """
        responses = []
        for conversation in conversations:
            if isinstance(conversation, str):
                warnings.warn('For batch generation based on several conversations, provide a list[str] for each conversation. '
                              'Using list[list[str]] will avoid this warning.')
            responses.append(self.generate(conversation, **kwargs))
        return responses
