from __future__ import annotations

import os
from dataclasses import dataclass
from pathlib import Path
from typing import Any

try:
    import tomllib
except ModuleNotFoundError:  # Python < 3.11
    import tomli as tomllib


@dataclass(frozen=True, slots=True)
class AgentSettings:
    model: str | None
    reasoning_effort: str | None


@dataclass(frozen=True, slots=True)
class StageAgentsSettings:
    agents: dict[str, AgentSettings]
    strict_agents: dict[str, AgentSettings]
    source_path: Path | None


def _as_nonempty_str(value: Any) -> str | None:
    if not isinstance(value, str):
        return None
    text = value.strip()
    return text if text else None


def _load_toml(path: Path) -> dict[str, Any]:
    with path.open("rb") as f:
        raw = tomllib.load(f)
    return raw if isinstance(raw, dict) else {}


def _merge_agent_settings(base: AgentSettings, raw: dict[str, Any] | None) -> AgentSettings:
    if not isinstance(raw, dict):
        return base
    model = _as_nonempty_str(raw.get("model")) or base.model
    effort = _as_nonempty_str(raw.get("reasoning_effort")) or base.reasoning_effort
    return AgentSettings(model=model, reasoning_effort=effort)


def _env_override(prefix: str, agent_key: str, field: str, *, qualifier: str | None = None) -> str | None:
    if qualifier:
        var = f"{prefix}_{agent_key.upper()}_{qualifier}_{field}"
    else:
        var = f"{prefix}_{agent_key.upper()}_{field}"
    return _as_nonempty_str(os.getenv(var))


def resolve_stage_agents_settings(
    *,
    stage_prefix: str,
    agent_keys: list[str],
    config_path: Path | None,
    default_config_path: Path | None,
) -> StageAgentsSettings:
    """
    Resolve per-agent settings for a stage (statement/proof/final).

    Precedence:
    - hard-coded defaults (None/None)
    - config file (if provided and parseable)
    - environment variables (if set and non-empty)

    TOML format:
      [agent_a]
      model = "..."
      reasoning_effort = "high"
    """
    defaults: dict[str, AgentSettings] = {k: AgentSettings(model=None, reasoning_effort=None) for k in agent_keys}

    raw: dict[str, Any] = {}
    used_path: Path | None = None
    candidate = config_path or default_config_path
    if candidate is not None and candidate.exists():
        try:
            raw = _load_toml(candidate)
            used_path = candidate
        except Exception:
            raw = {}
            used_path = None

    agents: dict[str, AgentSettings] = {}
    strict_agents: dict[str, AgentSettings] = {}
    for key in agent_keys:
        base = defaults[key]
        toml_key = f"agent_{key.lower()}"
        merged = _merge_agent_settings(base, raw.get(toml_key))
        merged = AgentSettings(
            model=_env_override(stage_prefix, key, "MODEL") or merged.model,
            reasoning_effort=_env_override(stage_prefix, key, "REASONING_EFFORT") or merged.reasoning_effort,
        )
        agents[key] = merged
        strict_raw = None
        if isinstance(raw.get(toml_key), dict):
            strict_raw = raw[toml_key].get("strict_retry")
        strict_merged = _merge_agent_settings(AgentSettings(model=None, reasoning_effort=None), strict_raw)
        strict_merged = AgentSettings(
            model=_env_override(stage_prefix, key, "MODEL", qualifier="STRICT_RETRY") or strict_merged.model,
            reasoning_effort=(
                _env_override(stage_prefix, key, "REASONING_EFFORT", qualifier="STRICT_RETRY")
                or strict_merged.reasoning_effort
            ),
        )
        strict_agents[key] = strict_merged

    return StageAgentsSettings(agents=agents, strict_agents=strict_agents, source_path=used_path)
