# chronoscore/schedulers.py
"""
Classical scheduling baselines and a generic schedule() helper used for evaluation.
Implements Rate Monotonic (RM) and Earliest Deadline First (EDF).
"""
import heapq
from collections import deque
from typing import List, Callable, Dict
from data.dataset_builder import Task
from chronoscore.environment import IDLE_TASK_ID

def schedule(tasks: List[Task], runtime: int, scheduling_algorithm: Callable[[List[int]], int]) -> List[int]:
    """
    Generic discrete-time scheduler: releases jobs and calls scheduling_algorithm(todo_tasks) each step.
    """
    # min-heap of (next_release_time, task)
    tasks_queue = [(0, t) for t in tasks]
    heapq.heapify(tasks_queue)
    todo_dict = {}  # task_id -> deque of [deadline, rem_exec]
    todo_tasks = []
    schedule = []
    for t in range(runtime):
        while tasks_queue and tasks_queue[0][0] == t:
            _, task = heapq.heappop(tasks_queue)
            todo_dict.setdefault(task.id, deque()).append([t + task.deadline, task.exectime])
            if task.id not in todo_tasks:
                todo_tasks.append(task.id)
            heapq.heappush(tasks_queue, (t + task.period, task))

        if not todo_tasks:
            schedule.append(IDLE_TASK_ID)
        else:
            task_id = scheduling_algorithm(todo_tasks)
            schedule.append(task_id)
            # execute
            todo_dict[task_id][0][1] -= 1
            if todo_dict[task_id][0][1] == 0:
                todo_dict[task_id].popleft()
                if not todo_dict[task_id]:
                    todo_tasks.remove(task_id)
    return schedule

def rate_monotonic_schedule(tasks: List[Task], runtime: int) -> List[int]:
    prioritized = [t.id for t in sorted(tasks, key=lambda x: x.period)]
    def rm_selector(todo):
        for tid in prioritized:
            if tid in todo:
                return tid
        return IDLE_TASK_ID
    return schedule(tasks, runtime, rm_selector)

def earliest_deadline_first(tasks: List[Task], runtime: int) -> List[int]:
    # maintain todo_dict expiration times inside closure via mutable capture
    todo_dict = {}
    def edf_selector(todo):
        if not todo:
            return IDLE_TASK_ID
        # choose tid with smallest current absolute deadline
        return min(todo, key=lambda tid: todo_dict[tid][0][0])  # compare first tuple's deadline
    # wrapper schedule that passes todo_dict
    # reuse generic schedule by patching schedule's local variable? simpler: reimplement small version here
    tasks_queue = [(0, t) for t in tasks]
    heapq.heapify(tasks_queue)
    todo_dict = {}
    todo_tasks = []
    schedule_list = []
    for t in range(runtime):
        while tasks_queue and tasks_queue[0][0] == t:
            _, task = heapq.heappop(tasks_queue)
            todo_dict.setdefault(task.id, deque()).append([t + task.deadline, task.exectime])
            if task.id not in todo_tasks:
                todo_tasks.append(task.id)
            heapq.heappush(tasks_queue, (t + task.period, task))
        if not todo_tasks:
            schedule_list.append(IDLE_TASK_ID)
        else:
            # pick min deadline
            best = min(todo_tasks, key=lambda tid: todo_dict[tid][0][0])
            schedule_list.append(best)
            todo_dict[best][0][1] -= 1
            if todo_dict[best][0][1] == 0:
                todo_dict[best].popleft()
                if not todo_dict[best]:
                    todo_tasks.remove(best)
    return schedule_list
