import os
from typing import Any, Optional

from dotenv import find_dotenv, load_dotenv
from langchain_huggingface import ChatHuggingFace, HuggingFacePipeline
from langchain_openai import AzureChatOpenAI, ChatOpenAI
import logging

class LLM_Factory:
    """Factory class to create LLM instances based on the model name.
    This is helpful for not initializing multiple identical LLMs and instead always use the same instance."""

    llm: Optional[Any] = None

    @staticmethod
    def initialize_llm(model_name: Optional[str] = None) -> Any:
        """Initialize and return the LLM instance based on the model name."""
        if LLM_Factory.llm is None:
            env_path = find_dotenv()
            load_dotenv(env_path)

            if model_name is None:
                model_name = os.getenv("SELECTED_MODEL_NAME")

            seed = os.getenv("SEED")

            match model_name:
                case "openai":
                    # Disable OpenAI and httpx logging
                    # Configure logging level for specific loggers by name
                    logging.getLogger("openai").setLevel(logging.ERROR)
                    logging.getLogger("httpx").setLevel(logging.ERROR)
                    
                    # Use Azure Chat OpenAI
                    model = os.getenv("MODEL_NAME")
                    if model == "gpt-4o-mini":
                        llm = AzureChatOpenAI(
                            azure_deployment=os.getenv("AZURE_OPENAI_DEPLOYMENT_NAME"),
                            api_version=os.getenv("AZURE_OPENAI_API_VERSION"),
                            temperature=0.5,
                            max_tokens=10000,
                            max_retries=10,
                            model=model,
                            seed=seed
                        )
                    elif model == "o3-mini":
                        llm = AzureChatOpenAI(
                            azure_deployment=os.getenv("AZURE_OPENAI_DEPLOYMENT_NAME"),
                            api_version=os.getenv("AZURE_OPENAI_API_VERSION"),
                            # temperature=0.5,
                            max_completion_tokens=10000,
                            max_retries=10,
                            model=model,
                            reasoning_effort="low",
                            seed=seed,
                        )

                    # Ensure environment variables for Azure OpenAI API are loaded
                    os.getenv("AZURE_OPENAI_ENDPOINT")
                    os.getenv("AZURE_OPENAI_API_KEY")

                case "Qwen/Qwen3-14B":
                    # We used VLLM. Open an endpoint by using "vllm serve Qwen/Qwen3-14B ..." 
                    llm = ChatOpenAI(
                        model='Qwen/Qwen3-14B',
                        openai_api_key="EMPTY",
                        openai_api_base="YOUR ENDPOINT FOR HERE",
                        max_tokens=10000,
                        temperature=0.5,
			            seed=4
                        )
                
            LLM_Factory.llm = llm

        return LLM_Factory.llm


def print_stream_and_return_state(stream):
    last_state = None
    for s in stream:
        message = s["messages"][-1]
        if isinstance(message, tuple):
            print(message)
        else:
            message.pretty_print()
        last_state = s
    return last_state


def init_custom_llm(model: str):
    pass
