"""
Lightweight experiment logging without third-party dependencies.
Saves experiment configs and results to JSON files in runs/ directory.
"""

import json
import os
import subprocess
from datetime import datetime
from pathlib import Path
from typing import Any

RUNS_DIR = Path("runs")


def get_git_hash() -> str | None:
    """Get current git commit hash, or None if not in a git repo."""
    try:
        result = subprocess.run(
            ["git", "rev-parse", "--short", "HEAD"],
            capture_output=True,
            text=True,
            timeout=5,
        )
        if result.returncode == 0:
            return result.stdout.strip()
    except (subprocess.TimeoutExpired, FileNotFoundError):
        pass
    return None


def generate_run_id(objective: str) -> str:
    """Generate a unique run ID based on timestamp and objective."""
    timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
    return f"{timestamp}_{objective}"


def log_experiment(
    config: dict[str, Any],
    results: dict[str, Any],
    run_id: str | None = None,
    run_dir: Path = RUNS_DIR,
) -> Path:
    """
    Log an experiment's config and results to a JSON file.

    Args:
        config: Hyperparameters and settings (n_arms, n_rounds, objective, etc.)
        results: Outcomes (regrets, final_swf, optimal_swf, etc.)
        run_id: Optional custom run ID. If None, auto-generated from timestamp.

    Returns:
        Path to the saved JSON file.
    """
    run_dir.mkdir(exist_ok=True)

    if run_id is None:
        objective = config.get("objective", "unknown")
        run_id = generate_run_id(objective)

    # Convert numpy/torch arrays to lists for JSON serialization
    def serialize(obj):
        if hasattr(obj, "tolist"):
            return obj.tolist()
        if hasattr(obj, "item"):
            return obj.item()
        return obj

    serialized_results = {k: serialize(v) for k, v in results.items()}
    serialized_config = {k: serialize(v) for k, v in config.items()}

    record = {
        "run_id": run_id,
        "timestamp": datetime.now().isoformat(),
        "git_hash": get_git_hash(),
        "config": serialized_config,
        "results": serialized_results,
    }

    filepath = run_dir / f"{run_id}.json"
    with open(filepath, "w") as f:
        json.dump(record, f, indent=2)

    print(f"Logged experiment to {filepath}")
    return filepath


def load_run(run_id: str, run_dir: Path = RUNS_DIR) -> dict:
    """Load a single run by its ID."""
    filepath = run_dir / f"{run_id}.json"
    with open(filepath) as f:
        return json.load(f)


def load_all_runs() -> list[dict]:
    """Load all runs from the runs/ directory."""
    if not RUNS_DIR.exists():
        return []

    runs = []
    for filepath in sorted(RUNS_DIR.glob("*.json")):
        with open(filepath) as f:
            runs.append(json.load(f))
    return runs


def filter_runs(
    runs: list[dict],
    objective: str | None = None,
    n_arms: int | None = None,
    num_alloc: int | None = None,
    **kwargs,
) -> list[dict]:
    """Filter runs by config values."""
    filtered = runs

    if objective is not None:
        filtered = [r for r in filtered if r["config"].get("objective") == objective]
    if n_arms is not None:
        filtered = [r for r in filtered if r["config"].get("n_arms") == n_arms]
    if num_alloc is not None:
        filtered = [r for r in filtered if r["config"].get("num_alloc") == num_alloc]

    for key, value in kwargs.items():
        filtered = [r for r in filtered if r["config"].get(key) == value]

    return filtered
