from __future__ import annotations

import json
import os
import uuid
from datetime import datetime, timezone
from pathlib import Path
from typing import Any

from .config import METRICS_DIR
from .log_utils import slugify


def _utc_now_iso() -> str:
    return datetime.now(timezone.utc).isoformat()


def _run_log_path(run_id: str) -> Path:
    return METRICS_DIR / f"run_{run_id}.jsonl"


def start_run(
    pipeline: str,
    *,
    stage: int | None = None,
    name_tag: str | None = None,
    data_file: str | None = None,
    extra: dict[str, Any] | None = None,
) -> str:
    stage_tag = f"stage{stage}" if stage is not None else "stage"
    sanitized_tag = (name_tag or "run").replace(os.sep, "_").replace(" ", "_")
    run_id = (
        f"{pipeline}_{stage_tag}_{sanitized_tag}_"
        f"{datetime.now(timezone.utc).strftime('%Y%m%dT%H%M%SZ')}_{os.getpid()}_{uuid.uuid4().hex[:8]}"
    )
    payload: dict[str, Any] = {"pipeline": pipeline, "stage": stage, "data_file": data_file, "name_tag": name_tag}
    if extra:
        payload.update(extra)
    log_event(run_id, "run_start", payload)
    return run_id


def log_event(run_id: str, event_type: str, data: dict[str, Any]) -> None:
    METRICS_DIR.mkdir(parents=True, exist_ok=True)
    event = {
        "ts": _utc_now_iso(),
        "run_id": run_id,
        "event": event_type,
        "data": data,
    }
    with _run_log_path(run_id).open("a", encoding="utf-8") as f:
        f.write(json.dumps(event, ensure_ascii=False) + "\n")


def finish_run(run_id: str, summary: dict[str, Any]) -> None:
    log_event(run_id, "run_end", summary)
    summary_path = METRICS_DIR / f"run_{run_id}_summary.json"
    summary_path.write_text(json.dumps(summary, ensure_ascii=False, indent=2), encoding="utf-8")

    # Convenience pointers for humans: keep "latest" summaries per pipeline (best-effort).
    pipeline = summary.get("pipeline")
    if isinstance(pipeline, str) and pipeline.strip():
        key = slugify(pipeline, max_len=40)
        (METRICS_DIR / f"latest_{key}_run_id.txt").write_text(run_id + "\n", encoding="utf-8")
        (METRICS_DIR / f"latest_{key}_summary.json").write_text(
            json.dumps(summary, ensure_ascii=False, indent=2),
            encoding="utf-8",
        )
    (METRICS_DIR / "latest_run_id.txt").write_text(run_id + "\n", encoding="utf-8")
    (METRICS_DIR / "latest_summary.json").write_text(
        json.dumps(summary, ensure_ascii=False, indent=2),
        encoding="utf-8",
    )
