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 FinalAgentsSettings:
    agent_a: AgentSettings
    agent_b: AgentSettings
    agent_c: AgentSettings
    agent_d: AgentSettings
    agent_a_strict_retry: 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 _env_override(var: str) -> str | None:
    return _as_nonempty_str(os.getenv(var))


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 resolve_final_agents_settings(config_path: Path | None) -> FinalAgentsSettings:
    """
    Resolve per-agent settings for the final (stage 3) pipeline.

    Precedence:
    - hard-coded defaults
    - config file (if provided and parseable)
    - environment variables (if set and non-empty)
    """
    default_a = AgentSettings(model="gpt-5.3-codex", reasoning_effort="medium")
    default_b = AgentSettings(model="gpt-5.3-codex", reasoning_effort="low")
    default_c = AgentSettings(model="gpt-5.3-codex", reasoning_effort="high")
    default_d = AgentSettings(model="gpt-5.3-codex", reasoning_effort="low")
    default_a_strict = AgentSettings(model=None, reasoning_effort="high")

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

    agent_a = _merge_agent_settings(default_a, raw.get("agent_a"))
    agent_b = _merge_agent_settings(default_b, raw.get("agent_b"))
    agent_c = _merge_agent_settings(default_c, raw.get("agent_c"))
    agent_d = _merge_agent_settings(default_d, raw.get("agent_d"))

    strict_raw = None
    if isinstance(raw.get("agent_a"), dict):
        strict_raw = raw["agent_a"].get("strict_retry")
    agent_a_strict = _merge_agent_settings(default_a_strict, strict_raw)

    # Optional env overrides (keep backwards compatibility with env-only setup).
    agent_a = AgentSettings(
        model=_env_override("FINAL_AGENT_A_MODEL") or agent_a.model,
        reasoning_effort=_env_override("FINAL_AGENT_A_REASONING_EFFORT") or agent_a.reasoning_effort,
    )
    agent_b = AgentSettings(
        model=_env_override("FINAL_AGENT_B_MODEL") or agent_b.model,
        reasoning_effort=_env_override("FINAL_AGENT_B_REASONING_EFFORT") or agent_b.reasoning_effort,
    )
    agent_c = AgentSettings(
        model=_env_override("FINAL_AGENT_C_MODEL") or agent_c.model,
        reasoning_effort=_env_override("FINAL_AGENT_C_REASONING_EFFORT") or agent_c.reasoning_effort,
    )
    agent_d = AgentSettings(
        model=_env_override("FINAL_AGENT_D_MODEL") or agent_d.model,
        reasoning_effort=_env_override("FINAL_AGENT_D_REASONING_EFFORT") or agent_d.reasoning_effort,
    )

    return FinalAgentsSettings(
        agent_a=agent_a,
        agent_b=agent_b,
        agent_c=agent_c,
        agent_d=agent_d,
        agent_a_strict_retry=agent_a_strict,
        source_path=used_path,
    )
