{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "f5b14808",
   "metadata": {},
   "source": [
    "# DFN"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2efb5921",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "ce8269a4",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "\n",
    "import cppimport\n",
    "\n",
    "import sys, time, math, io, contextlib\n",
    "from pathlib import Path\n",
    "from typing import Optional, List, Tuple, Dict, Any\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linear_sum_assignment\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "from IPython.display import display\n",
    "\n",
    "import gurobipy as gp\n",
    "from gurobipy import GRB\n",
    "\n",
    "_TOL = 1e-9"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "417583e0",
   "metadata": {},
   "source": [
    "## LEMON"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "62767112",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "repo = Path().resolve().parent\n",
    "if str(repo) not in sys.path:\n",
    "    sys.path.insert(0, str(repo))\n",
    "\n",
    "lemon_mcf = cppimport.imp(\"lemon_mcf\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "15e6bd1b",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "88381d28",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'status': 1, 'flow': array([1., 2., 0.]), 'potential': array([-2.,  0., -1.]), 'reduced_cost': array([0., 0., 4.]), 'at_capacity': array([False,  True, False]), 'total_cost': 4.0}\n",
      "{'value': 4.0, 'flow': array([2., 2., 2., 2.])}\n"
     ]
    }
   ],
   "source": [
    "n = 3\n",
    "src    = np.array([0, 0, 1], dtype=np.int64)\n",
    "dst    = np.array([1, 2, 2], dtype=np.int64)\n",
    "cost   = np.array([2.0, 1.0, 3.0], dtype=np.float64)\n",
    "cap    = np.array([5.0, 2.0, 4.0], dtype=np.float64)\n",
    "supply = np.array([3.0, -1.0, -2.0], dtype=np.float64)\n",
    "\n",
    "out_min_cost_flow = lemon_mcf.solve_mcf(n, src, dst, cost, cap, supply)\n",
    "print(out_min_cost_flow)\n",
    "\n",
    "n = 4\n",
    "src = np.array([0,0,1,2], dtype=np.int64)\n",
    "dst = np.array([1,2,3,3], dtype=np.int64)\n",
    "cap = np.array([3.0,2.0,2.0,4.0], dtype=np.float64)\n",
    "out_max_flow = lemon_mcf.max_flow(n, src, dst, cap, 0, 3)\n",
    "print(out_max_flow)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "75235271",
   "metadata": {},
   "source": [
    "## Dataset generators"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "67763cc2",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def make_mdvsp_dataset(\n",
    "    K: int,\n",
    "    filename: str,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    max_trips=None,\n",
    "    max_succ=None,\n",
    "):\n",
    "    \"\"\"Multiple-Depot Vehicle Scheduling (MDVSP) dataset.\n",
    "\n",
    "    Returns:\n",
    "      X: (K, m) integer-ish capacities (float32)\n",
    "      y: (K,) min-cost-flow objective values (float32)\n",
    "      gt: dict containing the fixed network pieces (for later evaluation)\n",
    "    \"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "\n",
    "    with open(filename) as f:\n",
    "        m, n, l = map(int, f.readline().split())\n",
    "        f.readline()  # blank line\n",
    "        trips = np.loadtxt(f, max_rows=n, dtype=np.int64)[:max_trips]\n",
    "        D = np.loadtxt(f, max_rows=l, dtype=np.int64)\n",
    "\n",
    "    p, s, q, e = trips.T\n",
    "    ntr = len(trips)\n",
    "\n",
    "    # node ids\n",
    "    SS = 0\n",
    "    depS = 1 + np.arange(m)\n",
    "    depT = 1 + m + np.arange(m)\n",
    "    trS  = 1 + 2*m + np.arange(ntr)\n",
    "    trT  = 1 + 2*m + ntr + np.arange(ntr)\n",
    "    TT = 1 + 2*m + 2*ntr\n",
    "    N = TT + 1\n",
    "\n",
    "    src, dst, cost, cap = [], [], [], []\n",
    "\n",
    "    # depot boundary arcs (capacity is what we learn/provide per sample)\n",
    "    for d in range(m):\n",
    "        src += [SS, int(depT[d])]\n",
    "        dst += [int(depS[d]), TT]\n",
    "        cost += [0.0, 0.0]\n",
    "        cap  += [0.0, 0.0]\n",
    "    idxSS = np.arange(0, 2*m, 2)  # arcs SS->depS\n",
    "    idxTT = np.arange(1, 2*m, 2)  # arcs depT->TT\n",
    "\n",
    "    # trip arcs (always cap=1)\n",
    "    src += trS.tolist()\n",
    "    dst += trT.tolist()\n",
    "    cost += [0.0] * ntr\n",
    "    cap  += [1.0] * ntr\n",
    "\n",
    "    # depot-to-trip and trip-to-depot arcs\n",
    "    for d in range(m):\n",
    "        # depS -> trS\n",
    "        src += [int(depS[d])] * ntr\n",
    "        dst += trS.tolist()\n",
    "        cost += (5000 + 10 * D[d, p]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "        # trT -> depT\n",
    "        src += trT.tolist()\n",
    "        dst += [int(depT[d])] * ntr\n",
    "        cost += (5000 + 10 * D[q, d]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "    # feasible trip successor arcs (trT -> next trS)\n",
    "    order = np.argsort(s)\n",
    "    p2, s2 = p[order], s[order]\n",
    "    max_succ_eff = ntr if max_succ is None else int(max_succ)\n",
    "\n",
    "    for i in range(ntr):\n",
    "        travel = D[q[i], p2]                          # time from trip i end depot -> next trip start depot\n",
    "        feas = np.flatnonzero(s2 >= e[i] + travel)[:max_succ_eff]\n",
    "        j = order[feas]\n",
    "        if j.size:\n",
    "            src += [int(trT[i])] * j.size\n",
    "            dst += trS[j].tolist()\n",
    "            cost += (8 * travel[feas] + 2 * (s[j] - e[i])).astype(float).tolist()\n",
    "            cap  += [1.0] * j.size\n",
    "\n",
    "    src = np.asarray(src, dtype=np.int64)\n",
    "    dst = np.asarray(dst, dtype=np.int64)\n",
    "    cost = np.asarray(cost, dtype=np.float64)\n",
    "    cap0 = np.asarray(cap, dtype=np.float64)\n",
    "\n",
    "    # sample capacities X and compute y via max-flow + min-cost-flow\n",
    "    X = rng.integers(x_min, np.asarray(x_max) + 1, size=(K, m)).astype(np.float64)\n",
    "    y = np.empty(K, dtype=np.float64)\n",
    "\n",
    "    for k in range(K):\n",
    "        cap_k = cap0.copy()\n",
    "        cap_k[idxSS] = X[k]\n",
    "        cap_k[idxTT] = X[k]\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap_k, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, dtype=np.float64)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "\n",
    "        y[k] = lemon_mcf.solve_mcf(N, src, dst, cost, cap_k, supply)[\"total_cost\"]\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = dict(\n",
    "        type=\"mdvsp\", N=int(N), SS=int(SS), TT=int(TT),\n",
    "        src=src, dst=dst, cost=cost, cap0=cap0, idxSS=idxSS, idxTT=idxTT\n",
    "    )\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n",
    "\n",
    "\n",
    "def generate_bipartite_subset_matching_dataset(\n",
    "    K: int, num_nodes: int, c_min: int, c_max: int, noise_std: float = 0.0, seed: int = 0\n",
    "):\n",
    "    \"\"\"Assignment-style dataset: choose a subset of left nodes, match to right nodes with min cost.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    C = rng.integers(c_min, c_max + 1, size=(num_nodes, num_nodes)).astype(np.float32)\n",
    "\n",
    "    X = np.zeros((K, num_nodes), dtype=np.float32)\n",
    "    y = np.zeros((K,), dtype=np.float32)\n",
    "\n",
    "    for k in range(K):\n",
    "        mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        while not mask.any():\n",
    "            mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        idx = np.flatnonzero(mask)\n",
    "        X[k, idx] = 1.0\n",
    "\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        y[k] = C[idx, :][r, c].sum()\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = {\"type\": \"assignment\", \"C\": C.astype(np.float32)}\n",
    "    return X, y, gt\n",
    "\n",
    "\n",
    "def generate_convex_quadratic_dataset(\n",
    "    K: int,\n",
    "    dim: int,\n",
    "    eigen_min: float,\n",
    "    eigen_max: float,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    x_star_zero: bool = False,\n",
    "):\n",
    "    \"\"\"y = (x - x*)^T Q (x - x*) + noise, with Q symmetric PSD.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    U, R = np.linalg.qr(rng.standard_normal((dim, dim)))\n",
    "    U *= np.sign(np.diag(R) + 1e-12)\n",
    "    Q = U @ np.diag(rng.uniform(eigen_min, eigen_max, dim)) @ U.T\n",
    "\n",
    "    x_star = np.zeros(dim, dtype=np.int64) if x_star_zero else rng.integers(x_min, x_max + 1, size=dim)\n",
    "\n",
    "    X = rng.integers(x_min, x_max + 1, size=(K, dim)).astype(np.float32)\n",
    "    d = X - x_star\n",
    "    y = np.einsum(\"bi,ij,bj->b\", d, Q, d) + noise_std * rng.normal(size=K)\n",
    "\n",
    "    gt = {\"type\": \"quadratic\", \"Q\": Q.astype(np.float32), \"x_star\": x_star.astype(np.int64)}\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1dfc62e4",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "89201b4a",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(50, 8) (50,) \n",
      " [[ 5.  5.  8. 10.  0.  1.  9. 10.]\n",
      " [ 2.  3.  9.  4.  3.  9.  2.  4.]\n",
      " [ 7.  6.  0.  0.  9.  8.  9.  5.]\n",
      " [ 8.  3.  4.  8.  1.  3.  1.  4.]\n",
      " [10.  1.  4.  4.  9.  2.  5.  2.]] \n",
      " [490251.12 366100.16 447590.56 326038.94 376371.84] \n",
      "\n",
      "(50, 10) (50,) \n",
      " [[0. 1. 1. 0. 0. 0. 0. 1. 0. 1.]\n",
      " [1. 0. 0. 0. 0. 0. 1. 1. 0. 0.]\n",
      " [0. 1. 0. 0. 1. 0. 1. 0. 1. 1.]\n",
      " [1. 1. 0. 0. 1. 1. 1. 1. 0. 1.]\n",
      " [0. 1. 0. 1. 1. 0. 0. 0. 1. 0.]] \n",
      " [ 9.  4. 13. 17. 11.] \n",
      "\n",
      "(50, 10) (50,) \n",
      " [[-47. -87.  -1.  69.  25. -87.  30. -31. -55. -14.]\n",
      " [ 75.  94. -72.  13.  53. -48. -46. -52. -58.  78.]\n",
      " [-57. -55. -75. -75.  56. -43.  61.  17.  72.  11.]\n",
      " [ 53.  62. -88.  12.  -9. -43. -10. -18.  -2.  64.]\n",
      " [ 65.  25.  42.  92.  28. -26. -84.  11. -54.  19.]] \n",
      " [1.5913905e+06 9.7035825e+05 7.8526031e+05 5.9535256e+05 1.0773938e+06] \n",
      "\n"
     ]
    }
   ],
   "source": [
    "# NOTE: MDVSP requires that `filename` exists on disk.\n",
    "X, y, _ = make_mdvsp_dataset(\n",
    "    K=50, filename=\"RN-8-3000-03.dat\", x_min=0, x_max=10, noise_std=1.0, seed=1, max_trips=200, max_succ=5\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")\n",
    "\n",
    "X, y, _ = generate_bipartite_subset_matching_dataset(\n",
    "    K=50, num_nodes=10, c_min=1, c_max=10, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")\n",
    "\n",
    "X, y, _ = generate_convex_quadratic_dataset(\n",
    "    K=50, dim=10, eigen_min=1.0, eigen_max=20.0, x_min=-100, x_max=100, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "70071f5f",
   "metadata": {},
   "source": [
    "## Models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "07278db8",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def _ste_round(x: torch.Tensor) -> torch.Tensor:\n",
    "    return x + (torch.round(x) - x).detach()\n",
    "\n",
    "class _MCFValue(torch.autograd.Function):\n",
    "    \n",
    "    @staticmethod\n",
    "    def forward(ctx, n_nodes, src, dst, cost, cap, supply):\n",
    "        n = int(n_nodes)\n",
    "\n",
    "        src = src.to(dtype=torch.int64).contiguous()\n",
    "        dst = dst.to(dtype=torch.int64).contiguous()\n",
    "\n",
    "        m = int(src.numel())\n",
    "        if (dst.numel() != m) or (cost.numel() != m) or (cap.numel() != m) or (supply.numel() != n):\n",
    "            raise ValueError(\"Bad shapes for MCF inputs.\")\n",
    "        if torch.abs(supply.double().sum()) > _TOL:\n",
    "            raise ValueError(\"Require sum(supply)=0\")\n",
    "\n",
    "        def as_np(t: torch.Tensor, dtype):\n",
    "            return t.detach().cpu().contiguous().view(-1).numpy().astype(dtype, copy=False)\n",
    "\n",
    "        out = lemon_mcf.solve_mcf(\n",
    "            n,\n",
    "            as_np(src, np.int64),\n",
    "            as_np(dst, np.int64),\n",
    "            as_np(cost, np.float64),\n",
    "            as_np(cap, np.float64),\n",
    "            as_np(supply, np.float64),\n",
    "            tol=_TOL,\n",
    "        )\n",
    "        if out[\"status\"] != 1:\n",
    "            raise RuntimeError(f\"LEMON failed (status={out['status']})\")\n",
    "\n",
    "        flow = out[\"flow\"]\n",
    "        pot = out[\"potential\"]\n",
    "        red = out[\"reduced_cost\"]\n",
    "        at = out.get(\"at_cap\", out.get(\"at_capacity\", None))\n",
    "        if at is None:\n",
    "            at = np.abs(flow - as_np(cap, np.float64)) <= _TOL\n",
    "\n",
    "        ctx.flow, ctx.pot, ctx.red, ctx.at = flow, pot, red, at\n",
    "        return cost.new_tensor(float(out[\"total_cost\"]))\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, g):\n",
    "        dev, dt = g.device, g.dtype\n",
    "        flow = torch.as_tensor(ctx.flow, device=dev, dtype=dt)\n",
    "        pot  = torch.as_tensor(ctx.pot,  device=dev, dtype=dt)\n",
    "        red  = torch.as_tensor(ctx.red,  device=dev, dtype=dt)\n",
    "        at   = torch.as_tensor(ctx.at,   device=dev, dtype=torch.bool)\n",
    "\n",
    "        grad_cost = flow\n",
    "        grad_cap  = torch.where(at, red, torch.zeros_like(red))\n",
    "        grad_sup  = pot.mean() - pot\n",
    "\n",
    "        return None, None, None, grad_cost * g, grad_cap * g, grad_sup * g\n",
    "\n",
    "\n",
    "class DFN(nn.Module):\n",
    "    def __init__(\n",
    "        self,\n",
    "        input_dim: int,\n",
    "        layer_sizes,\n",
    "        p_list,\n",
    "        big_cost: float = 1e6,\n",
    "        big_cap: float = 1e6,\n",
    "        seed: int = 0,\n",
    "        A_fixed=None,\n",
    "        alpha: float = 1e-6,\n",
    "        beta: float = -0.0,\n",
    "    ):\n",
    "        super().__init__()\n",
    "        self.alpha = float(alpha)\n",
    "        self.beta  = float(beta)\n",
    "\n",
    "        layer_sizes = list(map(int, layer_sizes))\n",
    "        if len(layer_sizes) < 2 or len(p_list) != len(layer_sizes) - 1:\n",
    "            raise ValueError(\"Need len(layer_sizes)>=2 and len(p_list)=len(layer_sizes)-1\")\n",
    "\n",
    "        self.n = int(sum(layer_sizes))\n",
    "        if self.n <= 0:\n",
    "            raise ValueError(\"sum(layer_sizes) must be > 0\")\n",
    "\n",
    "        # node indices per layer\n",
    "        layers, off = [], 0\n",
    "        for s in layer_sizes:\n",
    "            layers.append(torch.arange(off, off + s, dtype=torch.long))\n",
    "            off += s\n",
    "\n",
    "        L1, LK = layers[0], layers[-1]\n",
    "        if L1.numel() == 0 or LK.numel() == 0:\n",
    "            raise ValueError(\"First/last layer must be non-empty.\")\n",
    "\n",
    "        self.fix_node = int(LK[-1].item())\n",
    "        boundary = torch.cat([L1, LK[:-1]], 0)\n",
    "        self.register_buffer(\"boundary\", boundary)\n",
    "\n",
    "        gen = torch.Generator().manual_seed(int(seed))\n",
    "\n",
    "        def bipartite(U: torch.Tensor, V: torch.Tensor):\n",
    "            su, sv = int(U.numel()), int(V.numel())\n",
    "            return U.repeat_interleave(sv), V.repeat(su)\n",
    "\n",
    "        def sample_edges(U, V, p: float):\n",
    "            s, t = bipartite(U, V)\n",
    "            if p < 1.0:\n",
    "                keep = torch.rand(s.numel(), generator=gen) < float(p)\n",
    "                s, t = s[keep], t[keep]\n",
    "            return s, t\n",
    "\n",
    "        # learnable arcs between consecutive layers (both directions)\n",
    "        sf, tf, sb, tb = [], [], [], []\n",
    "        for i, p in enumerate(map(float, p_list)):\n",
    "            if not (0.0 <= p <= 1.0):\n",
    "                raise ValueError(\"p_list entries must be in [0,1]\")\n",
    "\n",
    "            s, t = sample_edges(layers[i], layers[i + 1], p)  # forward\n",
    "            sf.append(s); tf.append(t)\n",
    "\n",
    "            s, t = sample_edges(layers[i + 1], layers[i], p)  # backward\n",
    "            sb.append(s); tb.append(t)\n",
    "\n",
    "        src_param = torch.cat([torch.cat(sf, 0), torch.cat(sb, 0)], 0)\n",
    "        dst_param = torch.cat([torch.cat(tf, 0), torch.cat(tb, 0)], 0)\n",
    "        if src_param.numel() == 0:\n",
    "            raise ValueError(\"No learnable arcs (increase p_list / layer sizes).\")\n",
    "\n",
    "        s1, t1 = bipartite(L1, LK)\n",
    "        s2, t2 = bipartite(LK, L1)\n",
    "        src_fixed = torch.cat([s1, s2], 0)\n",
    "        dst_fixed = torch.cat([t1, t2], 0)\n",
    "        m_fixed = int(src_fixed.numel())\n",
    "\n",
    "        self.register_buffer(\"src\", torch.cat([src_param, src_fixed], 0))\n",
    "        self.register_buffer(\"dst\", torch.cat([dst_param, dst_fixed], 0))\n",
    "        self.register_buffer(\"cap_fixed\",  torch.full((m_fixed,), float(big_cap),  dtype=torch.float32))\n",
    "        self.register_buffer(\"cost_fixed\", torch.full((m_fixed,), float(big_cost), dtype=torch.float32))\n",
    "\n",
    "        nb = int(boundary.numel())\n",
    "        input_dim = int(input_dim)\n",
    "\n",
    "        m_param = int(src_param.numel())\n",
    "        self.cap_raw  = nn.Parameter(torch.zeros(m_param) + 0.542)\n",
    "        self.cost_raw = nn.Parameter(torch.randn(m_param) + 1.0)\n",
    "        self.b_raw    = nn.Parameter(torch.zeros(nb))\n",
    "\n",
    "        if A_fixed is None:\n",
    "            A = torch.zeros(nb, input_dim)\n",
    "            rows = torch.arange(nb)\n",
    "            A[rows, rows % input_dim] = 1.0\n",
    "            self.A = nn.Parameter(A)\n",
    "        else:\n",
    "            A_fixed = torch.as_tensor(A_fixed, dtype=torch.float32)\n",
    "            if A_fixed.shape != (nb, input_dim):\n",
    "                raise ValueError(f\"A_fixed must have shape {(nb, input_dim)}, got {tuple(A_fixed.shape)}\")\n",
    "            self.register_buffer(\"A\", A_fixed)\n",
    "\n",
    "    def forward(self, w: torch.Tensor) -> torch.Tensor:\n",
    "        capP  = _ste_round(F.softplus(self.cap_raw))\n",
    "        costP = self.cost_raw\n",
    "        b     = _ste_round(self.b_raw)\n",
    "        A     = _ste_round(self.A) if isinstance(self.A, nn.Parameter) else self.A\n",
    "\n",
    "        cap  = torch.cat([capP,  self.cap_fixed.to(w.device, w.dtype)], 0)\n",
    "        cost = torch.cat([costP, self.cost_fixed.to(w.device, w.dtype)], 0)\n",
    "\n",
    "        def one(w1: torch.Tensor) -> torch.Tensor:\n",
    "            supply = torch.zeros(self.n, device=w1.device, dtype=torch.float64)\n",
    "            supply[self.boundary] = (A.double() @ w1.double()) + b.double()\n",
    "            supply[self.fix_node] = -supply.sum()\n",
    "            return _MCFValue.apply(self.n, self.src, self.dst, cost, cap, supply)\n",
    "\n",
    "        out = one(w) if w.dim() == 1 else torch.stack([one(wi) for wi in w], 0)\n",
    "        return self.alpha * out + self.beta\n",
    "\n",
    "\n",
    "class MLP(nn.Module):\n",
    "    def __init__(self, in_dim, hidden_dims, out_dim):\n",
    "        super().__init__()\n",
    "        dims = [in_dim] + list(hidden_dims) + [out_dim]\n",
    "        layers = []\n",
    "        for a, b in zip(dims[:-2], dims[1:-1]):\n",
    "            layers += [nn.Linear(a, b), nn.ReLU()]\n",
    "        layers += [nn.Linear(dims[-2], dims[-1])]\n",
    "        self.net = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "\n",
    "class MaxAffine(nn.Module):\n",
    "    def __init__(self, in_dim: int, n_pieces: int):\n",
    "        super().__init__()\n",
    "        self.W = nn.Parameter(torch.randn(n_pieces, in_dim) / (in_dim**0.5))\n",
    "        self.b = nn.Parameter(torch.zeros(n_pieces))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        return (x @ self.W.T + self.b).max(dim=1).values\n",
    "\n",
    "\n",
    "class LSET(nn.Module):\n",
    "    def __init__(self, in_dim: int, n_pieces: int, T: float = 0.01):\n",
    "        super().__init__()\n",
    "        self.T = float(T)\n",
    "        if self.T == 0.0:\n",
    "            raise ValueError(\"T must be nonzero\")\n",
    "        self.A = nn.Parameter(torch.randn(n_pieces, in_dim) / (in_dim**0.5))\n",
    "        self.b = nn.Parameter(torch.zeros(n_pieces))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        z = (x @ self.A.t() + self.b) / self.T\n",
    "        return self.T * torch.logsumexp(z, dim=-1)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7cca0120",
   "metadata": {},
   "source": [
    "## Training Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0a7db872",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def _split_train_val_test(X: torch.Tensor, y: torch.Tensor, *, val_frac: float, test_frac: float, seed: int):\n",
    "    N = int(X.shape[0])\n",
    "    n_test = int(round(test_frac * N))\n",
    "    n_val  = int(round(val_frac  * N))\n",
    "    n_train = N - n_val - n_test\n",
    "    if n_train <= 0:\n",
    "        raise ValueError(\"splits too large; train set would be empty\")\n",
    "\n",
    "    g = torch.Generator().manual_seed(int(seed))\n",
    "    perm = torch.randperm(N, generator=g)\n",
    "    i_tr = perm[:n_train]\n",
    "    i_va = perm[n_train:n_train + n_val]\n",
    "    i_te = perm[n_train + n_val:]\n",
    "\n",
    "    return (X[i_tr], y[i_tr]), (X[i_va], y[i_va]), (X[i_te], y[i_te])\n",
    "\n",
    "\n",
    "def _fit_standardizer(Xtr: torch.Tensor, ytr: torch.Tensor, *, eps: float):\n",
    "    x_mean = Xtr.mean(0, keepdim=True)\n",
    "    x_std  = Xtr.std(0, unbiased=False, keepdim=True)\n",
    "    x_std  = torch.where(x_std < eps, torch.ones_like(x_std), x_std)\n",
    "\n",
    "    y_mean = ytr.mean()\n",
    "    y_std  = ytr.std(unbiased=False).clamp_min(eps)\n",
    "\n",
    "    return {\"x_mean\": x_mean, \"x_std\": x_std, \"y_mean\": y_mean, \"y_std\": y_std}\n",
    "\n",
    "\n",
    "def _apply_standardizer(X: torch.Tensor, y: torch.Tensor, scaler):\n",
    "    Xn = (X - scaler[\"x_mean\"]) / scaler[\"x_std\"]\n",
    "    yn = (y - scaler[\"y_mean\"]) / scaler[\"y_std\"]\n",
    "    return Xn, yn\n",
    "\n",
    "@torch.no_grad()\n",
    "def _mse_norm(model: nn.Module, loader: DataLoader, device: str):\n",
    "    model.eval()\n",
    "    tot, n = 0.0, 0\n",
    "    for xb, yb in loader:\n",
    "        xb, yb = xb.to(device), yb.to(device)\n",
    "        pred = model(xb).squeeze(-1)\n",
    "        tot += F.mse_loss(pred, yb, reduction=\"sum\").item()\n",
    "        n += yb.numel()\n",
    "    return tot / max(n, 1)\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def _predict_in_chunks(model: nn.Module, X: torch.Tensor, *, chunk: int):\n",
    "    out = []\n",
    "    for i in range(0, int(X.shape[0]), int(chunk)):\n",
    "        out.append(model(X[i:i + chunk]).squeeze(-1))\n",
    "    return torch.cat(out, 0)\n",
    "\n",
    "\n",
    "def generate_and_train_simple(dataset_type, dataset_params, model_type, model_params, train_params=None):\n",
    "    \n",
    "    tp = train_params or {}\n",
    "\n",
    "    epochs     = int(tp.get(\"epochs\", 200))\n",
    "    batch_sz   = int(tp.get(\"batch_size\", 32))\n",
    "    lr         = float(tp.get(\"lr\", 1e-3))\n",
    "    wd         = float(tp.get(\"weight_decay\", 0.0))\n",
    "    val_frac   = float(tp.get(\"val_frac\", 0.15))\n",
    "    test_frac  = float(tp.get(\"test_frac\", 0.15))\n",
    "    eps        = float(tp.get(\"eps\", 1e-8))\n",
    "    seed       = int(tp.get(\"seed\", 0))\n",
    "    device     = tp.get(\"device\", \"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "    plot_every = int(tp.get(\"plot_every\", 0) or 0)\n",
    "    plot_points= int(tp.get(\"plot_points\", 2048))\n",
    "    plot_chunk = int(tp.get(\"plot_chunk\", 4096))\n",
    "    print_stats= bool(tp.get(\"print_stats\", True))\n",
    "\n",
    "    # ---- data ----\n",
    "    dt = str(dataset_type).lower()\n",
    "    if dt == \"mdvsp\":\n",
    "        X, y, gt = make_mdvsp_dataset(**dataset_params)\n",
    "    elif dt == \"assignment\":\n",
    "        X, y, gt = generate_bipartite_subset_matching_dataset(**dataset_params)\n",
    "    elif dt == \"quadratic\":\n",
    "        X, y, gt = generate_convex_quadratic_dataset(**dataset_params)\n",
    "    else:\n",
    "        raise ValueError(\"dataset_type must be: mdvsp | assignment | quadratic\")\n",
    "\n",
    "    X = torch.as_tensor(X, dtype=torch.float32)\n",
    "    y = torch.as_tensor(y, dtype=torch.float32).view(-1)\n",
    "\n",
    "    # ---- print a quick dataset stats ----\n",
    "    if print_stats:\n",
    "        with torch.no_grad():\n",
    "            xmn = X.mean(0)\n",
    "            xsd = X.std(0, unbiased=False)\n",
    "            print(\n",
    "                f\"\\n--- Dataset stats ({dataset_type}) ---\\n\"\n",
    "                f\"  X: shape={tuple(X.shape)}  mean(mean)={xmn.mean():.3g}  std(mean)={xsd.mean():.3g}  \"\n",
    "                f\"min={float(X.min()):.3g}  max={float(X.max()):.3g}\\n\"\n",
    "                f\"  y: shape={tuple(y.shape)}  mean={float(y.mean()):.3g}  std={float(y.std(unbiased=False)):.3g}  \"\n",
    "                f\"min={float(y.min()):.3g}  max={float(y.max()):.3g}\\n\"\n",
    "            )\n",
    "\n",
    "    (Xtr, ytr), (Xva, yva), (Xte, yte) = _split_train_val_test(X, y, val_frac=val_frac, test_frac=test_frac, seed=seed)\n",
    "\n",
    "    scaler = _fit_standardizer(Xtr, ytr, eps=eps)\n",
    "    XtrN, ytrN = _apply_standardizer(Xtr, ytr, scaler)\n",
    "    XvaN, yvaN = _apply_standardizer(Xva, yva, scaler)\n",
    "    XteN, yteN = _apply_standardizer(Xte, yte, scaler)\n",
    "\n",
    "    train_loader = DataLoader(TensorDataset(XtrN, ytrN), batch_size=batch_sz, shuffle=True)\n",
    "    val_loader   = DataLoader(TensorDataset(XvaN, yvaN), batch_size=batch_sz, shuffle=False)\n",
    "\n",
    "    # subset for plotting\n",
    "    Nv = int(XvaN.shape[0])\n",
    "    if plot_points <= 0 or plot_points >= Nv:\n",
    "        plot_idx = torch.arange(Nv)\n",
    "    else:\n",
    "        g_plot = torch.Generator().manual_seed(seed + 12345)\n",
    "        plot_idx = torch.randperm(Nv, generator=g_plot)[:plot_points]\n",
    "\n",
    "    # ---- model ----\n",
    "    mt = str(model_type)\n",
    "    mp = dict(model_params)\n",
    "\n",
    "    if mt == \"DFN\":\n",
    "        model = DFN(**mp)\n",
    "        extra = f\"layers={mp.get('layer_sizes')} p_list={mp.get('p_list')} alpha={getattr(model,'alpha',None)} beta={getattr(model,'beta',None)}\"\n",
    "    else:\n",
    "        mp.pop(\"alpha\", None)\n",
    "        mp.pop(\"beta\", None)\n",
    "        if mt == \"MLP\":\n",
    "            model = MLP(**mp)\n",
    "            extra = f\"hidden={mp.get('hidden_dims')}\"\n",
    "        elif mt == \"MaxAffine\":\n",
    "            model = MaxAffine(**mp)\n",
    "            extra = f\"n_pieces={mp.get('n_pieces')}\"\n",
    "        elif mt == \"LSET\":\n",
    "            model = LSET(**mp)\n",
    "            extra = f\"n_pieces={mp.get('n_pieces')} T={mp.get('T')}\"\n",
    "        else:\n",
    "            raise ValueError(\"model_type must be: DFN | MLP | MaxAffine | LSET\")\n",
    "\n",
    "    model = model.to(device)\n",
    "    opt = torch.optim.Adam(model.parameters(), lr=lr, weight_decay=wd)\n",
    "\n",
    "    n_params = sum(p.numel() for p in model.parameters())\n",
    "    n_trainable = sum(p.numel() for p in model.parameters() if p.requires_grad)\n",
    "    print(\n",
    "        f\"\\n=== Run: {dataset_type} | {model_type} ===\\n\"\n",
    "        f\"  data: N={len(X)}  train/val/test={len(Xtr)}/{len(Xva)}/{len(Xte)}  dim={X.shape[1]}\\n\"\n",
    "        f\"  model: params={n_params:,} {extra}\\n\"\n",
    "        f\"  train: device={device}  epochs={epochs}  batch={batch_sz}  lr={lr:g}  wd={wd:g}  seed={seed}\\n\"\n",
    "    )\n",
    "\n",
    "    history = {\"train_mse_norm\": [], \"val_mse_norm\": []}\n",
    "    best_val, best_ep, best_state = float(\"inf\"), 0, None\n",
    "\n",
    "    live = display(None, display_id=True) if plot_every > 0 else None\n",
    "\n",
    "    for ep in range(1, epochs + 1):\n",
    "        model.train()\n",
    "        for xb, yb in train_loader:\n",
    "            xb, yb = xb.to(device), yb.to(device)\n",
    "            loss = F.mse_loss(model(xb).squeeze(-1), yb)\n",
    "            opt.zero_grad(set_to_none=True)\n",
    "            loss.backward()\n",
    "            opt.step()\n",
    "\n",
    "        tr_mse = _mse_norm(model, train_loader, device)\n",
    "        va_mse = _mse_norm(model, val_loader, device)\n",
    "        history[\"train_mse_norm\"].append(tr_mse)\n",
    "        history[\"val_mse_norm\"].append(va_mse)\n",
    "\n",
    "        if va_mse < best_val:\n",
    "            best_val, best_ep = va_mse, ep\n",
    "            best_state = {k: v.detach().cpu().clone() for k, v in model.state_dict().items()}\n",
    "\n",
    "        if plot_every > 0 and (ep == 1 or ep % plot_every == 0 or ep == epochs):\n",
    "            model.eval()\n",
    "            x_plot = XvaN[plot_idx].to(device)\n",
    "            y_true = yvaN[plot_idx].to(device)\n",
    "            y_pred = _predict_in_chunks(model, x_plot, chunk=plot_chunk)\n",
    "\n",
    "            fig, ax = plt.subplots(1, 2, figsize=(10, 4))\n",
    "            ax[0].plot(history[\"train_mse_norm\"], label=\"train\")\n",
    "            ax[0].plot(history[\"val_mse_norm\"], label=\"val\")\n",
    "            ax[0].set_yscale(\"log\")\n",
    "            ax[0].set_title(f\"Epoch {ep}/{epochs} | val MSE={va_mse:.3e} (norm)\")\n",
    "            ax[0].legend()\n",
    "\n",
    "            yt = y_true.detach().cpu().numpy()\n",
    "            yp = y_pred.detach().cpu().numpy()\n",
    "            ax[1].scatter(yt, yp, s=10)\n",
    "            lo = float(min(yt.min(), yp.min()))\n",
    "            hi = float(max(yt.max(), yp.max()))\n",
    "            ax[1].plot([lo, hi], [lo, hi])\n",
    "            ax[1].set_xlabel(\"y_true (norm)\")\n",
    "            ax[1].set_ylabel(\"y_pred (norm)\")\n",
    "            ax[1].set_title(f\"Val scatter (n={len(yt)})\")\n",
    "\n",
    "            plt.tight_layout()\n",
    "            live.update(fig)\n",
    "            plt.close(fig)\n",
    "\n",
    "    if best_state is not None:\n",
    "        model.load_state_dict(best_state)\n",
    "\n",
    "    if print_stats:\n",
    "        print(f\"[DONE] best val MSE (norm) = {best_val:.3e} @ epoch {best_ep}\\n\")\n",
    "\n",
    "    data = {\n",
    "        \"raw\":  {\"Xtr\": Xtr,  \"ytr\": ytr,  \"Xva\": Xva,  \"yva\": yva,  \"Xte\": Xte,  \"yte\": yte},\n",
    "        \"norm\": {\"Xtr\": XtrN, \"ytr\": ytrN, \"Xva\": XvaN, \"yva\": yvaN, \"Xte\": XteN, \"yte\": yteN},\n",
    "        \"scaler\": scaler,\n",
    "        \"best_val_mse_norm\": float(best_val),\n",
    "        \"best_epoch\": int(best_ep),\n",
    "        \"stats\": {\"n_params\": int(n_params), \"n_trainable\": int(n_trainable)},\n",
    "        \"true\": gt,\n",
    "        \"device\": device,\n",
    "    }\n",
    "\n",
    "    spec = {\n",
    "        \"model\": model_type,\n",
    "        \"extra\": extra,\n",
    "        \"n_params\": int(n_params),\n",
    "        \"n_trainable\": int(n_trainable),\n",
    "        \"model_params\": dict(model_params),\n",
    "        \"train_params\": {\n",
    "            \"epochs\": int(epochs),\n",
    "            \"batch_size\": int(batch_sz),\n",
    "            \"lr\": float(lr),\n",
    "            \"weight_decay\": float(wd),\n",
    "            \"seed\": int(seed),\n",
    "            \"device\": str(device),\n",
    "            \"val_frac\": float(val_frac),\n",
    "            \"test_frac\": float(test_frac),\n",
    "        },\n",
    "    }\n",
    "\n",
    "    return model, data, history, spec\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ace7f63f",
   "metadata": {},
   "source": [
    "## Optimization Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "38416530",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def eval_true_obj(gt, x):\n",
    "    if not isinstance(gt, dict):\n",
    "        return np.nan\n",
    "\n",
    "    x = np.asarray(x).reshape(-1)\n",
    "    t = gt.get(\"type\", None)\n",
    "\n",
    "    if t == \"quadratic\":\n",
    "        Q = np.asarray(gt[\"Q\"], float)\n",
    "        xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "        d = x.astype(float) - xs\n",
    "        return float(d @ Q @ d)\n",
    "\n",
    "    if t == \"assignment\":\n",
    "        C = np.asarray(gt[\"C\"], float)\n",
    "        idx = np.flatnonzero(x > 0.5)\n",
    "        if idx.size == 0:\n",
    "            return 0.0\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        return float(C[idx, :][r, c].sum())\n",
    "\n",
    "    if t == \"mdvsp\":\n",
    "        N, SS, TT = int(gt[\"N\"]), int(gt[\"SS\"]), int(gt[\"TT\"])\n",
    "        src = np.asarray(gt[\"src\"], np.int64)\n",
    "        dst = np.asarray(gt[\"dst\"], np.int64)\n",
    "        cost = np.asarray(gt[\"cost\"], float)\n",
    "        cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "        idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "        idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "        cap = cap0.copy()\n",
    "        cap[idxSS] = x.astype(float)\n",
    "        cap[idxTT] = x.astype(float)\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, float)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "        return float(lemon_mcf.solve_mcf(N, src, dst, cost, cap, supply)[\"total_cost\"])\n",
    "\n",
    "    return np.nan\n",
    "\n",
    "\n",
    "def local_search_l1_int(\n",
    "    f,\n",
    "    x0,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    delta: int,\n",
    "    sum_eq=None,\n",
    "    max_iters: int = 10_000,\n",
    "    print_every: int = 1,\n",
    "):\n",
    "    x  = np.asarray(x0,    int).ravel()\n",
    "    lo = np.asarray(x_min, int).ravel()\n",
    "    hi = np.asarray(x_max, int).ravel()\n",
    "    n = int(x.size)\n",
    "    delta = int(delta)\n",
    "\n",
    "    assert lo.size == n and hi.size == n\n",
    "    assert np.all(x >= lo) and np.all(x <= hi)\n",
    "    if sum_eq is not None:\n",
    "        sum_eq = int(sum_eq)\n",
    "        assert int(x.sum()) == sum_eq\n",
    "\n",
    "    def eval_batch(X):\n",
    "        y = np.asarray(f(X), float).reshape(-1)\n",
    "        return y\n",
    "\n",
    "    def ok(z):\n",
    "        if np.any(z < lo) or np.any(z > hi):\n",
    "            return False\n",
    "        if sum_eq is not None and int(z.sum()) != sum_eq:\n",
    "            return False\n",
    "        return True\n",
    "\n",
    "    # integer deltas with exact L1 = k\n",
    "    def deltas_exact(k):\n",
    "        d = np.zeros(n, int)\n",
    "\n",
    "        def rec(i, rem):\n",
    "            if i == n:\n",
    "                if rem == 0:\n",
    "                    yield d.copy()\n",
    "                return\n",
    "            for t in range(rem + 1):\n",
    "                if t == 0:\n",
    "                    d[i] = 0\n",
    "                    yield from rec(i + 1, rem)\n",
    "                else:\n",
    "                    d[i] = +t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "                    d[i] = -t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "            d[i] = 0\n",
    "\n",
    "        yield from rec(0, k)\n",
    "\n",
    "    t0 = time.perf_counter()\n",
    "    y = float(eval_batch(x[None, :])[0])\n",
    "    hist = [{\"iter\": 0, \"t\": 0.0, \"best_y\": y, \"x\": x.copy()}]\n",
    "    print(f\"iter=0  t=0.00s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    for it in range(1, int(max_iters) + 1):\n",
    "        cand = []\n",
    "        for k in range(1, delta + 1):\n",
    "            for dlt in deltas_exact(k):\n",
    "                z = x + dlt\n",
    "                if ok(z):\n",
    "                    cand.append(z)\n",
    "\n",
    "        if not cand:\n",
    "            print(f\"STOP: no feasible neighbors. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        Y = eval_batch(np.stack(cand, 0))\n",
    "        j = int(np.argmin(Y))\n",
    "        if float(Y[j]) >= y:\n",
    "            print(f\"STOP: local minimum. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        x, y = cand[j], float(Y[j])\n",
    "        hist.append({\"iter\": it, \"t\": time.perf_counter() - t0, \"best_y\": y, \"x\": x.copy()})\n",
    "        if it % max(1, int(print_every)) == 0:\n",
    "            print(f\"iter={it}  t={hist[-1]['t']:.2f}s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    return x, y, hist\n",
    "\n",
    "\n",
    "def _scaler_np(scaler):\n",
    "    xm = scaler[\"x_mean\"].detach().cpu().numpy().reshape(-1)\n",
    "    xs = scaler[\"x_std\"].detach().cpu().numpy().reshape(-1)\n",
    "    ym = float(scaler[\"y_mean\"].detach().cpu())\n",
    "    ys = float(scaler[\"y_std\"].detach().cpu())\n",
    "    return xm, xs, ym, ys\n",
    "\n",
    "\n",
    "def solve_dfn_ip_gurobi(dfn, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    x_mean, x_std, y_mean, y_std = _scaler_np(scaler)\n",
    "\n",
    "    cost = np.r_[dfn.cost_raw.detach().cpu().numpy(), dfn.cost_fixed.detach().cpu().numpy()]\n",
    "    cap  = np.r_[torch.round(F.softplus(dfn.cap_raw.detach())).cpu().numpy(), dfn.cap_fixed.detach().cpu().numpy()]\n",
    "\n",
    "    A = dfn.A.detach().cpu().numpy()\n",
    "    if isinstance(dfn.A, torch.nn.Parameter):\n",
    "        A = np.round(A)\n",
    "    b = np.round(dfn.b_raw.detach().cpu().numpy())\n",
    "\n",
    "    src = dfn.src.detach().cpu().numpy().astype(int)\n",
    "    dst = dfn.dst.detach().cpu().numpy().astype(int)\n",
    "    boundary = dfn.boundary.detach().cpu().numpy().astype(int)\n",
    "    fix = int(dfn.fix_node)\n",
    "    n = int(dfn.n)\n",
    "    m = int(src.size)\n",
    "    alpha = float(dfn.alpha)\n",
    "    beta  = float(dfn.beta)\n",
    "\n",
    "    out = [[] for _ in range(n)]\n",
    "    inn = [[] for _ in range(n)]\n",
    "    for e in range(m):\n",
    "        out[src[e]].append(e)\n",
    "        inn[dst[e]].append(e)\n",
    "\n",
    "    M = gp.Model(\"DFN_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    f = M.addVars(m, lb=0.0, ub=cap.tolist(), vtype=GRB.CONTINUOUS, name=\"f\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq)\n",
    "\n",
    "    xm_over_xs = x_mean / x_std\n",
    "    s = [0] * n\n",
    "    s_boundary = []\n",
    "    for r, v in enumerate(boundary):\n",
    "        const = float(b[r] - (A[r] * xm_over_xs).sum())\n",
    "        expr = const + gp.quicksum((A[r, j] / x_std[j]) * x[j] for j in range(d) if A[r, j] != 0)\n",
    "        s[v] = expr\n",
    "        s_boundary.append(expr)\n",
    "    s[fix] = -gp.quicksum(s_boundary)\n",
    "\n",
    "    for v in range(n):\n",
    "        M.addConstr(gp.quicksum(f[e] for e in out[v]) - gp.quicksum(f[e] for e in inn[v]) == s[v])\n",
    "\n",
    "    flow_cost = gp.quicksum(cost[e] * f[e] for e in range(m))\n",
    "    M.setObjective((alpha * flow_cost + beta) * y_std + y_mean, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No solution (Gurobi status {M.Status})\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None)}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_mlp_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    if not hasattr(base, \"net\"):\n",
    "        raise ValueError(\"Expected an MLP with attribute .net (nn.Sequential).\")\n",
    "    linears = [L for L in base.net if isinstance(L, torch.nn.Linear)]\n",
    "    if not linears:\n",
    "        raise ValueError(\"No Linear layers found in base.net\")\n",
    "\n",
    "    W = [L.weight.detach().cpu().numpy().astype(float) for L in linears]\n",
    "    b = [L.bias.detach().cpu().numpy().astype(float) for L in linears]\n",
    "\n",
    "    W[0] = W[0] / xs[None, :]\n",
    "    b[0] = b[0] - W[0] @ xm\n",
    "\n",
    "    W[-1] *= a_out\n",
    "    b[-1] = a_out * b[-1] + b_out\n",
    "\n",
    "    u = np.maximum(np.abs(x_min), np.abs(x_max))\n",
    "    preLU = []\n",
    "    for k in range(len(W) - 1):\n",
    "        U = np.abs(W[k]) @ u + np.abs(b[k])\n",
    "        preLU.append((-U, U))\n",
    "        u = np.maximum(0.0, U)\n",
    "\n",
    "    M = gp.Model(\"MLP_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    prev = [x[i] for i in range(d)]\n",
    "\n",
    "    for k in range(len(W) - 1):\n",
    "        Lk, Uk = preLU[k]\n",
    "        h = W[k].shape[0]\n",
    "\n",
    "        a = [M.addVar(lb=float(Lk[j]), ub=float(Uk[j]), name=f\"a{k}_{j}\") for j in range(h)]\n",
    "        z = [M.addVar(lb=0.0, ub=float(max(0.0, Uk[j])), name=f\"z{k}_{j}\") for j in range(h)]\n",
    "\n",
    "        for j in range(h):\n",
    "            M.addConstr(a[j] == b[k][j] + gp.quicksum(W[k][j, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "            Lj, Uj = float(Lk[j]), float(Uk[j])\n",
    "            if Uj <= 0.0:\n",
    "                M.addConstr(z[j] == 0.0)\n",
    "            elif Lj >= 0.0:\n",
    "                M.addConstr(z[j] == a[j])\n",
    "            else:\n",
    "                s = M.addVar(vtype=GRB.BINARY, name=f\"s{k}_{j}\")\n",
    "                M.addConstr(z[j] >= a[j])\n",
    "                M.addConstr(z[j] >= 0.0)\n",
    "                M.addConstr(z[j] <= Uj * s)\n",
    "                M.addConstr(z[j] <= a[j] - Lj * (1 - s))\n",
    "\n",
    "        prev = z\n",
    "\n",
    "    if W[-1].shape[0] != 1:\n",
    "        raise ValueError(f\"Expected scalar output, got out_dim={W[-1].shape[0]}\")\n",
    "\n",
    "    y_norm = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == b[-1][0] + gp.quicksum(W[-1][0, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "    M.setObjective(ys * y_norm + ym, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_maxaffine_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    W = base.W.detach().cpu().numpy().astype(float)\n",
    "    b = base.b.detach().cpu().numpy().astype(float)\n",
    "\n",
    "    Weff = W / xs[None, :]\n",
    "    beff = b - (Weff @ xm)\n",
    "\n",
    "    Weff *= a_out\n",
    "    beff  = a_out * beff + b_out\n",
    "\n",
    "    K = int(Weff.shape[0])\n",
    "\n",
    "    M = gp.Model(\"MaxAffine_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    t = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"t_norm\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(t >= beff[k] + gp.quicksum(Weff[k, j] * x[j] for j in range(d) if Weff[k, j] != 0.0))\n",
    "\n",
    "    M.setObjective(ys * t + ym, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_lset_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    A = model.A.detach().cpu().numpy().astype(float)\n",
    "    b = model.b.detach().cpu().numpy().astype(float)\n",
    "    T = float(model.T)\n",
    "    if T == 0.0:\n",
    "        raise ValueError(\"T must be nonzero\")\n",
    "\n",
    "    Aeff = A / xs[None, :]\n",
    "    beff = b - (Aeff @ xm)\n",
    "    K = int(Aeff.shape[0])\n",
    "\n",
    "    lin_lo = np.empty(K, dtype=float)\n",
    "    lin_hi = np.empty(K, dtype=float)\n",
    "    for k in range(K):\n",
    "        a = Aeff[k]\n",
    "        lo = beff[k]\n",
    "        hi = beff[k]\n",
    "        pos = a >= 0\n",
    "        lo += (a[pos] * x_min[pos]).sum() + (a[~pos] * x_max[~pos]).sum()\n",
    "        hi += (a[pos] * x_max[pos]).sum() + (a[~pos] * x_min[~pos]).sum()\n",
    "        lin_lo[k], lin_hi[k] = lo, hi\n",
    "\n",
    "    z_lo = lin_lo / T\n",
    "    z_hi = lin_hi / T\n",
    "    m_lo = float(np.max(z_lo))\n",
    "    m_hi = float(np.max(z_hi))\n",
    "\n",
    "    w_lo = float(np.min(z_lo - m_hi))  # <= 0\n",
    "    u_lo = float(np.exp(max(-700.0, w_lo)))\n",
    "    s_lo = max(1e-12, K * u_lo)\n",
    "    s_hi = float(K * 1.0)\n",
    "    v_lo = float(np.log(s_lo))\n",
    "    v_hi = float(np.log(s_hi))\n",
    "\n",
    "    yN_lo = float(T * (m_lo + v_lo))\n",
    "    yN_hi = float(T * (m_hi + v_hi))\n",
    "    y_lo  = float(ys * yN_lo + ym)\n",
    "    y_hi  = float(ys * yN_hi + ym)\n",
    "\n",
    "    M = gp.Model(\"lset_ip_stable_bounded\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    M.Params.FuncNonlinear = 1\n",
    "    M.Params.FeasibilityTol = 1e-9\n",
    "    M.Params.OptimalityTol  = 1e-9\n",
    "    M.Params.IntFeasTol     = 1e-9\n",
    "    M.Params.NumericFocus   = 3\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    z = M.addVars(K, lb=z_lo.tolist(), ub=z_hi.tolist(), vtype=GRB.CONTINUOUS, name=\"z\")\n",
    "    for k in range(K):\n",
    "        lin = beff[k] + gp.quicksum(Aeff[k, j] * x[j] for j in range(d) if Aeff[k, j] != 0.0)\n",
    "        M.addConstr(z[k] == lin / T, name=f\"zdef_{k}\")\n",
    "\n",
    "    m = M.addVar(lb=m_lo, ub=m_hi, vtype=GRB.CONTINUOUS, name=\"m\")\n",
    "    w = M.addVars(K, lb=w_lo, ub=0.0, vtype=GRB.CONTINUOUS, name=\"w\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(m >= z[k], name=f\"m_ge_z_{k}\")\n",
    "        M.addConstr(w[k] == z[k] - m, name=f\"wdef_{k}\")\n",
    "\n",
    "    u = M.addVars(K, lb=0.0, ub=1.0, vtype=GRB.CONTINUOUS, name=\"u\")\n",
    "    for k in range(K):\n",
    "        M.addGenConstrExp(w[k], u[k], name=f\"exp_{k}\")\n",
    "\n",
    "    s = M.addVar(lb=s_lo, ub=s_hi, vtype=GRB.CONTINUOUS, name=\"s\")\n",
    "    M.addConstr(s == gp.quicksum(u[k] for k in range(K)), name=\"sumexp_shifted\")\n",
    "\n",
    "    v = M.addVar(lb=v_lo, ub=v_hi, vtype=GRB.CONTINUOUS, name=\"v\")\n",
    "    M.addGenConstrLog(s, v, name=\"log_shifted\")\n",
    "\n",
    "    y_norm = M.addVar(lb=yN_lo, ub=yN_hi, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == T * (m + v), name=\"y_norm_def\")\n",
    "\n",
    "    y_raw = M.addVar(lb=y_lo, ub=y_hi, vtype=GRB.CONTINUOUS, name=\"y_raw\")\n",
    "    M.addConstr(y_raw == ys * y_norm + ym, name=\"y_raw_def\")\n",
    "\n",
    "    M.setObjective(y_raw, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution found. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], dtype=float)\n",
    "    y_star = float(y_raw.X)\n",
    "\n",
    "    info = {\n",
    "        \"status\": M.Status,\n",
    "        \"runtime\": M.Runtime,\n",
    "        \"gap\": getattr(M, \"MIPGap\", None),\n",
    "        \"sol_count\": M.SolCount,\n",
    "        \"obj_gurobi\": float(M.ObjVal),\n",
    "        \"obj_bound\": float(getattr(M, \"ObjBound\", float(\"nan\"))),\n",
    "    }\n",
    "    return x_star, y_star, info\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d4858979",
   "metadata": {},
   "source": [
    "## Test Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "569f2165",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def solve_ip(model_type, model, scaler, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    if model_type == \"DFN\":\n",
    "        return solve_dfn_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MLP\":\n",
    "        return solve_mlp_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MaxAffine\":\n",
    "        return solve_maxaffine_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"LSET\":\n",
    "        return solve_lset_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise ValueError(model_type)\n",
    "\n",
    "\n",
    "def suppress_stdout(fn, *args, silence: bool = False, **kwargs):\n",
    "    if not silence:\n",
    "        return fn(*args, **kwargs), \"\"\n",
    "    buf = io.StringIO()\n",
    "    with contextlib.redirect_stdout(buf):\n",
    "        out = fn(*args, **kwargs)\n",
    "    return out, buf.getvalue()\n",
    "\n",
    "\n",
    "def make_obj(model, scaler, device, chunk: int = 4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    @torch.no_grad()\n",
    "    def obj(Xraw):\n",
    "        Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "        if Xraw.dim() == 1:\n",
    "            Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "        outs = []\n",
    "        B = int(Xraw.shape[0])\n",
    "        for i in range(0, B, int(chunk)):\n",
    "            Xb = Xraw[i:i+chunk]\n",
    "            try:\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)  \n",
    "                y  = (yn * ys + ym).reshape(-1).detach().cpu() \n",
    "                if y.numel() != Xb.shape[0]:\n",
    "                    y = y.expand(Xb.shape[0]).contiguous()\n",
    "                outs.append(y)\n",
    "            except Exception as e:\n",
    "                obj._err_count += int(Xb.shape[0])\n",
    "                if obj._err_first is None:\n",
    "                    obj._err_first = repr(e)\n",
    "                outs.append(torch.full((int(Xb.shape[0]),), float(\"inf\"), device=\"cpu\"))\n",
    "\n",
    "        return torch.cat(outs, 0).numpy()\n",
    "\n",
    "    obj._err_count = 0\n",
    "    obj._err_first = None\n",
    "    return obj\n",
    "\n",
    "\n",
    "def safe_raw_mse(obj, Xraw, yraw):\n",
    "    yp = np.asarray(obj(Xraw), dtype=float).reshape(-1)\n",
    "    yt = yraw.detach().cpu().numpy().reshape(-1) if torch.is_tensor(yraw) else np.asarray(yraw, float).reshape(-1)\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "    mask = np.isfinite(yp)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions were non-finite (inf/nan)\"\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def safe_norm_mse(model, scaler, device, Xraw, yraw, *, chunk=4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "    if Xraw.dim() == 1:\n",
    "        Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "    yraw_t = yraw\n",
    "    if torch.is_tensor(yraw_t):\n",
    "        yraw_t = yraw_t.to(device=device, dtype=torch.float32)\n",
    "    else:\n",
    "        yraw_t = torch.as_tensor(yraw_t, dtype=torch.float32, device=device)\n",
    "    yraw_t = yraw_t.reshape(-1)\n",
    "\n",
    "    preds = []\n",
    "    B = int(Xraw.shape[0])\n",
    "    try:\n",
    "        with torch.no_grad():\n",
    "            for i in range(0, B, int(chunk)):\n",
    "                Xb = Xraw[i:i+chunk]\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)\n",
    "                if yn.numel() != Xb.shape[0]:\n",
    "                    yn = yn.expand(Xb.shape[0]).contiguous()\n",
    "                preds.append(yn.detach().cpu())\n",
    "    except Exception as e:\n",
    "        return np.nan, f\"model forward failed in safe_norm_mse: {repr(e)}\"\n",
    "\n",
    "    yp = torch.cat(preds, 0).numpy().reshape(-1)\n",
    "    yt = ((yraw_t - ym) / ys).detach().cpu().numpy().reshape(-1)\n",
    "\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "\n",
    "    mask = np.isfinite(yp) & np.isfinite(yt)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions or targets were non-finite (inf/nan)\"\n",
    "\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def t_to_best(hist, y_best):\n",
    "    for r in hist:\n",
    "        if abs(float(r.get(\"best_y\", np.inf)) - float(y_best)) < 1e-12:\n",
    "            return float(r.get(\"t\", float(\"nan\")))\n",
    "    return float(\"nan\")\n",
    "\n",
    "\n",
    "def check_ip_matches_obj(name, obj, x_ip, y_ip, *, strict: bool, tol: float):\n",
    "    y_obj = float(np.asarray(obj(np.asarray(x_ip)), dtype=float).reshape(-1)[0])\n",
    "    y_ip  = float(y_ip)\n",
    "    rel = abs(y_obj - y_ip) / (abs(y_obj) + 1e-12)\n",
    "    print(f\"[CHECK {name}] obj(x_ip)={y_obj:.6g}  ip_y={y_ip:.6g}  rel_err={rel:.3e}\")\n",
    "    if strict and rel > tol:\n",
    "        raise RuntimeError(f\"{name}: IP objective != obj() (rel_err={rel:.3e})\")\n",
    "    return rel\n",
    "\n",
    "\n",
    "def mean_se(x):\n",
    "    x = pd.to_numeric(pd.Series(x), errors=\"coerce\").to_numpy()\n",
    "    x = x[np.isfinite(x)]\n",
    "    n = int(x.shape[0])\n",
    "    if n == 0:\n",
    "        return np.nan, np.nan\n",
    "    m = float(x.mean())\n",
    "    se = float(x.std(ddof=1) / np.sqrt(n)) if n > 1 else 0.0\n",
    "    return m, se\n",
    "\n",
    "\n",
    "def fmt_mean_se(m, se):\n",
    "    if not np.isfinite(m):\n",
    "        return \"nan\"\n",
    "    if not np.isfinite(se):\n",
    "        return f\"{m:.6g}\"\n",
    "    return f\"{m:.6g} ± {se:.3g}\"\n",
    "\n",
    "\n",
    "def repr_solution(xs: pd.Series, seeds=None):\n",
    "    xs = xs.dropna().astype(str)\n",
    "    xs = xs[xs != \"None\"]\n",
    "    if xs.empty:\n",
    "        return None\n",
    "\n",
    "    # With seed information: show per-seed if solutions differ\n",
    "    if seeds is not None:\n",
    "        df = pd.DataFrame({\"seed\": pd.Series(seeds), \"x\": xs})\n",
    "        df = df.dropna()\n",
    "        df[\"x\"] = df[\"x\"].astype(str)\n",
    "        if df.empty:\n",
    "            return None\n",
    "        if df[\"x\"].nunique() == 1:\n",
    "            return df[\"x\"].iloc[0]\n",
    "        df = df.sort_values(\"seed\")\n",
    "        return \"\\n\".join([f\"seed={int(r.seed)}: {r.x}\" for r in df.itertuples(index=False)])\n",
    "\n",
    "    # No seed info: keep compact\n",
    "    if xs.nunique() == 1:\n",
    "        return xs.iloc[0]\n",
    "    vc = xs.value_counts()\n",
    "    top = vc.index[0]\n",
    "    n_unique = int(vc.shape[0])\n",
    "    return f\"{top} (+{n_unique-1} other)\"\n",
    "    \n",
    "\n",
    "# -----------------------------\n",
    "# Ground-truth optimum solvers\n",
    "# -----------------------------\n",
    "\n",
    "def solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    Q  = np.asarray(gt[\"Q\"], float)\n",
    "    xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "    n  = int(xs.shape[0])\n",
    "    assert Q.shape == (n, n)\n",
    "\n",
    "    m = gp.Model(\"gt_quadratic\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    x = m.addVars(n, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(n):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(n)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    expr = gp.quicksum(float(Q[i, j]) * (x[i] - float(xs[i])) * (x[j] - float(xs[j])) for i in range(n) for j in range(n))\n",
    "    m.setObjective(expr, GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT quadratic solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(n)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_assignment(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    C = np.asarray(gt[\"C\"], float)\n",
    "    n_rows, n_cols = C.shape\n",
    "    k = int(sum_eq)\n",
    "    if k > n_cols:\n",
    "        raise RuntimeError(f\"sum_eq={k} exceeds number of columns={n_cols} (infeasible)\")\n",
    "\n",
    "    m = gp.Model(\"gt_assignment\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    z = m.addVars(n_rows, vtype=GRB.BINARY, name=\"z\")  # select row i\n",
    "    y = m.addVars(n_rows, n_cols, vtype=GRB.BINARY, name=\"y\")  # assign row i to col j\n",
    "\n",
    "    m.addConstr(gp.quicksum(z[i] for i in range(n_rows)) == k, name=\"k_rows\")\n",
    "    for i in range(n_rows):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for j in range(n_cols)) == z[i], name=f\"assign_row_{i}\")\n",
    "    for j in range(n_cols):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for i in range(n_rows)) <= 1, name=f\"assign_col_{j}\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(C[i, j]) * y[i, j] for i in range(n_rows) for j in range(n_cols)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT assignment solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(z[i].X)) for i in range(n_rows)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    N  = int(gt[\"N\"])\n",
    "    SS = int(gt[\"SS\"])\n",
    "    TT = int(gt[\"TT\"])\n",
    "    src  = np.asarray(gt[\"src\"], np.int64)\n",
    "    dst  = np.asarray(gt[\"dst\"], np.int64)\n",
    "    cost = np.asarray(gt[\"cost\"], float)\n",
    "    cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "    idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "    idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "    E = int(src.shape[0])\n",
    "    k = int(idxSS.shape[0])\n",
    "    assert idxTT.shape[0] == k, \"Expect idxSS and idxTT to be same length\"\n",
    "\n",
    "    out_edges = [[] for _ in range(N)]\n",
    "    in_edges  = [[] for _ in range(N)]\n",
    "    for e in range(E):\n",
    "        out_edges[int(src[e])].append(e)\n",
    "        in_edges[int(dst[e])].append(e)\n",
    "\n",
    "    idxSS = idxSS.astype(int)\n",
    "    idxTT = idxTT.astype(int)\n",
    "    var_arc_to_i = {int(idxSS[i]): i for i in range(k)}\n",
    "    var_arc_to_i.update({int(idxTT[i]): i for i in range(k)})\n",
    "    var_arcs = sorted(var_arc_to_i.keys())\n",
    "\n",
    "    m = gp.Model(\"gt_mdvsp_true_opt\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    m.Params.NonConvex = 2\n",
    "\n",
    "    x = m.addVars(k, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(k):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(k)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    f = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, name=\"f\")\n",
    "\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            m.addConstr(f[e] <= x[i], name=f\"cap_var_{e}\")\n",
    "        else:\n",
    "            m.addConstr(f[e] <= float(cap0[e]), name=f\"cap_fix_{e}\")\n",
    "\n",
    "    F = m.addVar(vtype=GRB.CONTINUOUS, lb=0.0, name=\"F\")\n",
    "\n",
    "    # flow conservation: out - in = F at SS, = -F at TT, else 0\n",
    "    for v in range(N):\n",
    "        out_sum = gp.quicksum(f[e] for e in out_edges[v])\n",
    "        in_sum  = gp.quicksum(f[e] for e in in_edges[v])\n",
    "        rhs = F if v == SS else (-F if v == TT else 0.0)\n",
    "        m.addConstr(out_sum - in_sum == rhs, name=f\"flow_{v}\")\n",
    "\n",
    "    y = m.addVars(N, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"y\")\n",
    "    m.addConstr(y[SS] == 1.0, name=\"ySS\")\n",
    "    m.addConstr(y[TT] == 0.0, name=\"yTT\")\n",
    "\n",
    "    z = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"z\")\n",
    "    for e in range(E):\n",
    "        m.addConstr(z[e] >= y[int(src[e])] - y[int(dst[e])], name=f\"dual_{e}\")\n",
    "\n",
    "    dual_obj = gp.QuadExpr()\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            # bilinear term: x_i * z_e\n",
    "            dual_obj += x[i] * z[e]\n",
    "        else:\n",
    "            dual_obj += float(cap0[e]) * z[e]\n",
    "    m.addConstr(F == dual_obj, name=\"strong_duality\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(cost[e]) * f[e] for e in range(E)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT mdvsp true-opt failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(k)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status), \"F\": float(F.X)}\n",
    "    \n",
    "\n",
    "\n",
    "def solve_true_opt(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    \"\"\"Top-level dispatcher for ground-truth optimum.\"\"\"\n",
    "    if not isinstance(gt, dict):\n",
    "        raise RuntimeError(\"No ground-truth structure found (gt must be a dict).\")\n",
    "    t = gt.get(\"type\", None)\n",
    "    if t == \"quadratic\":\n",
    "        return solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"assignment\":\n",
    "        return solve_true_opt_assignment(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"mdvsp\":\n",
    "        return solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise RuntimeError(f\"Unknown gt type: {t}\")\n",
    "\n",
    "\n",
    "# -----------------------------\n",
    "# Benchmark runner\n",
    "# -----------------------------\n",
    "def run_benchmark(\n",
    "    *,\n",
    "    dataset_type: str,\n",
    "    dataset_params: dict,\n",
    "    runs: list[tuple[str, str, dict]],\n",
    "    train_base: dict,\n",
    "    lr_map: dict,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    n_seeds: int = 1,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    allow_plots_multi_seed: bool = True,\n",
    "    time_limit=None,\n",
    "):\n",
    "    seeds = list(range(int(n_seeds)))\n",
    "\n",
    "    learn_rows, opt_rows, spec_rows, fail_rows = [], [], [], []\n",
    "    gt_rows = []\n",
    "\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    for seed in seeds:\n",
    "        print(f\"\\n\\n===================== SEED {seed} =====================\")\n",
    "\n",
    "        for name, model_type, model_params_base in runs:\n",
    "            # ---- per-run params ----\n",
    "            dp = dict(dataset_params)\n",
    "            if vary_dataset_seed and \"seed\" in dp:\n",
    "                dp[\"seed\"] = int(seed)\n",
    "\n",
    "            mp = dict(model_params_base)\n",
    "            if vary_model_init_seed and \"seed\" in mp:\n",
    "                mp[\"seed\"] = int(seed)\n",
    "\n",
    "            tp = dict(train_base)\n",
    "            tp[\"seed\"] = int(seed)\n",
    "            tp[\"lr\"] = float(lr_map[model_type])\n",
    "\n",
    "            # avoid plot spam unless explicitly allowed\n",
    "            if (n_seeds > 1) and (tp.get(\"plot_every\", 0) not in (0, None)) and (not allow_plots_multi_seed):\n",
    "                tp[\"plot_every\"] = 0\n",
    "\n",
    "            # ---- TRAIN ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                out = generate_and_train_simple(dataset_type, dp, model_type, mp, tp)\n",
    "            except Exception as e:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=repr(e)))\n",
    "                learn_rows.append(dict(seed=seed, model=name, train_time=np.nan, best_epoch=np.nan,\n",
    "                                       best_val=np.nan, test=np.nan, train_err=repr(e)))\n",
    "                continue\n",
    "            train_time = time.perf_counter() - t0\n",
    "\n",
    "            if isinstance(out, tuple) and len(out) == 4:\n",
    "                model, data, hist, spec = out\n",
    "            elif isinstance(out, tuple) and len(out) == 3:\n",
    "                model, data, hist = out\n",
    "                n_params = sum(p.numel() for p in model.parameters())\n",
    "                spec = dict(n_params=int(n_params), extra=\"\", train_params=tp)\n",
    "            else:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=f\"Unexpected return: {type(out)}\"))\n",
    "                continue\n",
    "\n",
    "            # ---- model spec (seed 0 only) ----\n",
    "            if seed == seeds[0]:\n",
    "                spec_rows.append(dict(\n",
    "                    model=name,\n",
    "                    n_params=int(spec.get(\"n_params\", np.nan)),\n",
    "                    details=str(spec.get(\"extra\", \"\")),\n",
    "                    lr=float(spec.get(\"train_params\", {}).get(\"lr\", tp[\"lr\"])),\n",
    "                    batch_size=int(spec.get(\"train_params\", {}).get(\"batch_size\", tp[\"batch_size\"])),\n",
    "                    epochs=int(spec.get(\"train_params\", {}).get(\"epochs\", tp[\"epochs\"])),\n",
    "                ))\n",
    "\n",
    "            device = data[\"device\"]\n",
    "            scaler = data[\"scaler\"]\n",
    "            obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "            gt = data.get(\"true\", None)\n",
    "\n",
    "            # ---- compute GT optimum (once per seed) ----\n",
    "            if seed not in gt_cache_by_seed:\n",
    "                t_gt0 = time.perf_counter()\n",
    "                try:\n",
    "                    x_gt, y_gt, gt_info = solve_true_opt(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=False)\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    # evaluate with the existing helper to be extra sure\n",
    "                    y_gt_check = float(eval_true_obj(gt, x_gt))\n",
    "                    if np.isfinite(y_gt_check) and abs(y_gt_check - float(y_gt)) / (abs(float(y_gt_check)) + 1e-12) > 1e-6:\n",
    "                        fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT_CHECK\",\n",
    "                                              error=f\"gt solver mismatch: solver={y_gt:.6g} eval_true_obj={y_gt_check:.6g}\"))\n",
    "                    gt_cache_by_seed[seed] = dict(x=str(np.asarray(x_gt, int).tolist()),\n",
    "                                                  true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                                                  runtime=float(gt_time),\n",
    "                                                  err=None)\n",
    "                except Exception as e:\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "            # ---- learning metrics: best val (normalized) + test loss (normalized) ----\n",
    "            test_norm, err_te = safe_norm_mse(model, scaler, device, data['raw']['Xte'], data['raw']['yte'], chunk=int(tp.get('plot_chunk', 4096)))\n",
    "            if err_te:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"EVAL_TEST\", error=err_te))\n",
    "\n",
    "            if obj._err_count > 0:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                      error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "            val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "            best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "            best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "            learn_rows.append(dict(\n",
    "                seed=seed, model=name,\n",
    "                train_time=float(train_time),\n",
    "                best_epoch=best_ep,\n",
    "                best_val=float(best_val),\n",
    "                test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "                train_err=(err_te),\n",
    "            ))\n",
    "\n",
    "            # ---- LOCAL SEARCH ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "                (ls_out, _ls_log) = suppress_stdout(\n",
    "                    local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                    delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                    silence=silence_local_search\n",
    "                )\n",
    "                x_best, y_best, ls_hist = ls_out\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=str(np.asarray(x_best, int).tolist()),\n",
    "                    y=float(y_best),\n",
    "                    true_y=float(eval_true_obj(gt, x_best)),\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                    iters=int(len(ls_hist) - 1),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=np.nan, iters=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "            # ---- IP SOLVE (learned objective) ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                x_ip, y_ip, info = solve_ip(\n",
    "                    model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                    time_limit=time_limit, verbose=True\n",
    "                )\n",
    "                ip_time = time.perf_counter() - t0\n",
    "\n",
    "                rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "                if rel_err > ip_check_tol:\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                          error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=str(np.asarray(x_ip, int).tolist()),\n",
    "                    y=float(y_ip),\n",
    "                    true_y=float(eval_true_obj(gt, x_ip)),\n",
    "                    runtime=float(ip_time),\n",
    "                    status=(info.get(\"status\") if isinstance(info, dict) else None),\n",
    "                    gap=(info.get(\"gap\") if isinstance(info, dict) else None),\n",
    "                    ip_rel_err=float(rel_err),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ip_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ip_time),\n",
    "                    status=None, gap=None, ip_rel_err=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    spec_df  = pd.DataFrame(spec_rows).drop_duplicates(\"model\").sort_values(\"model\").reset_index(drop=True)\n",
    "    learn_df = pd.DataFrame(learn_rows).sort_values([\"model\", \"seed\"]).reset_index(drop=True)\n",
    "    opt_df   = pd.DataFrame(opt_rows).sort_values([\"model\", \"seed\", \"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # ---- per-seed gaps ----\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\", \"model\", \"ls_vs_ip_pct\", \"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        # IP true vs GT optimum true\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    for model in sorted(learn_df[\"model\"].unique()):\n",
    "        sub = learn_df[learn_df[\"model\"] == model]\n",
    "        m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "        learn_sum_rows.append(dict(model=model, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- OPTIMIZATION SUMMARY ----\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model]\n",
    "        row = {\"model\": model}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== MODEL SPECS (from seed 0 run) ===\")\n",
    "    if not spec_df.empty:\n",
    "        print(spec_df[[\"model\", \"n_params\", \"details\", \"lr\", \"batch_size\", \"epochs\"]].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not learn_summary_df.empty:\n",
    "        print(learn_summary_df.to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    return spec_df, learn_df, opt_df, fail_df, learn_summary_df, opt_summary_df, gt_df\n",
    "\n",
    "# ===========================\n",
    "# Manual-seed training + autosave helpers\n",
    "# ===========================\n",
    "import json\n",
    "import datetime\n",
    "\n",
    "def _to_jsonable(x):\n",
    "    \"\"\"Best-effort conversion to JSON-serializable types.\"\"\"\n",
    "    if x is None:\n",
    "        return None\n",
    "    if isinstance(x, (str, int, float, bool)):\n",
    "        return x\n",
    "    if isinstance(x, (np.integer, np.floating)):\n",
    "        return x.item()\n",
    "    if isinstance(x, (list, tuple)):\n",
    "        return [_to_jsonable(v) for v in x]\n",
    "    if isinstance(x, dict):\n",
    "        return {str(k): _to_jsonable(v) for k, v in x.items()}\n",
    "    if isinstance(x, np.ndarray):\n",
    "        return x.tolist()\n",
    "    if torch.is_tensor(x):\n",
    "        try:\n",
    "            return x.detach().cpu().tolist()\n",
    "        except Exception:\n",
    "            return str(x)\n",
    "    return str(x)\n",
    "\n",
    "\n",
    "def make_experiment_dir(root: Path, dataset_type: str, experiment_name: str | None = None):\n",
    "    \"\"\"\n",
    "    Creates an experiment folder and returns its Path.\n",
    "    \"\"\"\n",
    "    root = Path(root)\n",
    "    root.mkdir(parents=True, exist_ok=True)\n",
    "    ts = datetime.datetime.now().strftime(\"%Y%m%d-%H%M%S\")\n",
    "    if experiment_name is None or str(experiment_name).strip() == \"\":\n",
    "        experiment_name = \"run\"\n",
    "    exp_dir = root / f\"{dataset_type}__{experiment_name}__{ts}\"\n",
    "    exp_dir.mkdir(parents=True, exist_ok=True)\n",
    "    (exp_dir / \"models\").mkdir(exist_ok=True)\n",
    "    return exp_dir\n",
    "\n",
    "\n",
    "def _capture_stdout(fn, *args, **kwargs):\n",
    "    buf = io.StringIO()\n",
    "    with contextlib.redirect_stdout(buf):\n",
    "        out = fn(*args, **kwargs)\n",
    "    return out, buf.getvalue()\n",
    "\n",
    "\n",
    "def _save_manifest(exp_dir: Path, manifest: dict):\n",
    "    path = Path(exp_dir) / \"manifest.json\"\n",
    "    with open(path, \"w\", encoding=\"utf-8\") as f:\n",
    "        json.dump(_to_jsonable(manifest), f, indent=2, sort_keys=True)\n",
    "\n",
    "\n",
    "def save_trained_artifacts(\n",
    "    *,\n",
    "    exp_dir: Path,\n",
    "    dataset_type: str,\n",
    "    seed: int,\n",
    "    run_name: str,\n",
    "    model_type: str,\n",
    "    model_params: dict,\n",
    "    dataset_params: dict,\n",
    "    train_params: dict,\n",
    "    model: torch.nn.Module,\n",
    "    scaler: dict,\n",
    "    history: dict,\n",
    "    stdout_log: str,\n",
    "    raw_data: dict | None = None,\n",
    "    gt: Any | None = None,\n",
    "    spec: dict | None = None,\n",
    "    error: str | None = None,\n",
    "    overwrite: bool = False,\n",
    "):\n",
    "    \"\"\"\n",
    "    Saves model + scaler + history + logs to disk immediately after training finishes.\n",
    "    \"\"\"\n",
    "    exp_dir = Path(exp_dir)\n",
    "    run_dir = exp_dir / \"models\" / f\"seed={int(seed)}\" / f\"model={str(run_name)}\"\n",
    "    run_dir.mkdir(parents=True, exist_ok=True)\n",
    "\n",
    "    meta_path = run_dir / \"meta.json\"\n",
    "    if meta_path.exists() and not overwrite:\n",
    "        stamp = datetime.datetime.now().strftime(\"%Y%m%d-%H%M%S\")\n",
    "        run_dir = exp_dir / \"models\" / f\"seed={int(seed)}\" / f\"model={str(run_name)}__{stamp}\"\n",
    "        run_dir.mkdir(parents=True, exist_ok=True)\n",
    "        meta_path = run_dir / \"meta.json\"\n",
    "\n",
    "    meta = dict(\n",
    "        dataset_type=str(dataset_type),\n",
    "        seed=int(seed),\n",
    "        run_name=str(run_name),\n",
    "        model_type=str(model_type),\n",
    "        model_params=dict(model_params),\n",
    "        dataset_params=dict(dataset_params),\n",
    "        train_params=dict(train_params),\n",
    "        spec=(dict(spec) if isinstance(spec, dict) else None),\n",
    "        error=error,\n",
    "        saved_at=datetime.datetime.now().isoformat(),\n",
    "    )\n",
    "    with open(meta_path, \"w\", encoding=\"utf-8\") as f:\n",
    "        json.dump(_to_jsonable(meta), f, indent=2, sort_keys=True)\n",
    "\n",
    "    # model state\n",
    "    torch.save(\n",
    "        {\n",
    "            \"model_type\": str(model_type),\n",
    "            \"model_params\": dict(model_params),\n",
    "            \"state_dict\": model.state_dict(),\n",
    "        },\n",
    "        run_dir / \"model_state.pt\",\n",
    "    )\n",
    "\n",
    "    # scaler + training history\n",
    "    torch.save(scaler, run_dir / \"scaler.pt\")\n",
    "    torch.save(history, run_dir / \"history.pt\")\n",
    "\n",
    "    # optional raw data (can be big!)\n",
    "    if raw_data is not None:\n",
    "        torch.save(raw_data, run_dir / \"raw_data.pt\")\n",
    "\n",
    "    # optional gt (best-effort; may fail depending on object type)\n",
    "    if gt is not None:\n",
    "        try:\n",
    "            torch.save(gt, run_dir / \"gt.pt\")\n",
    "        except Exception:\n",
    "            pass\n",
    "\n",
    "    # logs\n",
    "    with open(run_dir / \"stdout.txt\", \"w\", encoding=\"utf-8\") as f:\n",
    "        f.write(stdout_log or \"\")\n",
    "\n",
    "    return run_dir\n",
    "\n",
    "\n",
    "def load_trained_artifacts(run_dir: Path, *, map_location: str = \"cpu\"):\n",
    "    \"\"\"\n",
    "    Loads a saved model+scaler+history from disk and reconstructs the model object.\n",
    "    \"\"\"\n",
    "    run_dir = Path(run_dir)\n",
    "    payload = torch.load(run_dir / \"model_state.pt\", map_location=map_location)\n",
    "    mt = payload[\"model_type\"]\n",
    "    mp = payload[\"model_params\"]\n",
    "\n",
    "    # Rebuild model with the same constructors used in generate_and_train_simple\n",
    "    if mt == \"DFN\":\n",
    "        model = DFN(**mp)\n",
    "    elif mt == \"MLP\":\n",
    "        mp2 = dict(mp); mp2.pop(\"alpha\", None); mp2.pop(\"beta\", None)\n",
    "        model = MLP(**mp2)\n",
    "    elif mt == \"MaxAffine\":\n",
    "        mp2 = dict(mp); mp2.pop(\"alpha\", None); mp2.pop(\"beta\", None)\n",
    "        model = MaxAffine(**mp2)\n",
    "    elif mt == \"LSET\":\n",
    "        mp2 = dict(mp); mp2.pop(\"alpha\", None); mp2.pop(\"beta\", None)\n",
    "        model = LSET(**mp2)\n",
    "    else:\n",
    "        raise ValueError(f\"Unknown model_type: {mt}\")\n",
    "\n",
    "    model.load_state_dict(payload[\"state_dict\"])\n",
    "    model.eval()\n",
    "\n",
    "    scaler = torch.load(run_dir / \"scaler.pt\", map_location=map_location)\n",
    "    history = torch.load(run_dir / \"history.pt\", map_location=map_location)\n",
    "\n",
    "    raw_path = run_dir / \"raw_data.pt\"\n",
    "    raw_data = torch.load(raw_path, map_location=map_location) if raw_path.exists() else None\n",
    "\n",
    "    gt_path = run_dir / \"gt.pt\"\n",
    "    gt = None\n",
    "    if gt_path.exists():\n",
    "        try:\n",
    "            gt = torch.load(gt_path, map_location=map_location)\n",
    "        except Exception:\n",
    "            gt = None\n",
    "\n",
    "    meta = {}\n",
    "    meta_path = run_dir / \"meta.json\"\n",
    "    if meta_path.exists():\n",
    "        with open(meta_path, \"r\", encoding=\"utf-8\") as f:\n",
    "            meta = json.load(f)\n",
    "\n",
    "    return dict(model=model, scaler=scaler, history=history, raw_data=raw_data, gt=gt, meta=meta, run_dir=str(run_dir))\n",
    "\n",
    "\n",
    "def init_learn_bundle(dataset_type: str, dataset_params: dict, runs: list, train_base: dict, lr_map: dict, exp_dir: Path):\n",
    "    return dict(\n",
    "        dataset_type=str(dataset_type),\n",
    "        dataset_params=dict(dataset_params),\n",
    "        runs=list(runs),\n",
    "        train_base=dict(train_base),\n",
    "        lr_map=dict(lr_map),\n",
    "        exp_dir=str(exp_dir),\n",
    "        trained={},      # (seed, name) -> in-memory objects\n",
    "        learn_rows=[],   # running log of learning metrics\n",
    "        fail_rows=[],\n",
    "        manifest_version=1,\n",
    "    )\n",
    "\n",
    "\n",
    "def train_one_and_save(\n",
    "    bundle: dict,\n",
    "    *,\n",
    "    seed: int,\n",
    "    run_name: str,\n",
    "    model_type: str,\n",
    "    model_params_base: dict,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    allow_plots: bool = True,\n",
    "    save_data: bool = True,\n",
    "    overwrite: bool = False,\n",
    "):\n",
    "    \"\"\"\n",
    "    Train ONE (seed, model) and autosave to disk immediately when done.\n",
    "    Updates bundle in-place and also writes manifest.json.\n",
    "    \"\"\"\n",
    "    dataset_type = bundle[\"dataset_type\"]\n",
    "    base_dp = bundle[\"dataset_params\"]\n",
    "    train_base = bundle[\"train_base\"]\n",
    "    lr_map = bundle[\"lr_map\"]\n",
    "    exp_dir = Path(bundle[\"exp_dir\"])\n",
    "\n",
    "    # per-run params\n",
    "    dp = dict(base_dp)\n",
    "    if vary_dataset_seed and \"seed\" in dp:\n",
    "        dp[\"seed\"] = int(seed)\n",
    "\n",
    "    mp = dict(model_params_base)\n",
    "    if vary_model_init_seed and \"seed\" in mp:\n",
    "        mp[\"seed\"] = int(seed)\n",
    "\n",
    "    tp = dict(train_base)\n",
    "    tp[\"seed\"] = int(seed)\n",
    "    tp[\"lr\"] = float(lr_map[model_type])\n",
    "\n",
    "    # Avoid plot spam unless you want it.\n",
    "    if not allow_plots:\n",
    "        tp[\"plot_every\"] = 0\n",
    "\n",
    "    # TRAIN (capture stdout so it gets saved)\n",
    "    t0 = time.perf_counter()\n",
    "    stdout_log = \"\"\n",
    "    run_dir = None\n",
    "    try:\n",
    "        out, stdout_log = _capture_stdout(generate_and_train_simple, dataset_type, dp, model_type, mp, tp)\n",
    "        train_time = time.perf_counter() - t0\n",
    "\n",
    "        if isinstance(out, tuple) and len(out) == 4:\n",
    "            model, data, hist, spec = out\n",
    "        elif isinstance(out, tuple) and len(out) == 3:\n",
    "            model, data, hist = out\n",
    "            n_params = sum(p.numel() for p in model.parameters())\n",
    "            spec = dict(n_params=int(n_params), extra=\"\", train_params=tp)\n",
    "        else:\n",
    "            raise RuntimeError(f\"Unexpected return from generate_and_train_simple: {type(out)}\")\n",
    "\n",
    "        model.eval()\n",
    "\n",
    "        # optional raw data (for full reproducibility / later eval without regen)\n",
    "        raw_data = None\n",
    "        if save_data and isinstance(data, dict) and \"raw\" in data:\n",
    "            raw_data = {\n",
    "                \"Xtr\": data[\"raw\"][\"Xtr\"].detach().cpu(),\n",
    "                \"ytr\": data[\"raw\"][\"ytr\"].detach().cpu(),\n",
    "                \"Xva\": data[\"raw\"][\"Xva\"].detach().cpu(),\n",
    "                \"yva\": data[\"raw\"][\"yva\"].detach().cpu(),\n",
    "                \"Xte\": data[\"raw\"][\"Xte\"].detach().cpu(),\n",
    "                \"yte\": data[\"raw\"][\"yte\"].detach().cpu(),\n",
    "            }\n",
    "\n",
    "        scaler = {k: (v.detach().cpu() if torch.is_tensor(v) else v) for k, v in data.get(\"scaler\", {}).items()}\n",
    "        gt = data.get(\"true\", None)\n",
    "\n",
    "        # save immediately (store cpu weights; load anywhere)\n",
    "        run_dir = save_trained_artifacts(\n",
    "            exp_dir=exp_dir,\n",
    "            dataset_type=dataset_type,\n",
    "            seed=seed,\n",
    "            run_name=run_name,\n",
    "            model_type=model_type,\n",
    "            model_params=mp,\n",
    "            dataset_params=dp,\n",
    "            train_params=tp,\n",
    "            model=model.cpu(),\n",
    "            scaler=scaler,\n",
    "            history=hist,\n",
    "            stdout_log=stdout_log,\n",
    "            raw_data=raw_data,\n",
    "            gt=gt,\n",
    "            spec=spec,\n",
    "            error=None,\n",
    "            overwrite=overwrite,\n",
    "        )\n",
    "\n",
    "        # compute quick learning metrics\n",
    "        device = \"cpu\"\n",
    "        Xte = raw_data[\"Xte\"] if raw_data is not None else data[\"raw\"][\"Xte\"].detach().cpu()\n",
    "        yte = raw_data[\"yte\"] if raw_data is not None else data[\"raw\"][\"yte\"].detach().cpu()\n",
    "        test_norm, err_te = safe_norm_mse(model.cpu(), scaler, device, Xte, yte, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "\n",
    "        val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "        best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "        best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "        bundle[\"trained\"][(int(seed), str(run_name))] = dict(\n",
    "            seed=int(seed),\n",
    "            name=str(run_name),\n",
    "            model_type=str(model_type),\n",
    "            model=model.cpu(),\n",
    "            scaler=scaler,\n",
    "            device=\"cpu\",\n",
    "            gt=gt,\n",
    "            train_params=tp,\n",
    "            dataset_params=dp,\n",
    "            model_params=mp,\n",
    "            history=hist,\n",
    "            run_dir=str(run_dir),\n",
    "        )\n",
    "        bundle[\"learn_rows\"].append(dict(\n",
    "            seed=int(seed),\n",
    "            model=str(run_name),\n",
    "            train_time=float(train_time),\n",
    "            best_epoch=best_ep,\n",
    "            best_val=float(best_val),\n",
    "            test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "            train_err=err_te,\n",
    "            run_dir=str(run_dir),\n",
    "        ))\n",
    "\n",
    "    except Exception as e:\n",
    "        train_time = time.perf_counter() - t0\n",
    "        # try to save failure metadata too\n",
    "        try:\n",
    "            run_dir = save_trained_artifacts(\n",
    "                exp_dir=exp_dir,\n",
    "                dataset_type=dataset_type,\n",
    "                seed=seed,\n",
    "                run_name=run_name,\n",
    "                model_type=model_type,\n",
    "                model_params=mp,\n",
    "                dataset_params=dp,\n",
    "                train_params=tp,\n",
    "                model=torch.nn.Identity(),\n",
    "                scaler={},\n",
    "                history={},\n",
    "                stdout_log=stdout_log,\n",
    "                raw_data=None,\n",
    "                gt=None,\n",
    "                spec=None,\n",
    "                error=repr(e),\n",
    "                overwrite=True,\n",
    "            )\n",
    "        except Exception:\n",
    "            run_dir = None\n",
    "\n",
    "        bundle[\"fail_rows\"].append(dict(seed=int(seed), model=str(run_name), stage=\"TRAIN\", error=repr(e), run_dir=str(run_dir) if run_dir else None))\n",
    "        bundle[\"learn_rows\"].append(dict(seed=int(seed), model=str(run_name), train_time=float(train_time),\n",
    "                                         best_epoch=np.nan, best_val=np.nan, test=np.nan, train_err=repr(e),\n",
    "                                         run_dir=str(run_dir) if run_dir else None))\n",
    "\n",
    "    # write manifest after EVERY model\n",
    "    manifest = dict(\n",
    "        dataset_type=bundle[\"dataset_type\"],\n",
    "        exp_dir=bundle[\"exp_dir\"],\n",
    "        completed=[{\"seed\": s, \"model\": m, \"run_dir\": info.get(\"run_dir\")} for (s,m), info in bundle[\"trained\"].items()],\n",
    "        failures=bundle.get(\"fail_rows\", []),\n",
    "    )\n",
    "    _save_manifest(Path(bundle[\"exp_dir\"]), manifest)\n",
    "    return bundle\n",
    "\n",
    "\n",
    "def summarize_learning_bundle(bundle: dict):\n",
    "    \"\"\"Build learning summary tables over whatever you have trained so far.\"\"\"\n",
    "    learn_df = pd.DataFrame(bundle.get(\"learn_rows\", []))\n",
    "    fail_df = pd.DataFrame(bundle.get(\"fail_rows\", []))\n",
    "    if learn_df.empty:\n",
    "        learn_df = pd.DataFrame(columns=[\"seed\",\"model\",\"train_time\",\"best_epoch\",\"best_val\",\"test\",\"train_err\",\"run_dir\"])\n",
    "\n",
    "    learn_sum_rows = []\n",
    "    if not learn_df.empty and \"model\" in learn_df.columns:\n",
    "        for model_name in sorted(learn_df[\"model\"].unique()):\n",
    "            sub = learn_df[learn_df[\"model\"] == model_name]\n",
    "            m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "            m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "            m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "            learn_sum_rows.append(dict(model=model_name, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    return learn_df, fail_df, learn_summary_df\n",
    "\n",
    "\n",
    "def load_experiment_bundle(exp_dir: Path, *, map_location: str = \"cpu\"):\n",
    "    \"\"\"\n",
    "    Reconstruct a learn_bundle from disk (useful after kernel restart).\n",
    "    \"\"\"\n",
    "    exp_dir = Path(exp_dir)\n",
    "    manifest_path = exp_dir / \"manifest.json\"\n",
    "    if not manifest_path.exists():\n",
    "        raise FileNotFoundError(f\"No manifest.json found in {exp_dir}\")\n",
    "\n",
    "    with open(manifest_path, \"r\", encoding=\"utf-8\") as f:\n",
    "        manifest = json.load(f)\n",
    "\n",
    "    trained = {}\n",
    "    learn_rows = []\n",
    "    fail_rows = list(manifest.get(\"failures\", []))\n",
    "\n",
    "    model_dirs = sorted((exp_dir / \"models\").glob(\"seed=*/model=*\"))\n",
    "    for run_dir in model_dirs:\n",
    "        try:\n",
    "            loaded = load_trained_artifacts(run_dir, map_location=map_location)\n",
    "            meta = loaded[\"meta\"] or {}\n",
    "            if meta.get(\"error\"):\n",
    "                continue\n",
    "            seed = int(meta.get(\"seed\"))\n",
    "            name = str(meta.get(\"run_name\"))\n",
    "            mt = str(meta.get(\"model_type\"))\n",
    "            mp = dict(meta.get(\"model_params\", {}))\n",
    "            dp = dict(meta.get(\"dataset_params\", {}))\n",
    "            tp = dict(meta.get(\"train_params\", {}))\n",
    "\n",
    "            trained[(seed, name)] = dict(\n",
    "                seed=seed, name=name, model_type=mt,\n",
    "                model=loaded[\"model\"],\n",
    "                scaler=loaded[\"scaler\"],\n",
    "                device=str(map_location),\n",
    "                gt=loaded.get(\"gt\", None),\n",
    "                train_params=tp,\n",
    "                dataset_params=dp,\n",
    "                model_params=mp,\n",
    "                history=loaded[\"history\"],\n",
    "                run_dir=str(run_dir),\n",
    "            )\n",
    "\n",
    "            hist = loaded[\"history\"] or {}\n",
    "            val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "            best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "            best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "            raw = loaded.get(\"raw_data\", None)\n",
    "            test_norm = np.nan\n",
    "            if raw is not None:\n",
    "                test_norm, _ = safe_norm_mse(loaded[\"model\"], loaded[\"scaler\"], map_location, raw[\"Xte\"], raw[\"yte\"], chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "\n",
    "            learn_rows.append(dict(\n",
    "                seed=seed, model=name, train_time=np.nan, best_epoch=best_ep,\n",
    "                best_val=best_val, test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "                train_err=None, run_dir=str(run_dir),\n",
    "            ))\n",
    "        except Exception:\n",
    "            continue\n",
    "\n",
    "    bundle = dict(\n",
    "        dataset_type=str(manifest.get(\"dataset_type\")),\n",
    "        dataset_params={},\n",
    "        runs=[],\n",
    "        train_base={},\n",
    "        lr_map={},\n",
    "        exp_dir=str(exp_dir),\n",
    "        trained=trained,\n",
    "        learn_rows=learn_rows,\n",
    "        fail_rows=fail_rows,\n",
    "        manifest_version=1,\n",
    "    )\n",
    "    return bundle\n",
    "\n",
    "\n",
    "def optimize_bundle(\n",
    "    bundle: dict,\n",
    "    *,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    time_limit=None,\n",
    "):\n",
    "    \"\"\"\n",
    "    Run optimization on whatever models currently exist in `bundle[\"trained\"]`.\n",
    "    (No retraining.)\n",
    "    \"\"\"\n",
    "    trained = bundle.get(\"trained\", {})\n",
    "    if not trained:\n",
    "        raise ValueError(\"No trained models in bundle. Train some first (or load_experiment_bundle).\")\n",
    "\n",
    "    opt_rows, fail_rows = [], []\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    # compute GT optimum once per seed (if gt available)\n",
    "    for (seed, _name), info in trained.items():\n",
    "        seed = int(seed)\n",
    "        if seed in gt_cache_by_seed:\n",
    "            continue\n",
    "        gt = info.get(\"gt\", None)\n",
    "        if gt is None:\n",
    "            gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=np.nan, err=\"No gt available\")\n",
    "            continue\n",
    "        t_gt0 = time.perf_counter()\n",
    "        try:\n",
    "            x_gt, y_gt, _ = solve_true_opt(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=False)\n",
    "            gt_time = time.perf_counter() - t_gt0\n",
    "            y_gt_check = float(eval_true_obj(gt, x_gt))\n",
    "            gt_cache_by_seed[seed] = dict(\n",
    "                x=str(np.asarray(x_gt, int).tolist()),\n",
    "                true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                runtime=float(gt_time),\n",
    "                err=None,\n",
    "            )\n",
    "        except Exception as e:\n",
    "            gt_time = time.perf_counter() - t_gt0\n",
    "            gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "            fail_rows.append(dict(seed=seed, model=\"(GT)\", stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "    # per-model optimization\n",
    "    keys = sorted(trained.keys(), key=lambda k: (int(k[0]), str(k[1])))\n",
    "    for (seed, name) in keys:\n",
    "        info = trained[(seed, name)]\n",
    "        model_type = info[\"model_type\"]\n",
    "        model = info[\"model\"]\n",
    "        scaler = info[\"scaler\"]\n",
    "        device = info.get(\"device\", \"cpu\")\n",
    "        gt = info.get(\"gt\", None)\n",
    "        tp = info.get(\"train_params\", {})\n",
    "\n",
    "        print(f\"\\n\\n===================== OPTIMIZATION | SEED {seed} | {name} =====================\")\n",
    "\n",
    "        obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "        if obj._err_count > 0:\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                  error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "        # LOCAL SEARCH\n",
    "        t0 = time.perf_counter()\n",
    "        try:\n",
    "            obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "            (ls_out, _ls_log) = suppress_stdout(\n",
    "                local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                silence=silence_local_search\n",
    "            )\n",
    "            x_best, y_best, ls_hist = ls_out\n",
    "            ls_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"LS\",\n",
    "                x=str(np.asarray(x_best, int).tolist()),\n",
    "                y=float(y_best),\n",
    "                true_y=float(eval_true_obj(gt, x_best)) if gt is not None else np.nan,\n",
    "                runtime=float(ls_time),\n",
    "                t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                iters=int(len(ls_hist) - 1),\n",
    "                err=None,\n",
    "            ))\n",
    "        except Exception as e:\n",
    "            ls_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"LS\",\n",
    "                x=None, y=np.nan, true_y=np.nan,\n",
    "                runtime=float(ls_time),\n",
    "                t_best=np.nan, iters=np.nan,\n",
    "                err=repr(e),\n",
    "            ))\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "        # IP SOLVE\n",
    "        t0 = time.perf_counter()\n",
    "        try:\n",
    "            x_ip, y_ip, info_ip = solve_ip(\n",
    "                model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                time_limit=time_limit, verbose=True\n",
    "            )\n",
    "            ip_time = time.perf_counter() - t0\n",
    "\n",
    "            rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "            if rel_err > ip_check_tol:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                      error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"IP\",\n",
    "                x=str(np.asarray(x_ip, int).tolist()),\n",
    "                y=float(y_ip),\n",
    "                true_y=float(eval_true_obj(gt, x_ip)) if gt is not None else np.nan,\n",
    "                runtime=float(ip_time),\n",
    "                status=(info_ip.get(\"status\") if isinstance(info_ip, dict) else None),\n",
    "                gap=(info_ip.get(\"gap\") if isinstance(info_ip, dict) else None),\n",
    "                ip_rel_err=float(rel_err),\n",
    "                err=None,\n",
    "            ))\n",
    "        except Exception as e:\n",
    "            ip_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"IP\",\n",
    "                x=None, y=np.nan, true_y=np.nan,\n",
    "                runtime=float(ip_time),\n",
    "                status=None, gap=None, ip_rel_err=np.nan,\n",
    "                err=repr(e),\n",
    "            ))\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    opt_df = pd.DataFrame(opt_rows).sort_values([\"model\",\"seed\",\"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\",\"model\",\"stage\",\"error\"])\n",
    "    else:\n",
    "        fail_df = fail_df.sort_values([\"stage\",\"model\",\"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # summary table (mean ± SE)\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\",\"model\",\"ls_vs_ip_pct\",\"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\",\"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\",\"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model_name in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model_name]\n",
    "        row = {\"model\": model_name}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model_name] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    return opt_df, fail_df, opt_summary_df, gt_df, gap_seed_df\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e88e7a1f",
   "metadata": {},
   "source": [
    "## Tests"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "813038a0-30b4-4138-b56a-397a9bc381a9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[MDVSP] inferred in_dim=8 from Xtmp shape=(5000, 8)\n",
      "\n",
      "\n",
      "===================== SEED 0 =====================\n",
      "\n",
      "--- Dataset stats (mdvsp) ---\n",
      "  X: shape=(5000, 8)  mean(mean)=12.5  std(mean)=7.49  min=0  max=25\n",
      "  y: shape=(5000,)  mean=1.01e+06  std=2.12e+05  min=2.72e+05  max=1.75e+06\n",
      "\n",
      "\n",
      "=== Run: mdvsp | DFN ===\n",
      "  data: N=5000  train/val/test=3500/750/750  dim=8\n",
      "  model: params=16,663 layers=[16, 128, 16] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[DONE] best val MSE (norm) = 3.867e-06 @ epoch 68\n",
      "\n",
      "Set parameter Username\n",
      "Set parameter LicenseID to value 2685751\n",
      "Academic license - for non-commercial use only - expires 2026-07-09\n"
     ]
    }
   ],
   "source": [
    "# ===================== MDVSP DATASET =====================\n",
    "N_SEEDS = 10\n",
    "VARY_DATASET_SEED = False\n",
    "VARY_MODEL_INIT_SEED = True\n",
    "STRICT_IP_CHECK = False\n",
    "IP_CHECK_TOL = 1e-4\n",
    "SILENCE_LOCAL_SEARCH = False\n",
    "ALLOW_PLOTS_MULTI_SEED = True\n",
    "\n",
    "dataset_type = \"mdvsp\"\n",
    "dataset_params = dict(\n",
    "    K=5000,\n",
    "    filename=\"RN-8-3000-03.dat\",\n",
    "    x_min=0,\n",
    "    x_max=25,\n",
    "    noise_std=0.0,\n",
    "    seed=0,\n",
    "    max_trips=2000,\n",
    "    max_succ=50,\n",
    ")\n",
    "\n",
    "Xtmp, _, _ = make_mdvsp_dataset(**dataset_params)\n",
    "in_dim = int(np.asarray(Xtmp).shape[1])\n",
    "print(f\"[MDVSP] inferred in_dim={in_dim} from Xtmp shape={np.asarray(Xtmp).shape}\")\n",
    "\n",
    "train_base = dict(\n",
    "    epochs=1000,\n",
    "    batch_size=8,\n",
    "    val_frac=0.15,\n",
    "    test_frac=0.15,\n",
    "    seed=0,\n",
    "    device=(\"cuda\" if torch.cuda.is_available() else \"cpu\"),\n",
    "    eps=1e-8,\n",
    "    weight_decay=0.0,\n",
    "    plot_every=10,\n",
    "    plot_points=256,\n",
    "    plot_chunk=256,\n",
    ")\n",
    "\n",
    "# ---- DFN (learnable A) ----\n",
    "dfn_params = dict(\n",
    "    input_dim=in_dim, layer_sizes=[16, 128, 16], p_list=[1, 1],\n",
    "    seed=0, alpha=5e-3, beta=-2.0\n",
    ")\n",
    "\n",
    "# ---- DFN (fixed A = I) ----\n",
    "L1 = 5\n",
    "LK = in_dim + 1 - L1\n",
    "assert LK > 0, f\"in_dim too small for DFN_AfixI with L1={L1} (got in_dim={in_dim})\"\n",
    "dfn_Afix_params = dict(\n",
    "    input_dim=in_dim,\n",
    "    layer_sizes=[L1, 400, LK],\n",
    "    p_list=[1, 1],\n",
    "    seed=0,\n",
    "    alpha=5e-3,\n",
    "    beta=-2.0,\n",
    "    A_fixed=np.eye(in_dim, dtype=np.float32),\n",
    ")\n",
    "\n",
    "# ---- other models ----\n",
    "mlp_params  = dict(in_dim=in_dim, hidden_dims=[128, 128], out_dim=1)\n",
    "maff_params = dict(in_dim=in_dim, n_pieces=1600)\n",
    "lset_params = dict(in_dim=in_dim, n_pieces=1600, T=0.05)\n",
    "\n",
    "lr_map = dict(DFN=1e-1, MLP=1e-3, MaxAffine=1e-3, LSET=1e-3)\n",
    "time_limit = 300\n",
    "\n",
    "runs = [\n",
    "    (\"DFN\",        \"DFN\", dfn_params),\n",
    "    (\"DFN_AfixI\",  \"DFN\", dfn_Afix_params),\n",
    "    (\"MLP\",        \"MLP\", mlp_params),\n",
    "    (\"MaxAffine\",  \"MaxAffine\", maff_params),\n",
    "    (\"LSET\",       \"LSET\", lset_params),\n",
    "]\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fff87f19",
   "metadata": {},
   "source": [
    "### Learning (manual seeds + autosave)\n",
    "\n",
    "Train any subset of seeds/models you want. Each finished model is saved to disk immediately.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fed0344f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ---- LEARNING (manual seeds; autosaves each model as it finishes) ----\n",
    "ARTIFACT_ROOT = Path(\"artifacts\")\n",
    "EXPERIMENT_NAME = \"manual\"  # change this name whenever you want\n",
    "\n",
    "# Pick EXACT seeds/models to train (edit these whenever you want):\n",
    "SEEDS_TO_TRAIN = [0]\n",
    "MODELS_TO_TRAIN = None  # e.g. [\"DFN\", \"MLP\"]  (None = train all)\n",
    "\n",
    "# What to save:\n",
    "SAVE_DATA = True   # saves train/val/test tensors (can be big, but fully reproducible)\n",
    "OVERWRITE = False  # if False, won't clobber; makes a timestamped folder instead\n",
    "\n",
    "exp_dir = make_experiment_dir(ARTIFACT_ROOT, dataset_type, EXPERIMENT_NAME)\n",
    "learn_bundle = init_learn_bundle(dataset_type, dataset_params, runs, train_base, lr_map, exp_dir)\n",
    "\n",
    "# TIP: for maximum control, you can comment out the loop and call `train_one_and_save(...)` manually.\n",
    "for seed in SEEDS_TO_TRAIN:\n",
    "    for (name, model_type, model_params) in runs:\n",
    "        if (MODELS_TO_TRAIN is not None) and (name not in MODELS_TO_TRAIN):\n",
    "            continue\n",
    "        learn_bundle = train_one_and_save(\n",
    "            learn_bundle,\n",
    "            seed=seed,\n",
    "            run_name=name,\n",
    "            model_type=model_type,\n",
    "            model_params_base=model_params,\n",
    "            vary_dataset_seed=VARY_DATASET_SEED,\n",
    "            vary_model_init_seed=VARY_MODEL_INIT_SEED,\n",
    "            allow_plots=ALLOW_PLOTS_MULTI_SEED,\n",
    "            save_data=SAVE_DATA,\n",
    "            overwrite=OVERWRITE,\n",
    "        )\n",
    "\n",
    "learn_df, learn_fail_df, learn_summary_df = summarize_learning_bundle(learn_bundle)\n",
    "try:\n",
    "    display(learn_summary_df)\n",
    "except NameError:\n",
    "    pass\n",
    "\n",
    "print(f\"Artifacts saved under: {exp_dir}\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c8eaab98",
   "metadata": {},
   "source": [
    "### Optimization (rerunnable)\n",
    "\n",
    "Edit constraints/objective settings and rerun without retraining.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d3124efd",
   "metadata": {},
   "outputs": [],
   "source": [
    "# feasible integer box + feasible x0 / sum_eq\n",
    "xmin  = np.full(in_dim, int(dataset_params[\"x_min\"]), dtype=int)\n",
    "xmax  = np.full(in_dim, int(dataset_params[\"x_max\"]), dtype=int)\n",
    "\n",
    "rng = np.random.default_rng(train_base[\"seed\"])\n",
    "x0    = rng.integers(xmin, xmax + 1, size=in_dim, dtype=np.int64)\n",
    "delta = 2\n",
    "sum_eq = int(x0.sum())\n",
    "\n",
    "# ---- OPTIMIZATION (rerunnable; NO retraining) ----\n",
    "# If you restarted the kernel, load a previous run like:\n",
    "# learn_bundle = load_experiment_bundle(Path(\"path/to/your/exp_dir\"))\n",
    "\n",
    "opt_df, opt_fail_df, opt_summary_df, gt_df, gap_seed_df = optimize_bundle(\n",
    "    learn_bundle,\n",
    "    x0=x0, xmin=xmin, xmax=xmax,\n",
    "    delta=delta, sum_eq=sum_eq,\n",
    "    strict_ip_check=STRICT_IP_CHECK,\n",
    "    ip_check_tol=IP_CHECK_TOL,\n",
    "    silence_local_search=SILENCE_LOCAL_SEARCH,\n",
    "    time_limit=time_limit,\n",
    ")\n",
    "\n",
    "try:\n",
    "    display(opt_summary_df)\n",
    "except NameError:\n",
    "    pass\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:dfn]",
   "language": "python",
   "name": "conda-env-dfn-py"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
