import heapq
import random
from collections import deque
from typing import Callable, Dict, List, Tuple
from utils import IDLE_TASK_ID, Task

random.seed(1)


def rate_monotonic_schedule(tasks: List[Task], runtime: int) -> List[int]:
    """
    Schedules tasks according to the rate-monotonic scheduling algorithm where the period is inversely proportional to the priority of a task. (i.e. the task with the shortest period has highest priority.)
    """
    req_utilization = sum(task.exectime / task.period for task in tasks)
    n = len(tasks)
    utilization_rm = n * (2 ** (1 / n) - 1)
    assert (
        req_utilization <= utilization_rm
    ), "Tasks are not schedulable using the RM scheduling algorithm."

    prioritized_tasks = [
        task.id for task in sorted(tasks, key=lambda task: task.period)
    ]

    def rate_monotonic_scheduling(todo_tasks: List[Task]) -> int:
        for task_id in prioritized_tasks:
            if task_id in todo_tasks:
                return task_id
        return IDLE_TASK_ID

    return schedule(
        tasks=tasks, runtime=runtime, scheduling_algorithm=rate_monotonic_scheduling
    )


def randomly_schedule(tasks: List[Task], runtime: int) -> List[int]:
    """
    Randomly schedules tasks. Assumes that if there's any task to be worked on, it will work on it rather than stay idle.
    """

    def random_scheduling(todo_tasks: List[Task]) -> int:
        return random.choice(todo_tasks)

    return schedule(
        tasks=tasks, runtime=runtime, scheduling_algorithm=random_scheduling
    )


def schedule(
    tasks: List[Task],
    runtime: int,
    scheduling_algorithm: Callable[[List[int]], int],
    todo_dict: Dict[int, deque] = None
) -> List[int]:
    """
    Generic scheduler: at each time step, releases new jobs into todo_dict,
    then calls scheduling_algorithm(todo_tasks) to pick which task_id to run.
    """

    # Min-heap to track next release times
    tasks_queue: List[Tuple[int, Task]] = []
    for task in tasks:
        heapq.heappush(tasks_queue, (0, task))

    # Tracks which jobs are ready: todo_tasks list + their (deadline, rem_exec) deques
    todo_tasks: List[int] = []
    if todo_dict is None:
        todo_dict = {}  # type: Dict[int, deque]

    schedule: List[int] = []

    for t in range(runtime):
        # release new jobs whose release time == t
        while tasks_queue and tasks_queue[0][0] == t:
            _, task = heapq.heappop(tasks_queue)

            if task.id not in todo_dict:
                todo_dict[task.id] = deque()
                todo_tasks.append(task.id)

            # append (absolute_deadline, remaining_exec) for this instance
            todo_dict[task.id].append([t + task.deadline, task.exectime])

            # schedule next release
            heapq.heappush(tasks_queue, (t + task.period, task))

        # pick next task
        if not todo_tasks:
            task_id = IDLE_TASK_ID
        else:
            task_id = scheduling_algorithm(todo_tasks)

        schedule.append(task_id)

        # execute one time unit
        if task_id != IDLE_TASK_ID:
            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]:
                    # no more pending jobs for this task
                    del todo_dict[task_id]
                    todo_tasks.remove(task_id)

    return schedule
	
def earliest_deadline_first(tasks: List[Task], runtime: int) -> List[int]:

    todo_dict: Dict[int, deque] = {}

    def edf_selector(todo_tasks: List[int]) -> int:
        if not todo_tasks:
            return IDLE_TASK_ID
  
        return min(todo_tasks, key=lambda tid: todo_dict[tid][0][0])

    return schedule(tasks, runtime, edf_selector, todo_dict)