{
 "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": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(10, 8) (10,) \n",
      " [[284. 307. 453. 571.  20.  86. 494. 570.]\n",
      " [149. 187. 522. 254. 164. 497. 154. 245.]\n",
      " [386. 330.  51.  16. 520. 452. 503. 323.]\n",
      " [491. 198. 272. 473.  74. 182.  74. 272.]\n",
      " [587.  80. 230. 242. 543. 122. 301. 157.]] \n",
      " [2.8892958e+07 2.2397388e+07 2.6657474e+07 2.1041728e+07 2.3339302e+07] \n",
      " 3.6817302e+06 \n",
      "\n",
      "(10, 10) (10,) \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",
      " [ 665.  226.  993. 1351.  851.] \n",
      " 296.35287 \n",
      "\n",
      "(10, 10) (10,) \n",
      " [[-5. -9.  0.  7.  3. -9.  3. -3. -6. -1.]\n",
      " [ 8. 10. -8.  1.  5. -5. -5. -5. -6.  8.]\n",
      " [-6. -6. -8. -8.  6. -4.  6.  2.  8.  1.]\n",
      " [ 6.  7. -9.  1. -1. -4. -1. -2.  0.  7.]\n",
      " [ 7.  3.  4. 10.  3. -3. -9.  1. -6.  2.]] \n",
      " [16987.797  9943.988  8211.554  5990.925 11936.206] \n",
      " 2931.8052 \n",
      "\n"
     ]
    }
   ],
   "source": [
    "# NOTE: MDVSP requires that `filename` exists on disk.\n",
    "X, y, _ = make_mdvsp_dataset(\n",
    "    K=10, filename=\"RN-8-3000-05.dat\", x_min=0, x_max=600, noise_std=0.0, seed=1, max_trips=5000, max_succ=50\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\", y.std(), \"\\n\")\n",
    "\n",
    "X, y, _ = generate_bipartite_subset_matching_dataset(\n",
    "    K=10, num_nodes=10, c_min=1, c_max=1000, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\", y.std(), \"\\n\")\n",
    "\n",
    "X, y, _ = generate_convex_quadratic_dataset(\n",
    "    K=10, dim=10, eigen_min=1.0, eigen_max=20.0, x_min=-10, x_max=10, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\", y.std(), \"\\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\ndef _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\ndef _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()\ndef _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()\ndef _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\ndef _quad_true_on_grid(gt: dict, X: np.ndarray) -> np.ndarray:\n    Q = np.asarray(gt[\"Q\"], dtype=np.float64)\n    x_star = np.asarray(gt[\"x_star\"], dtype=np.float64).reshape(1, -1)\n    D = X.astype(np.float64) - x_star\n    y = np.einsum(\"bi,ij,bj->b\", D, Q, D)\n    return y.astype(np.float32)\n\n\n@torch.no_grad()\ndef _model_pred_on_grid(model: nn.Module, scaler: dict, X: np.ndarray, device: str, *, chunk: int = 4096) -> np.ndarray:\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    Xt = torch.as_tensor(X, dtype=torch.float32, device=device)\n    Xn = (Xt - xm) / xs\n    yN = _predict_in_chunks(model, Xn, chunk=chunk)\n    y = yN * ys + ym\n    return y.detach().cpu().numpy()\n\n\ndef _make_axis_slices_nd(\n    dim: int,\n    xmin: int,\n    xmax: int,\n    *,\n    n_slices: int = 5,\n    vary_dim: int = 0,\n    base_x=None,\n    fixed_dim=None,\n    fixed_vals=None,\n    slice_points=None,\n):\n    \"\"\"Create axis-aligned 1D slices over an integer range for any input dimension.\n\n    Each slice fixes all coordinates except `vary_dim`, which is swept over integers in [xmin, xmax].\n\n    You can specify slices in two ways:\n      1) `slice_points`: a list of length-`dim` integer vectors. Each vector defines the fixed coordinates\n         for one slice (the varying coordinate will be overwritten by the sweep).\n      2) `base_x` (+ optional `fixed_dim` and `fixed_vals`): start from a base integer vector and create\n         multiple slices by changing one fixed coordinate (like the dim=2 case).\n\n    Returns (xs_line, slices) where `slices` is a list of dicts with keys:\n        vary_dim (int), base_x (np.ndarray shape [dim]), X (np.ndarray shape [P, dim])\n    \"\"\"\n    dim = int(dim)\n    xmin = int(xmin); xmax = int(xmax)\n    if xmin > xmax:\n        xmin, xmax = xmax, xmin\n\n    xs_line = np.arange(xmin, xmax + 1, dtype=np.int64)\n\n    if dim <= 0:\n        raise ValueError(\"dim must be >= 1\")\n\n    vary_dim = int(vary_dim) if vary_dim is not None else 0\n    vary_dim = max(0, min(vary_dim, dim - 1))\n\n    # ---- build base points for slices ----\n    points = []\n\n    if slice_points is not None:\n        for p in list(slice_points)[: max(int(n_slices), 1)]:\n            p = np.asarray(p, dtype=np.int64).reshape(-1)\n            if p.size < dim:\n                p = np.pad(p, (0, dim - p.size), mode=\"constant\")\n            elif p.size > dim:\n                p = p[:dim]\n            points.append(p.copy())\n    else:\n        if base_x is None:\n            base = np.zeros(dim, dtype=np.int64)\n        else:\n            base = np.asarray(base_x, dtype=np.int64).reshape(-1)\n            if base.size < dim:\n                base = np.pad(base, (0, dim - base.size), mode=\"constant\")\n            elif base.size > dim:\n                base = base[:dim]\n            base = base.copy()\n\n        if dim == 1:\n            points = [base]\n        else:\n            if fixed_dim is None:\n                fd = 0 if vary_dim != 0 else 1\n            else:\n                fd = int(fixed_dim)\n                if fd < 0 or fd >= dim or fd == vary_dim:\n                    fd = 0 if vary_dim != 0 else 1\n\n            if fixed_vals is None:\n                if int(n_slices) <= 1:\n                    fvals = [int((xmin + xmax) // 2)]\n                else:\n                    fvals = np.linspace(xmin, xmax, int(n_slices))\n                    fvals = [int(round(v)) for v in fvals]\n                    seen = set()\n                    fvals = [v for v in fvals if not (v in seen or seen.add(v))]\n                    if len(fvals) < int(n_slices):\n                        allv = list(range(xmin, xmax + 1))\n                        j = 0\n                        while len(fvals) < int(n_slices) and allv:\n                            v = allv[j % len(allv)]\n                            if v not in seen:\n                                fvals.append(v); seen.add(v)\n                            j += 1\n            else:\n                fvals = [int(v) for v in list(fixed_vals)[: max(int(n_slices), 1)]]\n\n            for fv in fvals[: max(int(n_slices), 1)]:\n                p = base.copy()\n                p[fd] = int(fv)\n                points.append(p)\n\n    slices = []\n    for p in points[: max(int(n_slices), 1)]:\n        X = np.repeat(p.reshape(1, dim), xs_line.size, axis=0).astype(np.float32)\n        X[:, vary_dim] = xs_line.astype(np.float32)\n        slices.append({\"vary_dim\": int(vary_dim), \"base_x\": p.astype(np.int64), \"X\": X})\n\n    return xs_line, slices\n\n\ndef _plot_slices_nd(xs_line: np.ndarray, slices, y_true_list, y_pred_list, *, title: str = \"1D slices\"):\n    \"\"\"Plot multiple 1D slices; each slice overlays true vs predicted.\"\"\"\n\n    def _fmt_fixed(base_x: np.ndarray, vary_dim: int, *, max_items: int = 10) -> str:\n        items = []\n        for j, v in enumerate(base_x.tolist()):\n            if j == vary_dim:\n                continue\n            items.append(f\"x{j+1}={int(v)}\")\n        if not items:\n            return \"no fixed coords\"\n        if len(items) > max_items:\n            items = items[:max_items] + [\"...\"]\n        return \", \".join(items)\n\n    n = len(slices)\n    fig_h = max(2.2 * n, 3.0)\n    fig, axs = plt.subplots(n, 1, figsize=(7.0, fig_h), sharex=True)\n    if n == 1:\n        axs = [axs]\n\n    for i, sl in enumerate(slices):\n        ax = axs[i]\n        yt = y_true_list[i]\n        yp = y_pred_list[i]\n        ax.plot(xs_line, yt, label=\"true\")\n        ax.plot(xs_line, yp, label=\"pred\")\n        vd = int(sl[\"vary_dim\"])\n        base = np.asarray(sl[\"base_x\"], dtype=np.int64).reshape(-1)\n        ax.set_title(f\"vary x{vd+1} (integers), fix: {_fmt_fixed(base, vd)}\")\n        ax.grid(True, alpha=0.25)\n        if i == 0:\n            ax.legend(loc=\"best\")\n\n    vd0 = int(slices[0][\"vary_dim\"])\n    axs[-1].set_xlabel(f\"x{vd0+1}\")\n    fig.suptitle(title, y=0.995)\n    fig.tight_layout()\n    return fig\n\n\ndef 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    \n    # ---- optional 1D slice plots (axis-aligned, any dim; quadratic gt only) ----\n    slice_every = int(tp.get(\"slice_every\", 0) or 0)\n    slice_xmin  = int(tp.get(\"slice_xmin\", dataset_params.get(\"x_min\", -5)))\n    slice_xmax  = int(tp.get(\"slice_xmax\", dataset_params.get(\"x_max\",  5)))\n    slice_chunk = int(tp.get(\"slice_chunk\", plot_chunk))\n    slice_n     = int(tp.get(\"slice_n\", 5))\n    slice_vary_dim = int(tp.get(\"slice_vary_dim\", 0))\n    slice_base_x   = tp.get(\"slice_base_x\", None)     # length dim (ints)\n    slice_fixed_dim= tp.get(\"slice_fixed_dim\", None)  # which fixed coordinate to vary across slices (optional)\n    slice_fixed_vals = tp.get(\"slice_fixed_vals\", None)\n    slice_points   = tp.get(\"slice_points\", None)     # list of base points (optional)\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    # ---- optional 1D slices setup (quadratic gt only) ----\n\n    slice_disp = None\n    slice_xs = None\n    slice_specs = None\n    slice_true_list = None\n    if slice_every > 0 and dt == \"quadratic\" and isinstance(gt, dict) and int(X.shape[1]) >= 1:\n        slice_disp = display(None, display_id=True)  # canvas: 1D slices (true vs pred), updated live\n        dim = int(X.shape[1])\n        slice_xs, slice_specs = _make_axis_slices_nd(\n            dim,\n            slice_xmin,\n            slice_xmax,\n            n_slices=slice_n,\n            vary_dim=slice_vary_dim,\n            base_x=slice_base_x,\n            fixed_dim=slice_fixed_dim,\n            fixed_vals=slice_fixed_vals,\n            slice_points=slice_points,\n        )\n        slice_true_list = [_quad_true_on_grid(gt, sl[\"X\"]) for sl in slice_specs]\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    # show untrained model's 1D slices (epoch 0)\n    if slice_disp is not None:\n        model.eval()\n        y_pred_list = [\n            _model_pred_on_grid(model, scaler, sl[\"X\"], device, chunk=slice_chunk)\n            for sl in slice_specs\n        ]\n        fig = _plot_slices_nd(slice_xs, slice_specs, slice_true_list, y_pred_list, title=\"True vs Model slices @ init\")\n        slice_disp.update(fig)\n        plt.close(fig)\n\n\n\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\n        # update the 1D slice plots live\n        if slice_disp is not None and (ep == 1 or ep % slice_every == 0 or ep == epochs):\n            model.eval()\n            y_pred_list = [\n                _model_pred_on_grid(model, scaler, sl[\"X\"], device, chunk=slice_chunk)\n                for sl in slice_specs\n            ]\n            fig = _plot_slices_nd(slice_xs, slice_specs, slice_true_list, y_pred_list, title=f\"True vs Model slices @ epoch {ep}\")\n            slice_disp.update(fig)\n            plt.close(fig)\n\n    if best_state is not None:\n        model.load_state_dict(best_state)\n\n\n        # show the best model's 1D slices\n        if slice_disp is not None:\n            model.eval()\n            y_pred_list = [\n                _model_pred_on_grid(model, scaler, sl[\"X\"], device, chunk=slice_chunk)\n                for sl in slice_specs\n            ]\n            fig = _plot_slices_nd(slice_xs, slice_specs, slice_true_list, y_pred_list, title=f\"True vs Model slices @ best epoch {best_ep}\")\n            slice_disp.update(fig)\n            plt.close(fig)\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": {},
   "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": {},
   "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"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e88e7a1f",
   "metadata": {},
   "source": [
    "## Tests"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8edf3f97",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "===================== SEED 0 =====================\n",
      "\n",
      "--- Dataset stats (quadratic) ---\n",
      "  X: shape=(2000, 2)  mean(mean)=0.0488  std(mean)=3.19  min=-5  max=5\n",
      "  y: shape=(2000,)  mean=371  std=246  min=0  max=1.07e+03\n",
      "\n",
      "\n",
      "=== Run: quadratic | DFN ===\n",
      "  data: N=2000  train/val/test=1400/300/300  dim=2\n",
      "  model: params=16,477 layers=[16, 128, 16] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=250  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"
    }
   ],
   "source": "# ===================== Quadratic DATASET =====================\nN_SEEDS = 1\n\nVARY_DATASET_SEED = False\nVARY_MODEL_INIT_SEED = True\nSTRICT_IP_CHECK = False\nIP_CHECK_TOL = 1e-4\n\nSILENCE_LOCAL_SEARCH = False\nALLOW_PLOTS_MULTI_SEED = True\n\ndataset_type = \"quadratic\"\ndataset_params = dict(\n    K=2000, dim=2, eigen_min=1, eigen_max=20.0,\n    x_min=-5, x_max=5, noise_std=0.0, seed=0\n)\nin_dim = int(dataset_params[\"dim\"])\n\ntrain_base = dict(\n    epochs=250,\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=1,\n    plot_points=128,\n    plot_chunk=128,\n    slice_every=5,\n    slice_xmin=-10,\n    slice_xmax=10,\n    slice_n=5,\n    slice_vary_dim=0,\n)\n\n\n# ---- DFN (learnable A) ----\ndfn_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) ----\ndfnA_layer_sizes = [2, 400, 1]\nassert dfnA_layer_sizes[0] + dfnA_layer_sizes[-1] == in_dim + 1, \"Need |L1|+|LK| = dim+1 for A_fixed=I\"\n\ndfn_Afix_params = dict(\n    input_dim=in_dim,\n    layer_sizes=dfnA_layer_sizes,\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),  # shape (dim, dim)\n)\n\n# ---- other models ----\nmlp_params  = dict(in_dim=in_dim, hidden_dims=[128, 128], out_dim=1)\nmaff_params = dict(in_dim=in_dim, n_pieces=1600)\nlset_params = dict(in_dim=in_dim, n_pieces=1600, T=0.05)\n\nlr_map = dict(DFN=1e-1, MLP=1e-3, MaxAffine=1e-3, LSET=1e-3)\n\ntime_limit = 300\nx0    = np.array([5,-2,1,-1,7,3,0,2,-3,-1], dtype=int)\nxmin  = np.full(in_dim, -10, dtype=int)\nxmax  = np.full(in_dim,  10, dtype=int)\ndelta = 2\nsum_eq = int(x0.sum())\n\nruns = [\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\n_ = run_benchmark(\n    dataset_type=dataset_type,\n    dataset_params=dataset_params,\n    runs=runs,\n    train_base=train_base,\n    lr_map=lr_map,\n    x0=x0, xmin=xmin, xmax=xmax,\n    delta=delta, sum_eq=sum_eq,\n    n_seeds=N_SEEDS,\n    vary_dataset_seed=VARY_DATASET_SEED,\n    vary_model_init_seed=VARY_MODEL_INIT_SEED,\n    strict_ip_check=STRICT_IP_CHECK,\n    ip_check_tol=IP_CHECK_TOL,\n    silence_local_search=SILENCE_LOCAL_SEARCH,\n    allow_plots_multi_seed=ALLOW_PLOTS_MULTI_SEED,\n    time_limit=time_limit,\n)"
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e1000bc1-6ad1-4e81-a1e5-e8b71262b2ae",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "17338b1f-eb58-4251-adbe-4c6f6da0343d",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9fdc998c-5ecd-4cdd-bef1-44ec947693c6",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}