# utils/task_utils.py
"""
Task utilities: Task NamedTuple, LCM utilities, read/write, sanity checks, printing helpers.
This file centralizes small helpers used across the project.
"""
from typing import List, NamedTuple
from math import gcd
from textwrap import dedent
import json
import os

class Task(NamedTuple):
    id: int
    period: int
    exectime: int
    deadline: int

IDLE_TASK_ID = -1

def read_tasks(filename: str) -> List[Task]:
    """
    Read CSV or simple text lines: either "id, period, exectime, deadline" or "period, exectime, deadline".
    """
    tasks = []
    with open(filename, "r", encoding="utf-8") as fh:
        for i, ln in enumerate(fh):
            ln = ln.strip()
            if not ln or ln.startswith("#"):
                continue
            parts = [int(x.strip()) for x in ln.split(",")]
            if len(parts) == 4:
                tid, p, e, d = parts
            else:
                p, e, d = parts
                tid = i
            tasks.append(Task(id=tid, period=p, exectime=e, deadline=d))
    return tasks

def write_tasks(tasks: List[Task], filename: str):
    os.makedirs(os.path.dirname(filename) or ".", exist_ok=True)
    with open(filename, "w", encoding="utf-8") as fh:
        for t in tasks:
            fh.write(f"{t.id}, {t.period}, {t.exectime}, {t.deadline}\n")

def lcm_of_periods(tasks: List[Task]) -> int:
    """Compute least common multiple of periods."""
    l = 1
    for t in tasks:
        l = l * t.period // gcd(l, t.period)
    return l

def assert_is_schedulable(tasks: List[Task], tol: float = 1.0):
    util = sum(t.exectime / t.period for t in tasks)
    assert util <= tol + 1e-9, f"Total utilization {util} > {tol}"

def print_by_task(tasks: List[Task], schedule: List[int]):
    """
    Print per-task execution timeline similar to earlier helper.
    """
    task_executions = {t.id: [] for t in tasks}
    for idx, a in enumerate(schedule):
        if a != IDLE_TASK_ID:
            task_executions[a].append(idx)
    for t in tasks:
        print(f"Task {t.id} executed at time steps: {task_executions[t.id]}")
