import asyncio
import yaml
import json
import time
from typing import Dict, List, Optional

from core.environment import WorkEnvironment
from core.history_manager import HistoryManager
from core.messages_parser import MessagesParser
from agents.organizer import Organizer
from agents.executor import Executor
from agents.explorer import Explorer
# from data.MOCK_DATASET import MOCK_DATASET



async def run_single_case(idx: int, dataset_item: Dict, config: Dict, history_manager: HistoryManager, save_dir: str):

    data_id = str(dataset_item.get("meta", {}).get("id") or dataset_item.get("id", "") or "unknown")
    dataset_id = f"{idx:06d}_{data_id}"
    # print(f"\n" + "="*60)
    # print(f">>> [System] Starting Pipeline for Dataset ID: {dataset_id}")


    msg_parser = MessagesParser()
    try:
        parsed_data = msg_parser.load_datasets_messages(dataset_item)
    except Exception as e:
        print(f"!!! [Error] Failed to parse dataset item {dataset_id}: {e}")
        return

    initial_query = parsed_data["initial_query"]
    tools = parsed_data["tools"]
    full_messages = parsed_data["full_messages"]

    org_cfg = config.get('organizer_model', config.get('model', {}))
    exec_cfg = config.get('executor_model', config.get('model', {}))
    exp_cfg = config.get('explorer_model', config.get('model', {}))

    organizer = Organizer(
        history_manager=history_manager,
        api_key=org_cfg.get('api_key'),
        base_url=org_cfg.get('base_url'),
        model=org_cfg.get('model'),
        temperature=org_cfg.get('temperature', 0.0)
    )

    executor_template = Executor(
        api_key=exec_cfg.get('api_key'),
        base_url=exec_cfg.get('base_url'),
        model=exec_cfg.get('model'),
        temperature=exec_cfg.get('temperature', 0.0)
    )

    explorer = Explorer(
        api_key=exp_cfg.get('api_key'),
        base_url=exp_cfg.get('base_url'),
        model=exp_cfg.get('model'),
        temperature=exp_cfg.get('temperature', 0.7)
    )

    # print(f">>> [Organizer] Starting Turn 0 for Query: '{initial_query}'")
    turn_index = 0
    explorer_turn_count = 0
    step = 0
    finish_msg = ''
    pending_tasks = []

    current_env, initial_dispatch_list = await organizer.start(initial_query, tools, dataset_id)
    # current_env.debug_print_executors()

    current_msg_index = 0
    for i, msg in enumerate(full_messages):
        if msg['role'] == 'user' and msg['content'] == initial_query:
            current_msg_index = i
            break

    if initial_dispatch_list:
        for context in initial_dispatch_list:
            current_env.register_executor(context)
            task = asyncio.create_task(executor_template.execute(context))
            pending_tasks.append(task)

    # print(f">>> [System] Entering Event Loop for ID: {dataset_id}...")
    loop_count = 0
    MAX_LOOPS = 200

    while True:
        loop_count += 1
        if loop_count > MAX_LOOPS:
            print(f"!!! [System] Max loops reached for {dataset_id}. Force stopping.")
            break

        if not pending_tasks and not current_env.active_registry:

            if finish_msg != '':
                print(f"\n✅ [Goal Reached] ID: {dataset_id} | Turn {turn_index} Finished. Msg: {finish_msg}")
                next_user_query = None

                for i in range(current_msg_index + 1, len(full_messages)):
                    if full_messages[i]['role'] == 'user':
                        next_user_query = full_messages[i]['content']
                        current_msg_index = i
                        # print(f">>> [Dataset] Found next user turn: '{next_user_query}'")
                        break

                use_explorer = config.get('system', {}).get('use_explorer', False)
                max_explorer_turn = config.get('system', {}).get('explorer_turn', 0)

                if not next_user_query and use_explorer:
                    if explorer_turn_count < max_explorer_turn:
                        reasoning_content, next_user_query = await explorer.get_append_query(current_env)
                        explorer_turn_count += 1

                if next_user_query:

                    previous_objective = current_env.objective[:]
                    previous_tools = current_env.tools
                    previous_scratchpad = current_env.scratchpads[-1]

                    current_env = WorkEnvironment()
                    current_env.tools = previous_tools
                    current_env.objective = previous_objective
                    current_env.objective.append(next_user_query) # Append New Query
                    current_env.scratchpads = [previous_scratchpad] # Inherit Context


                    turn_index += 1
                    step = 0
                    finish_msg = ''

                    # print(f">>> [Organizer] Planning for Turn {turn_index} (Step {step})...")

                    current_env, dispatch_list = await organizer.start(
                        initial_message=next_user_query,
                        tools=current_env.tools,
                        dataset_id=dataset_id,
                        env=current_env
                    )

                    for context in dispatch_list:
                        # print(f"    - Dispatching new executor: {context}...")
                        current_env.register_executor(context)
                        new_task = asyncio.create_task(executor_template.execute(context))
                        pending_tasks.append(new_task)
                    continue
                else:
                    # print(f">>> [System] No further queries for {dataset_id}. Case Complete.")
                    break

            # Edge Case
            if not current_env.active_registry and finish_msg == '':
                 pass # Plan

        if pending_tasks:
            done, pending = await asyncio.wait(pending_tasks, return_when=asyncio.FIRST_COMPLETED)
            pending_tasks = list(pending)

            for task in done:
                executor_return = task.result()
                # Phase 1: Update
                current_env = organizer.update_env(current_env, executor_return)

                # Phase 2: Plan
                step += 1
                thought, dispatch_list, finish_msg = await organizer.plan_and_dispatch(
                    current_env, dataset_id, step, turn_index
                )

                for context in dispatch_list:
                    new_task = asyncio.create_task(executor_template.execute(context))
                    pending_tasks.append(new_task)
        else:
            # Idle check
            step += 1
            thought, dispatch_list, finish_msg = await organizer.plan_and_dispatch(
                current_env, dataset_id, step, turn_index
            )

            if finish_msg != '':
                continue

            if not dispatch_list:
                # print(thought)
                # current_env.debug_print_executors()
                print(f"!!! [System] Deadlock detected for {dataset_id}: No tasks, no finish_msg. Aborting.")
                break

            for context in dispatch_list:
                new_task = asyncio.create_task(executor_template.execute(context))
                pending_tasks.append(new_task)


    system_msg = next((m['content'] for m in full_messages if m.get("role") == "system"), None)
    history_records = history_manager.get_history(dataset_id)

    if history_records:
        msg_parser.transform_ENV_to_messages(system_msg, history_records, dataset_id, tools, save_dir)
        history_manager.clear_buffer(dataset_id)


async def single_test():
    with open("configs/explorer_test.yaml", 'r') as f:
        config = yaml.safe_load(f)

    from datetime import datetime
    save_dir = f"data/MOCK_DATASET/{datetime.now().strftime('%Y-%m-%d %H:%M:%S')}"

    history_manager = HistoryManager(save_dir)


    # print(MOCK_DATASET[:1])
    # assert 0
    # for idx, item in enumerate(MOCK_DATASET):
    #     await run_single_case(idx, item, config, history_manager, save_dir)
    #     time.sleep(1)

    print("\n" + "="*60)
    print(">>> [Main] All cases finished.")

if __name__ == "__main__":
    asyncio.run(single_test())