from __future__ import annotations

from typing import Any, Dict, List, Tuple
import copy
import itertools

Path = Tuple[str, ...]
_LITERAL_KEY = "_literal"

def _is_literal_dict(d: dict) -> bool:
    return isinstance(d, dict) and set(d.keys()) == {_LITERAL_KEY}

def _unwrap_literals(x):
    if isinstance(x, dict):
        if _is_literal_dict(x):
            return _unwrap_literals(x[_LITERAL_KEY])
        return {k: _unwrap_literals(v) for k, v in x.items()}
    if isinstance(x, list):
        return [_unwrap_literals(v) for v in x]
    return x

def _find_list_fields(cfg: Any, prefix: Path = ()) -> List[Tuple[Path, list]]:
    out: List[Tuple[Path, list]] = []
    if isinstance(cfg, dict):
        if _is_literal_dict(cfg):
            return out
        for k, v in cfg.items():
            out += _find_list_fields(v, prefix + (k,))
    elif isinstance(cfg, list):
        out.append((prefix, cfg))
    return out

def _set_in(d: Dict[str, Any], path: Path, value: Any) -> None:
    for k in path[:-1]:
        d = d.setdefault(k, {})
    d[path[-1]] = value

def _expand_dict(d: Dict[str, Any]) -> List[Dict[str, Any]]:
    fields = _find_list_fields(d)
    if not fields:
        return [copy.deepcopy(_unwrap_literals(d))]
    paths = [p for p, _ in fields]
    grids = [vals for _, vals in fields]
    out: List[Dict[str, Any]] = []
    for combo in itertools.product(*grids):
        inst = copy.deepcopy(d)
        for p, v in zip(paths, combo):
            _set_in(inst, p, v)
        out.append(_unwrap_literals(inst))
    return out

def _expand_named_entry(entry: Dict[str, Any], *, config_key="config") -> List[Tuple[str, Dict[str, Any], Dict[str, Any]]]:
    names_val = entry.get("name")
    names = names_val if isinstance(names_val, list) else [names_val]

    cfg_base = entry.get(config_key, {}) or {}
    cfg_variants = _expand_dict(cfg_base)

    meta = {k: v for k, v in entry.items() if k not in ("name", config_key)}

    out: List[Tuple[str, Dict[str, Any], Dict[str, Any]]] = []
    for n, cfg in itertools.product(names, cfg_variants):
        out.append((n, meta, cfg))
    return out

def assemble_runs(cfg: Dict[str, Any]) -> List[Dict[str, Any]]:
    base = copy.deepcopy(cfg)

    pdes = [p for p in base.get("pdes", []) if p.get("active", True)]
    models = [m for m in base.get("models", []) if m.get("active", True)]
    opts = [o for o in base.get("optimizers", []) if o.get("active", True)]

    if not pdes:
        raise ValueError("config must contain a non-empty or active 'pdes' list")
    if not models:
        raise ValueError("config must contain a non-empty or active 'models' list")
    if not opts:
        raise ValueError("config must contain a non-empty or active 'optimizers' list")

    top_base = {k: v for k, v in base.items() if k not in ("pdes", "models", "optimizers")}
    top_variants = _expand_dict(top_base)

    pde_variants: List[Tuple[str, dict, dict]] = []
    for p_entry in pdes:
        pde_variants.extend(_expand_named_entry(p_entry, config_key="config"))

    model_variants: List[Tuple[str, dict, dict]] = []
    for m_entry in models:
        model_variants.extend(_expand_named_entry(m_entry, config_key="config"))

    opt_variants: List[Tuple[str, dict, dict]] = []
    for o_entry in opts:
        opt_variants.extend(_expand_named_entry(o_entry, config_key="config"))

    runs: List[Dict[str, Any]] = []
    for top_cfg in top_variants:
        for (p_name, p_meta, p_cfg) in pde_variants:
            for (m_name, m_meta, m_cfg) in model_variants:
                for (o_name, o_meta, o_cfg) in opt_variants:
                    run = copy.deepcopy(top_cfg)

                    run["pde"] = p_name
                    run["pde_config"] = p_cfg
                    run["activation"] = m_cfg.get("activation", run.get("activation", "tanh"))

                    run["exp_name"] = f"{m_name}-{p_name}-{run['activation']}"

                    run["model_config"] = m_cfg
                    if p_cfg.get("fourier_embeddings", None) is not None or p_cfg.get("fourier_embeddings") is not False:
                        run["model_config"]["fourier_embeddings"] = p_cfg.get("fourier_embeddings", None)
                    if p_cfg.get("batch_size", None) is not None:
                        run["training"] = run.get("training", {})
                        run["training"]["batch_size"] = p_cfg.get("batch_size", None)

                    run["input_dim"] = int(m_cfg.get("input_dim", run.get("input_dim", 2)))
                    run["init_batch_size"] = int(run.get("init", {}).get("batch_size", 1))

                    run["optim"] = {"optimizer": o_name, **o_cfg}
                    run["model_meta"] = {"name": m_name, **m_meta, "config": m_cfg}
                    run["_names"] = {"pde": p_name, "model": m_name, "optim": o_name}

                    runs.append(run)

    return runs
