from __future__ import annotations

import fcntl
import hashlib
import json
import os
import time
import uuid
from typing import Any, Dict, Optional, Tuple

from python_src.io_utils import read_jsonl


class EventLog:
    """Append-only event log with idempotency support."""

    def __init__(self, experiments_dir: Optional[str] = None, repo_root: Optional[str] = None) -> None:
        if repo_root is None:
            repo_root = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
        self.repo_root = repo_root
        self.experiments_dir = experiments_dir or os.path.join(repo_root, "experiments")

    def _exp_dir(self, exp_id: str) -> str:
        return os.path.join(self.experiments_dir, exp_id)

    def _events_path(self, exp_id: str) -> str:
        return os.path.join(self._exp_dir(exp_id), "events.jsonl")

    def events_path(self, exp_id: str) -> str:
        return self._events_path(exp_id)

    def _lock_path(self, exp_id: str) -> str:
        return os.path.join(self._exp_dir(exp_id), "events.lock")

    @staticmethod
    def build_dedup_key(
        exp_id: str,
        event_type: str,
        task_tag: Optional[str],
        piece_id: Optional[str],
        run_id: Optional[str],
    ) -> str:
        raw = f"{exp_id}|{event_type}|{task_tag or ''}|{piece_id or ''}|{run_id or ''}"
        return hashlib.sha1(raw.encode("utf-8")).hexdigest()

    def _has_dedup_key(self, exp_id: str, dedup_key: str) -> bool:
        events_path = self._events_path(exp_id)
        if not os.path.exists(events_path):
            return False
        with open(events_path, "r", encoding="utf-8") as handle:
            for line in handle:
                line = line.strip()
                if not line:
                    continue
                try:
                    data = json.loads(line)
                except json.JSONDecodeError:
                    continue
                if data.get("dedup_key") == dedup_key:
                    return True
        return False

    def append_event(
        self,
        exp_id: str,
        event_type: str,
        task_tag: Optional[str],
        piece_id: Optional[str],
        run_id: Optional[str],
        payload: Optional[Dict[str, Any]] = None,
    ) -> Tuple[bool, Dict[str, Any]]:
        """Append an event if not already present. Returns (appended, event)."""
        exp_dir = self._exp_dir(exp_id)
        os.makedirs(exp_dir, exist_ok=True)

        lock_path = self._lock_path(exp_id)
        os.makedirs(os.path.dirname(lock_path), exist_ok=True)

        dedup_key = self.build_dedup_key(exp_id, event_type, task_tag, piece_id, run_id)
        dedup_enabled = event_type != "manual_resume"
        event = {
            "event_id": uuid.uuid4().hex,
            "dedup_key": dedup_key,
            "exp_id": exp_id,
            "event_type": event_type,
            "task_tag": task_tag,
            "piece_id": piece_id,
            "run_id": run_id,
            "timestamp": int(time.time()),
            "payload": payload or {},
        }

        with open(lock_path, "w") as lock_file:
            fcntl.flock(lock_file.fileno(), fcntl.LOCK_EX)
            try:
                if dedup_enabled and self._has_dedup_key(exp_id, dedup_key):
                    return False, event
                events_path = self._events_path(exp_id)
                with open(events_path, "a", encoding="utf-8") as handle:
                    handle.write(json.dumps(event, ensure_ascii=True))
                    handle.write("\n")
            finally:
                fcntl.flock(lock_file.fileno(), fcntl.LOCK_UN)

        return True, event

    def read_latest_event(self, exp_id: str) -> Optional[Dict[str, Any]]:
        """Read the latest event record for an experiment."""
        rows = self.read_recent_events(exp_id, 1)
        if not rows:
            return None
        return rows[-1]

    def read_recent_events(self, exp_id: str, limit: int) -> list[Dict[str, Any]]:
        rows = read_jsonl(self._events_path(exp_id), limit=max(0, limit))
        return [row for row in rows if isinstance(row, dict)]
