"""CIRBench configuration loader with debug logs."""
from __future__ import annotations
from pathlib import Path
from typing import Any, Dict, List, Optional
import os, yaml, re
from pydantic import BaseModel, ValidationError, Field
from .utils.logging_utils import get_logger, debug_on

logger = get_logger()

class Toolchain(BaseModel):
    clang: Optional[str] = None
    opt: Optional[str] = None
    llc: Optional[str] = None
    llvm_size: Optional[str] = None
    llvm_mca: Optional[str] = None
    llvm_as: Optional[str] = None
    alive_tv: Optional[str] = None

class PromptConfig(BaseModel):
    default_template: str = "You are a compiler IR assistant.\n<IR>{{ RAW_IR }}</IR>\n{{ TASK }}"
    enforce_blocks: bool = False
    shots_dir: Optional[str] = None
    shots: List[str] = []
    track_templates: Dict[str, str] = {}

class Budget(BaseModel):
    time_s: int = 60
    tokens: int = 2048
    mem_mb: int = 1024

class TrackConfig(BaseModel):
    name: str
    enabled: bool = True
    mode: str = "direct"
    k: int = 1
    budget: Budget = Budget()

class ValidatorConfig(BaseModel):
    verifier: Dict[str, Any] = {"enabled": True}
    equiv_intra: Dict[str, Any] = {"enabled": False, "timeout_s": 10}
    spec: Dict[str, Any] = {"enabled": False}
    obs: Dict[str, Any] = {"enabled": False}

class Guardrails(BaseModel):
    roi_required: bool = False
    delta_limits: Dict[str, int] = {"max_inst_delta": 256, "max_blocks_affected": 32}
    struct_hash_lock_non_roi: bool = False
    isa_whitelist: List[str] = []

class LoggingConfig(BaseModel):
    level: str = "INFO"
    dir: str = "runs/logs"
    json_mode: bool = True
    console_rich: bool = True

class ProjectConfig(BaseModel):
    name: str = "CIRBench"
    version: str = "0.4.1"
    seed: int = 1337
    output_dir: str = "runs"

class ModelConfig(BaseModel):
    kind: str
    name: str
    params: Dict[str, Any] = {}

class CIRBenchConfig(BaseModel):
    project: ProjectConfig = ProjectConfig()
    tracks: List[TrackConfig]
    governance: Dict[str, Any] = {"hidden_set": False, "rotation_policy": "none"}
    validators: ValidatorConfig = ValidatorConfig()
    guardrails: Guardrails = Guardrails()
    models: List[ModelConfig] = []
    logging: LoggingConfig = LoggingConfig()
    toolchain: Toolchain = Toolchain()
    prompt_cfg: PromptConfig = PromptConfig()
    prompt_only: bool = False
    from_files: Optional[Path] = None
    from_kind: str = "resp"

def _expand_env_vars(text: str) -> str:
    def repl(m):
        name = m.group(1) or m.group(2)
        return os.getenv(name, "")
    return re.sub(r"\$\{([^}]+)\}|\$([A-Za-z_][A-Za-z0-9_]*)", repl, text)

def _env_bool(name: str) -> bool:
    v = (os.getenv(name) or "").strip().lower()
    return v in ("1", "true", "yes", "on")

def load_cfg(path: Path) -> CIRBenchConfig:
    text = path.read_text(encoding="utf-8")
    text = _expand_env_vars(text)
    data = yaml.safe_load(text)
    try:
        cfg = CIRBenchConfig.model_validate(data)
        po = _env_bool("CIRBENCH_PROMPT_ONLY")
        ff = os.getenv("CIRBENCH_FROM_FILES") or None
        fk = (os.getenv("CIRBENCH_FROM_KIND") or "").strip().lower() or None
        if po: cfg.prompt_only = True
        if ff: cfg.from_files = Path(ff)
        if fk in ("resp", "pred"): cfg.from_kind = fk
        if debug_on():
            kinds = [f"{m.name}({m.kind})" for m in cfg.models]
            logger.info(f"[cfg] loaded {path} models={kinds} out_dir={cfg.project.output_dir}")
        return cfg
    except ValidationError as e:
        raise SystemExit(f"Config validation failed for {path}\n{e}") from e

def summary(cfg: CIRBenchConfig) -> str:
    lines = []
    lines.append(f"Project: {cfg.project.name} v{cfg.project.version}  seed={cfg.project.seed}")
    lines.append(f"Output:  {cfg.project.output_dir}")
    lines.append(f"Tracks:  {', '.join([t.name+('/'+t.mode) for t in cfg.tracks if t.enabled])}")
    lines.append(f"Models:  {', '.join([m.name+'('+m.kind+')' for m in cfg.models]) or '—'}")
    lines.append(f"Logging: level={cfg.logging.level} dir={cfg.logging.dir} json={cfg.logging.json_mode}")
    lines.append(f"Run-opts: prompt_only={cfg.prompt_only} from_files={str(cfg.from_files) if cfg.from_files else '—'} kind={cfg.from_kind}")
    return "\n".join(lines)