from __future__ import annotations
from typing import Dict, Any, List
import os, re, math

# -------- path utils --------
def normpath(p: str) -> str:
    if p is None:
        return ""
    return os.path.normpath(p.replace("\\\\", "/").replace("\\", "/"))

def get_step_idx_from_path(p: str) -> int:
    m = re.search(r"(?:h|v)_step_(\d+)\.json$", p)
    return int(m.group(1)) if m else -1

# -------- time utils --------
def detect_time_unit(value: float) -> str:
    v = float(value)
    if v >= 1e12: return "ns"
    if v >= 1e9:  return "us"
    if v >= 1e6:  return "ms"
    return "s"

def to_ms(value: float, unit: str) -> float:
    if unit == "ns": return float(value) / 1e6
    if unit == "us": return float(value) / 1e3
    if unit == "ms": return float(value)
    if unit == "s":  return float(value) * 1e3
    return float(value)

def pct(arr: List[float], q: float) -> float:
    if not arr:
        return 0.0
    arr = sorted(arr)
    if q <= 0: return arr[0]
    if q >= 100: return arr[-1]
    k = (len(arr)-1) * q/100.0
    f = math.floor(k); c = math.ceil(k)
    if f == c: return arr[int(k)]
    return arr[f] + (arr[c] - arr[f]) * (k - f)

def normalize_op_name(name: str) -> str:
    if not name:
        return ""
    s = str(name).strip()
    s = re.sub(r'\s+at\s+.*$', '', s)
    s = re.sub(r',\s*op_id\s*=\s*\d+', '', s, flags=re.IGNORECASE)
    s = re.sub(r',\s*seq\s*=\s*\d+', '', s, flags=re.IGNORECASE)
    s = s.rstrip(' ,')
    return s
