import os
from dotenv import load_dotenv
load_dotenv()

OPENAI_API_KEY = os.getenv("OPENAI_API_KEY")

from typing import Tuple, Optional

from orchestrator.openai.config import Configuration
from orchestrator.prompts import orchestrator_prompt, yaml_orchestrator_prompt, list_orchestrator_prompt
from orchestrator.template import graph_template, yaml_template
from orchestrator.utils import extract_graph, validate_graph, extract_yaml_graph, validate_yaml_graph, validate_list, list2yaml
from orchestrator.schemas import SelectedBlocks

from langchain_openai import ChatOpenAI
from langchain_core.runnables import RunnableConfig

from agents.state import OverallState
from agents.block_registry import SELECTION_REGISTRY
from agents.utils import get_user_question


class Orchestrator:
    def __init__(self, config: RunnableConfig):
        self.config = Configuration.from_runnable_config(config)
        self.llm = ChatOpenAI(model=self.config.orchestrator_model, api_key=OPENAI_API_KEY)

    def _orchestrate_and_validate(self, message_stream: list) -> Tuple[bool, Optional[str]]:
        # Load block info from the block registry and re-format
        # block_info = "\n".join([f"- {block['block_name']}: {block['block_description']}" for block in REGISTRY])
        
        # Acquire info from the current state
        # current_sub_question = current_state.get('current_sub_question', get_user_question(current_state['messages']))
        # current_summary = current_state.get('current_summary')

        # # Format the orchestrator prompt
        # formatted_prompt = orchestrator_prompt.format(question=current_sub_question,
        #                                         current_summary=current_summary,
        #                                         list_of_building_blocks=block_info,
        #                                         json_template=graph_template)
        # # Invoke the orchestrator
        # response = self.llm.invoke(formatted_prompt)

        response = self.llm.invoke(message_stream)

        # Extract the JSON template from the response
        success, graph_json_str = extract_graph(response.content)
        
        if not success:
            # raise ValueError("Failed to extract valid JSON graph from orchestrator response")
            return False, "Failed to extract valid JSON graph from orchestrator response"

        # Validate the result
        is_valid, error_message = validate_graph(graph_json_str)
        
        if not is_valid:
            # raise ValueError(f"Invalid graph structure: {error_message}")
            return False, f"Invalid graph structure: {error_message}"
            
        return True, graph_json_str

    def _orchestrate_and_validate_yaml(self, message_stream: list) -> Tuple[bool, Optional[str]]:
        structured_llm = self.llm.with_structured_output(SelectedBlocks)
        response = structured_llm.invoke(message_stream)
        block_list = response.block_names
        success, error_message = validate_list(block_list)
        if not success:
            return False, f"Invalid block list: {error_message}"
        graph_yaml_str = list2yaml(block_list)
        is_valid, error_message = validate_yaml_graph(graph_yaml_str)
        if not is_valid:
            return False, f"Invalid graph structure: {error_message}"
        return True, graph_yaml_str

    def orchestrate_loop(self, current_state: OverallState) -> str:
        # Acquire info from the current state
        original_question = get_user_question(current_state['messages'])
        current_sub_question = current_state.get('current_sub_question', get_user_question(current_state['messages']))
        current_summary = current_state.get('current_summary')

        # Load block info from the block registry and re-format
        block_info = "\n".join([f"- Block Name: {block['block_name']} (type: {block['block_type']}): {block['block_description']}" for block in SELECTION_REGISTRY])

        if current_state.get('instruction_state', {}).get('orchestrator_instructions'):
            experiences = f"**Some experiences that might be userful:**\n{'\n'.join(current_state.get('instruction_state', {}).get('orchestrator_instructions'))}"
        else:
            experiences = ""

         # # Format the orchestrator prompt
        formatted_prompt = list_orchestrator_prompt.format(original_question=original_question,
                                                            question=current_sub_question,
                                                            current_summary=current_summary,
                                                            list_of_building_blocks=block_info,
                                                            experiences=experiences)
        message_stream = [
            ("system", formatted_prompt),
        ]

        while True:
            success, message = self._orchestrate_and_validate_yaml(message_stream)
            if success:
                break
            print(f"Orchestration failed: {message}, retrying...")
            message_stream.append(("user", message))
                
        return message