# from langchain_openai import ChatOpenAI
from langchain_qwq import ChatQwen
from agents.block_registry import SELECTION_REGISTRY
from instructor.schemas import Instruction
from instructor.prompts import *
from memory.schemas import WorkflowExecutionEntry
from typing import List
import os
import json

from dotenv import load_dotenv
load_dotenv(dotenv_path=".env")

# OPENAI_API_KEY = os.getenv("OPENAI_API_KEY")
DASHSCOPE_API_KEY = os.getenv("DASHSCOPE_API_KEY")

def generate_orchestrator_instructions(current_question: str, current_sub_goal: str, retrieved_entries: List[WorkflowExecutionEntry], model: str = "qwen-max") -> List[str]:
    llm = ChatQwen(
        model=model,
        api_key=DASHSCOPE_API_KEY,
    )
    
    structured_llm = llm.with_structured_output(Instruction)

    histories = ""

    for i,entry in enumerate(retrieved_entries, 1):
        question = entry.question
        sub_goal = entry.sub_question
        before_summary = entry.before_summary
        after_summary = entry.summary
        workflow = entry.workflow
        successful = entry.can_answer_sub_question
        execution_history = orchestrator_execution_history_template.format(question=question, sub_goal=sub_goal, before_summary=before_summary, after_summary=after_summary, workflow=workflow, successful=successful)
        histories += f"{i}. {execution_history}"
        histories += "\n\n"

    # 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])

    prompt = instruct_orchestrator_prompt.format(execution_history=histories, question=current_question, sub_goal=current_sub_goal, available_building_blocks=block_info)

    response = structured_llm.invoke(prompt)
    return response.instructions

def generate_search_instructions(current_question: str, current_sub_goal: str, retrieved_entries: List[WorkflowExecutionEntry], model: str = "qwen2.5-32b-instruct") -> List[str]:
    llm = ChatQwen(
        model=model,
        api_key=DASHSCOPE_API_KEY,
    )

    structured_llm = llm.with_structured_output(Instruction)

    histories = ""


    for i,entry in enumerate(retrieved_entries, 1):
        question = entry.question
        sub_goal = entry.sub_question
        search_keywords = ", ".join(entry.searcher_execution_memory.new_keywords_added)
        search_results = json.dumps([result.dict() for result in entry.searcher_execution_memory.new_search_results], indent=4)
        successful = entry.can_answer_sub_question
        execution_history = search_execution_history_template.format(question=question, sub_goal=sub_goal, search_keywords=search_keywords, search_results=search_results, successful=successful)
        histories += f"{i}. {execution_history}"
        histories += "\n\n"

    prompt = instruct_search_prompt.format(execution_history=histories, question=current_question, sub_goal=current_sub_goal)

    response = structured_llm.invoke(prompt)
    return response.instructions

def generate_browse_instructions(current_question: str, current_sub_goal: str, retrieved_entries: List[WorkflowExecutionEntry], model: str = "qwen2.5-32b-instruct") -> List[str]:
    llm = ChatQwen(
        model=model,
        api_key=DASHSCOPE_API_KEY,
    )

    structured_llm = llm.with_structured_output(Instruction)

    histories = ""

    for i,entry in enumerate(retrieved_entries, 1):
        question = entry.question
        sub_goal = entry.sub_question
        new_found_references = json.dumps([ref.dict() for ref in entry.browser_execution_memory.new_references_found], indent=4)
        successful = entry.can_answer_sub_question

        execution_history = browse_execution_history_template.format(question=question, sub_goal=sub_goal, new_found_references=new_found_references, successful=successful)

        histories += f"{i}. {execution_history}"
        histories += "\n\n"

    prompt = instruct_browse_prompt.format(execution_history=histories, question=current_question, sub_goal=current_sub_goal)

    response = structured_llm.invoke(prompt)
    return response.instructions