{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "f5b14808",
   "metadata": {},
   "source": [
    "# DFN"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2efb5921",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "ce8269a4",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "\n",
    "import cppimport\n",
    "\n",
    "import sys, time, math, io, contextlib\n",
    "from pathlib import Path\n",
    "from typing import Optional, List, Tuple, Dict, Any\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linear_sum_assignment\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "from IPython.display import display\n",
    "\n",
    "import gurobipy as gp\n",
    "from gurobipy import GRB\n",
    "\n",
    "_TOL = 1e-9"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "417583e0",
   "metadata": {},
   "source": [
    "## LEMON"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "62767112",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "repo = Path().resolve().parent\n",
    "if str(repo) not in sys.path:\n",
    "    sys.path.insert(0, str(repo))\n",
    "\n",
    "lemon_mcf = cppimport.imp(\"lemon_mcf\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "15e6bd1b",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "88381d28",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'status': 1, 'flow': array([1., 2., 0.]), 'potential': array([-2.,  0., -1.]), 'reduced_cost': array([0., 0., 4.]), 'at_capacity': array([False,  True, False]), 'total_cost': 4.0}\n",
      "{'value': 4.0, 'flow': array([2., 2., 2., 2.])}\n"
     ]
    }
   ],
   "source": [
    "n = 3\n",
    "src    = np.array([0, 0, 1], dtype=np.int64)\n",
    "dst    = np.array([1, 2, 2], dtype=np.int64)\n",
    "cost   = np.array([2.0, 1.0, 3.0], dtype=np.float64)\n",
    "cap    = np.array([5.0, 2.0, 4.0], dtype=np.float64)\n",
    "supply = np.array([3.0, -1.0, -2.0], dtype=np.float64)\n",
    "\n",
    "out_min_cost_flow = lemon_mcf.solve_mcf(n, src, dst, cost, cap, supply)\n",
    "print(out_min_cost_flow)\n",
    "\n",
    "n = 4\n",
    "src = np.array([0,0,1,2], dtype=np.int64)\n",
    "dst = np.array([1,2,3,3], dtype=np.int64)\n",
    "cap = np.array([3.0,2.0,2.0,4.0], dtype=np.float64)\n",
    "out_max_flow = lemon_mcf.max_flow(n, src, dst, cap, 0, 3)\n",
    "print(out_max_flow)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "75235271",
   "metadata": {},
   "source": [
    "## Dataset generators"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "67763cc2",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def make_mdvsp_dataset(\n",
    "    K: int,\n",
    "    filename: str,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    max_trips=None,\n",
    "    max_succ=None,\n",
    "):\n",
    "    \"\"\"Multiple-Depot Vehicle Scheduling (MDVSP) dataset.\n",
    "\n",
    "    Returns:\n",
    "      X: (K, m) integer-ish capacities (float32)\n",
    "      y: (K,) min-cost-flow objective values (float32)\n",
    "      gt: dict containing the fixed network pieces (for later evaluation)\n",
    "    \"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "\n",
    "    with open(filename) as f:\n",
    "        m, n, l = map(int, f.readline().split())\n",
    "        f.readline()  # blank line\n",
    "        trips = np.loadtxt(f, max_rows=n, dtype=np.int64)[:max_trips]\n",
    "        D = np.loadtxt(f, max_rows=l, dtype=np.int64)\n",
    "\n",
    "    p, s, q, e = trips.T\n",
    "    ntr = len(trips)\n",
    "\n",
    "    # node ids\n",
    "    SS = 0\n",
    "    depS = 1 + np.arange(m)\n",
    "    depT = 1 + m + np.arange(m)\n",
    "    trS  = 1 + 2*m + np.arange(ntr)\n",
    "    trT  = 1 + 2*m + ntr + np.arange(ntr)\n",
    "    TT = 1 + 2*m + 2*ntr\n",
    "    N = TT + 1\n",
    "\n",
    "    src, dst, cost, cap = [], [], [], []\n",
    "\n",
    "    # depot boundary arcs (capacity is what we learn/provide per sample)\n",
    "    for d in range(m):\n",
    "        src += [SS, int(depT[d])]\n",
    "        dst += [int(depS[d]), TT]\n",
    "        cost += [0.0, 0.0]\n",
    "        cap  += [0.0, 0.0]\n",
    "    idxSS = np.arange(0, 2*m, 2)  # arcs SS->depS\n",
    "    idxTT = np.arange(1, 2*m, 2)  # arcs depT->TT\n",
    "\n",
    "    # trip arcs (always cap=1)\n",
    "    src += trS.tolist()\n",
    "    dst += trT.tolist()\n",
    "    cost += [0.0] * ntr\n",
    "    cap  += [1.0] * ntr\n",
    "\n",
    "    # depot-to-trip and trip-to-depot arcs\n",
    "    for d in range(m):\n",
    "        # depS -> trS\n",
    "        src += [int(depS[d])] * ntr\n",
    "        dst += trS.tolist()\n",
    "        cost += (5000 + 10 * D[d, p]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "        # trT -> depT\n",
    "        src += trT.tolist()\n",
    "        dst += [int(depT[d])] * ntr\n",
    "        cost += (5000 + 10 * D[q, d]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "    # feasible trip successor arcs (trT -> next trS)\n",
    "    order = np.argsort(s)\n",
    "    p2, s2 = p[order], s[order]\n",
    "    max_succ_eff = ntr if max_succ is None else int(max_succ)\n",
    "\n",
    "    for i in range(ntr):\n",
    "        travel = D[q[i], p2]                          # time from trip i end depot -> next trip start depot\n",
    "        feas = np.flatnonzero(s2 >= e[i] + travel)[:max_succ_eff]\n",
    "        j = order[feas]\n",
    "        if j.size:\n",
    "            src += [int(trT[i])] * j.size\n",
    "            dst += trS[j].tolist()\n",
    "            cost += (8 * travel[feas] + 2 * (s[j] - e[i])).astype(float).tolist()\n",
    "            cap  += [1.0] * j.size\n",
    "\n",
    "    src = np.asarray(src, dtype=np.int64)\n",
    "    dst = np.asarray(dst, dtype=np.int64)\n",
    "    cost = np.asarray(cost, dtype=np.float64)\n",
    "    cap0 = np.asarray(cap, dtype=np.float64)\n",
    "\n",
    "    # sample capacities X and compute y via max-flow + min-cost-flow\n",
    "    X = rng.integers(x_min, np.asarray(x_max) + 1, size=(K, m)).astype(np.float64)\n",
    "    y = np.empty(K, dtype=np.float64)\n",
    "\n",
    "    for k in range(K):\n",
    "        cap_k = cap0.copy()\n",
    "        cap_k[idxSS] = X[k]\n",
    "        cap_k[idxTT] = X[k]\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap_k, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, dtype=np.float64)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "\n",
    "        y[k] = lemon_mcf.solve_mcf(N, src, dst, cost, cap_k, supply)[\"total_cost\"]\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = dict(\n",
    "        type=\"mdvsp\", N=int(N), SS=int(SS), TT=int(TT),\n",
    "        src=src, dst=dst, cost=cost, cap0=cap0, idxSS=idxSS, idxTT=idxTT\n",
    "    )\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n",
    "\n",
    "\n",
    "def generate_bipartite_subset_matching_dataset(\n",
    "    K: int, num_nodes: int, c_min: int, c_max: int, noise_std: float = 0.0, seed: int = 0\n",
    "):\n",
    "    \"\"\"Assignment-style dataset: choose a subset of left nodes, match to right nodes with min cost.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    C = rng.integers(c_min, c_max + 1, size=(num_nodes, num_nodes)).astype(np.float32)\n",
    "\n",
    "    X = np.zeros((K, num_nodes), dtype=np.float32)\n",
    "    y = np.zeros((K,), dtype=np.float32)\n",
    "\n",
    "    for k in range(K):\n",
    "        mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        while not mask.any():\n",
    "            mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        idx = np.flatnonzero(mask)\n",
    "        X[k, idx] = 1.0\n",
    "\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        y[k] = C[idx, :][r, c].sum()\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = {\"type\": \"assignment\", \"C\": C.astype(np.float32)}\n",
    "    return X, y, gt\n",
    "\n",
    "\n",
    "def generate_convex_quadratic_dataset(\n",
    "    K: int,\n",
    "    dim: int,\n",
    "    eigen_min: float,\n",
    "    eigen_max: float,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    x_star_zero: bool = False,\n",
    "):\n",
    "    \"\"\"y = (x - x*)^T Q (x - x*) + noise, with Q symmetric PSD.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    U, R = np.linalg.qr(rng.standard_normal((dim, dim)))\n",
    "    U *= np.sign(np.diag(R) + 1e-12)\n",
    "    Q = U @ np.diag(rng.uniform(eigen_min, eigen_max, dim)) @ U.T\n",
    "\n",
    "    x_star = np.zeros(dim, dtype=np.int64) if x_star_zero else rng.integers(x_min, x_max + 1, size=dim)\n",
    "\n",
    "    X = rng.integers(x_min, x_max + 1, size=(K, dim)).astype(np.float32)\n",
    "    d = X - x_star\n",
    "    y = np.einsum(\"bi,ij,bj->b\", d, Q, d) + noise_std * rng.normal(size=K)\n",
    "\n",
    "    gt = {\"type\": \"quadratic\", \"Q\": Q.astype(np.float32), \"x_star\": x_star.astype(np.int64)}\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1dfc62e4",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "89201b4a",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(50, 8) (50,) \n",
      " [[ 5.  5.  8. 10.  0.  1.  9. 10.]\n",
      " [ 2.  3.  9.  4.  3.  9.  2.  4.]\n",
      " [ 7.  6.  0.  0.  9.  8.  9.  5.]\n",
      " [ 8.  3.  4.  8.  1.  3.  1.  4.]\n",
      " [10.  1.  4.  4.  9.  2.  5.  2.]] \n",
      " [490251.12 366100.16 447590.56 326038.94 376371.84] \n",
      "\n",
      "(50, 10) (50,) \n",
      " [[0. 1. 1. 0. 0. 0. 0. 1. 0. 1.]\n",
      " [1. 0. 0. 0. 0. 0. 1. 1. 0. 0.]\n",
      " [0. 1. 0. 0. 1. 0. 1. 0. 1. 1.]\n",
      " [1. 1. 0. 0. 1. 1. 1. 1. 0. 1.]\n",
      " [0. 1. 0. 1. 1. 0. 0. 0. 1. 0.]] \n",
      " [ 9.  4. 13. 17. 11.] \n",
      "\n",
      "(50, 10) (50,) \n",
      " [[-47. -87.  -1.  69.  25. -87.  30. -31. -55. -14.]\n",
      " [ 75.  94. -72.  13.  53. -48. -46. -52. -58.  78.]\n",
      " [-57. -55. -75. -75.  56. -43.  61.  17.  72.  11.]\n",
      " [ 53.  62. -88.  12.  -9. -43. -10. -18.  -2.  64.]\n",
      " [ 65.  25.  42.  92.  28. -26. -84.  11. -54.  19.]] \n",
      " [1.5913905e+06 9.7035825e+05 7.8526031e+05 5.9535256e+05 1.0773938e+06] \n",
      "\n"
     ]
    }
   ],
   "source": [
    "# NOTE: MDVSP requires that `filename` exists on disk.\n",
    "X, y, _ = make_mdvsp_dataset(\n",
    "    K=50, filename=\"RN-8-3000-03.dat\", x_min=0, x_max=10, noise_std=1.0, seed=1, max_trips=200, max_succ=5\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")\n",
    "\n",
    "X, y, _ = generate_bipartite_subset_matching_dataset(\n",
    "    K=50, num_nodes=10, c_min=1, c_max=10, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")\n",
    "\n",
    "X, y, _ = generate_convex_quadratic_dataset(\n",
    "    K=50, dim=10, eigen_min=1.0, eigen_max=20.0, x_min=-100, x_max=100, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "70071f5f",
   "metadata": {},
   "source": [
    "## Models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "07278db8",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def _ste_round(x: torch.Tensor) -> torch.Tensor:\n",
    "    return x + (torch.round(x) - x).detach()\n",
    "\n",
    "class _MCFValue(torch.autograd.Function):\n",
    "    \n",
    "    @staticmethod\n",
    "    def forward(ctx, n_nodes, src, dst, cost, cap, supply):\n",
    "        n = int(n_nodes)\n",
    "\n",
    "        src = src.to(dtype=torch.int64).contiguous()\n",
    "        dst = dst.to(dtype=torch.int64).contiguous()\n",
    "\n",
    "        m = int(src.numel())\n",
    "        if (dst.numel() != m) or (cost.numel() != m) or (cap.numel() != m) or (supply.numel() != n):\n",
    "            raise ValueError(\"Bad shapes for MCF inputs.\")\n",
    "        if torch.abs(supply.double().sum()) > _TOL:\n",
    "            raise ValueError(\"Require sum(supply)=0\")\n",
    "\n",
    "        def as_np(t: torch.Tensor, dtype):\n",
    "            return t.detach().cpu().contiguous().view(-1).numpy().astype(dtype, copy=False)\n",
    "\n",
    "        out = lemon_mcf.solve_mcf(\n",
    "            n,\n",
    "            as_np(src, np.int64),\n",
    "            as_np(dst, np.int64),\n",
    "            as_np(cost, np.float64),\n",
    "            as_np(cap, np.float64),\n",
    "            as_np(supply, np.float64),\n",
    "            tol=_TOL,\n",
    "        )\n",
    "        if out[\"status\"] != 1:\n",
    "            raise RuntimeError(f\"LEMON failed (status={out['status']})\")\n",
    "\n",
    "        flow = out[\"flow\"]\n",
    "        pot = out[\"potential\"]\n",
    "        red = out[\"reduced_cost\"]\n",
    "        at = out.get(\"at_cap\", out.get(\"at_capacity\", None))\n",
    "        if at is None:\n",
    "            at = np.abs(flow - as_np(cap, np.float64)) <= _TOL\n",
    "\n",
    "        ctx.flow, ctx.pot, ctx.red, ctx.at = flow, pot, red, at\n",
    "        return cost.new_tensor(float(out[\"total_cost\"]))\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, g):\n",
    "        dev, dt = g.device, g.dtype\n",
    "        flow = torch.as_tensor(ctx.flow, device=dev, dtype=dt)\n",
    "        pot  = torch.as_tensor(ctx.pot,  device=dev, dtype=dt)\n",
    "        red  = torch.as_tensor(ctx.red,  device=dev, dtype=dt)\n",
    "        at   = torch.as_tensor(ctx.at,   device=dev, dtype=torch.bool)\n",
    "\n",
    "        grad_cost = flow\n",
    "        grad_cap  = torch.where(at, red, torch.zeros_like(red))\n",
    "        grad_sup  = pot.mean() - pot\n",
    "\n",
    "        return None, None, None, grad_cost * g, grad_cap * g, grad_sup * g\n",
    "\n",
    "\n",
    "class DFN(nn.Module):\n",
    "    def __init__(\n",
    "        self,\n",
    "        input_dim: int,\n",
    "        layer_sizes,\n",
    "        p_list,\n",
    "        big_cost: float = 1e6,\n",
    "        big_cap: float = 1e6,\n",
    "        seed: int = 0,\n",
    "        A_fixed=None,\n",
    "        alpha: float = 1e-6,\n",
    "        beta: float = -0.0,\n",
    "    ):\n",
    "        super().__init__()\n",
    "        self.alpha = float(alpha)\n",
    "        self.beta  = float(beta)\n",
    "\n",
    "        layer_sizes = list(map(int, layer_sizes))\n",
    "        if len(layer_sizes) < 2 or len(p_list) != len(layer_sizes) - 1:\n",
    "            raise ValueError(\"Need len(layer_sizes)>=2 and len(p_list)=len(layer_sizes)-1\")\n",
    "\n",
    "        self.n = int(sum(layer_sizes))\n",
    "        if self.n <= 0:\n",
    "            raise ValueError(\"sum(layer_sizes) must be > 0\")\n",
    "\n",
    "        # node indices per layer\n",
    "        layers, off = [], 0\n",
    "        for s in layer_sizes:\n",
    "            layers.append(torch.arange(off, off + s, dtype=torch.long))\n",
    "            off += s\n",
    "\n",
    "        L1, LK = layers[0], layers[-1]\n",
    "        if L1.numel() == 0 or LK.numel() == 0:\n",
    "            raise ValueError(\"First/last layer must be non-empty.\")\n",
    "\n",
    "        self.fix_node = int(LK[-1].item())\n",
    "        boundary = torch.cat([L1, LK[:-1]], 0)\n",
    "        self.register_buffer(\"boundary\", boundary)\n",
    "\n",
    "        gen = torch.Generator().manual_seed(int(seed))\n",
    "\n",
    "        def bipartite(U: torch.Tensor, V: torch.Tensor):\n",
    "            su, sv = int(U.numel()), int(V.numel())\n",
    "            return U.repeat_interleave(sv), V.repeat(su)\n",
    "\n",
    "        def sample_edges(U, V, p: float):\n",
    "            s, t = bipartite(U, V)\n",
    "            if p < 1.0:\n",
    "                keep = torch.rand(s.numel(), generator=gen) < float(p)\n",
    "                s, t = s[keep], t[keep]\n",
    "            return s, t\n",
    "\n",
    "        # learnable arcs between consecutive layers (both directions)\n",
    "        sf, tf, sb, tb = [], [], [], []\n",
    "        for i, p in enumerate(map(float, p_list)):\n",
    "            if not (0.0 <= p <= 1.0):\n",
    "                raise ValueError(\"p_list entries must be in [0,1]\")\n",
    "\n",
    "            s, t = sample_edges(layers[i], layers[i + 1], p)  # forward\n",
    "            sf.append(s); tf.append(t)\n",
    "\n",
    "            s, t = sample_edges(layers[i + 1], layers[i], p)  # backward\n",
    "            sb.append(s); tb.append(t)\n",
    "\n",
    "        src_param = torch.cat([torch.cat(sf, 0), torch.cat(sb, 0)], 0)\n",
    "        dst_param = torch.cat([torch.cat(tf, 0), torch.cat(tb, 0)], 0)\n",
    "        if src_param.numel() == 0:\n",
    "            raise ValueError(\"No learnable arcs (increase p_list / layer sizes).\")\n",
    "\n",
    "        s1, t1 = bipartite(L1, LK)\n",
    "        s2, t2 = bipartite(LK, L1)\n",
    "        src_fixed = torch.cat([s1, s2], 0)\n",
    "        dst_fixed = torch.cat([t1, t2], 0)\n",
    "        m_fixed = int(src_fixed.numel())\n",
    "\n",
    "        self.register_buffer(\"src\", torch.cat([src_param, src_fixed], 0))\n",
    "        self.register_buffer(\"dst\", torch.cat([dst_param, dst_fixed], 0))\n",
    "        self.register_buffer(\"cap_fixed\",  torch.full((m_fixed,), float(big_cap),  dtype=torch.float32))\n",
    "        self.register_buffer(\"cost_fixed\", torch.full((m_fixed,), float(big_cost), dtype=torch.float32))\n",
    "\n",
    "        nb = int(boundary.numel())\n",
    "        input_dim = int(input_dim)\n",
    "\n",
    "        m_param = int(src_param.numel())\n",
    "        self.cap_raw  = nn.Parameter(torch.zeros(m_param) + 0.542)\n",
    "        self.cost_raw = nn.Parameter(torch.randn(m_param) + 1.0)\n",
    "        self.b_raw    = nn.Parameter(torch.zeros(nb))\n",
    "\n",
    "        if A_fixed is None:\n",
    "            A = torch.zeros(nb, input_dim)\n",
    "            rows = torch.arange(nb)\n",
    "            A[rows, rows % input_dim] = 1.0\n",
    "            self.A = nn.Parameter(A)\n",
    "        else:\n",
    "            A_fixed = torch.as_tensor(A_fixed, dtype=torch.float32)\n",
    "            if A_fixed.shape != (nb, input_dim):\n",
    "                raise ValueError(f\"A_fixed must have shape {(nb, input_dim)}, got {tuple(A_fixed.shape)}\")\n",
    "            self.register_buffer(\"A\", A_fixed)\n",
    "\n",
    "    def forward(self, w: torch.Tensor) -> torch.Tensor:\n",
    "        capP  = _ste_round(F.softplus(self.cap_raw))\n",
    "        costP = self.cost_raw\n",
    "        b     = _ste_round(self.b_raw)\n",
    "        A     = _ste_round(self.A) if isinstance(self.A, nn.Parameter) else self.A\n",
    "\n",
    "        cap  = torch.cat([capP,  self.cap_fixed.to(w.device, w.dtype)], 0)\n",
    "        cost = torch.cat([costP, self.cost_fixed.to(w.device, w.dtype)], 0)\n",
    "\n",
    "        def one(w1: torch.Tensor) -> torch.Tensor:\n",
    "            supply = torch.zeros(self.n, device=w1.device, dtype=torch.float64)\n",
    "            supply[self.boundary] = (A.double() @ w1.double()) + b.double()\n",
    "            supply[self.fix_node] = -supply.sum()\n",
    "            return _MCFValue.apply(self.n, self.src, self.dst, cost, cap, supply)\n",
    "\n",
    "        out = one(w) if w.dim() == 1 else torch.stack([one(wi) for wi in w], 0)\n",
    "        return self.alpha * out + self.beta\n",
    "\n",
    "\n",
    "class MLP(nn.Module):\n",
    "    def __init__(self, in_dim, hidden_dims, out_dim):\n",
    "        super().__init__()\n",
    "        dims = [in_dim] + list(hidden_dims) + [out_dim]\n",
    "        layers = []\n",
    "        for a, b in zip(dims[:-2], dims[1:-1]):\n",
    "            layers += [nn.Linear(a, b), nn.ReLU()]\n",
    "        layers += [nn.Linear(dims[-2], dims[-1])]\n",
    "        self.net = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "\n",
    "class MaxAffine(nn.Module):\n",
    "    def __init__(self, in_dim: int, n_pieces: int):\n",
    "        super().__init__()\n",
    "        self.W = nn.Parameter(torch.randn(n_pieces, in_dim) / (in_dim**0.5))\n",
    "        self.b = nn.Parameter(torch.zeros(n_pieces))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        return (x @ self.W.T + self.b).max(dim=1).values\n",
    "\n",
    "\n",
    "class LSET(nn.Module):\n",
    "    def __init__(self, in_dim: int, n_pieces: int, T: float = 0.01):\n",
    "        super().__init__()\n",
    "        self.T = float(T)\n",
    "        if self.T == 0.0:\n",
    "            raise ValueError(\"T must be nonzero\")\n",
    "        self.A = nn.Parameter(torch.randn(n_pieces, in_dim) / (in_dim**0.5))\n",
    "        self.b = nn.Parameter(torch.zeros(n_pieces))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        z = (x @ self.A.t() + self.b) / self.T\n",
    "        return self.T * torch.logsumexp(z, dim=-1)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7cca0120",
   "metadata": {},
   "source": [
    "## Training Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0a7db872",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def _split_train_val_test(X: torch.Tensor, y: torch.Tensor, *, val_frac: float, test_frac: float, seed: int):\n",
    "    N = int(X.shape[0])\n",
    "    n_test = int(round(test_frac * N))\n",
    "    n_val  = int(round(val_frac  * N))\n",
    "    n_train = N - n_val - n_test\n",
    "    if n_train <= 0:\n",
    "        raise ValueError(\"splits too large; train set would be empty\")\n",
    "\n",
    "    g = torch.Generator().manual_seed(int(seed))\n",
    "    perm = torch.randperm(N, generator=g)\n",
    "    i_tr = perm[:n_train]\n",
    "    i_va = perm[n_train:n_train + n_val]\n",
    "    i_te = perm[n_train + n_val:]\n",
    "\n",
    "    return (X[i_tr], y[i_tr]), (X[i_va], y[i_va]), (X[i_te], y[i_te])\n",
    "\n",
    "\n",
    "def _fit_standardizer(Xtr: torch.Tensor, ytr: torch.Tensor, *, eps: float):\n",
    "    x_mean = Xtr.mean(0, keepdim=True)\n",
    "    x_std  = Xtr.std(0, unbiased=False, keepdim=True)\n",
    "    x_std  = torch.where(x_std < eps, torch.ones_like(x_std), x_std)\n",
    "\n",
    "    y_mean = ytr.mean()\n",
    "    y_std  = ytr.std(unbiased=False).clamp_min(eps)\n",
    "\n",
    "    return {\"x_mean\": x_mean, \"x_std\": x_std, \"y_mean\": y_mean, \"y_std\": y_std}\n",
    "\n",
    "\n",
    "def _apply_standardizer(X: torch.Tensor, y: torch.Tensor, scaler):\n",
    "    Xn = (X - scaler[\"x_mean\"]) / scaler[\"x_std\"]\n",
    "    yn = (y - scaler[\"y_mean\"]) / scaler[\"y_std\"]\n",
    "    return Xn, yn\n",
    "\n",
    "@torch.no_grad()\n",
    "def _mse_norm(model: nn.Module, loader: DataLoader, device: str):\n",
    "    model.eval()\n",
    "    tot, n = 0.0, 0\n",
    "    for xb, yb in loader:\n",
    "        xb, yb = xb.to(device), yb.to(device)\n",
    "        pred = model(xb).squeeze(-1)\n",
    "        tot += F.mse_loss(pred, yb, reduction=\"sum\").item()\n",
    "        n += yb.numel()\n",
    "    return tot / max(n, 1)\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def _predict_in_chunks(model: nn.Module, X: torch.Tensor, *, chunk: int):\n",
    "    out = []\n",
    "    for i in range(0, int(X.shape[0]), int(chunk)):\n",
    "        out.append(model(X[i:i + chunk]).squeeze(-1))\n",
    "    return torch.cat(out, 0)\n",
    "\n",
    "\n",
    "def generate_and_train_simple(dataset_type, dataset_params, model_type, model_params, train_params=None):\n",
    "    \n",
    "    tp = train_params or {}\n",
    "\n",
    "    epochs     = int(tp.get(\"epochs\", 200))\n",
    "    batch_sz   = int(tp.get(\"batch_size\", 32))\n",
    "    lr         = float(tp.get(\"lr\", 1e-3))\n",
    "    wd         = float(tp.get(\"weight_decay\", 0.0))\n",
    "    val_frac   = float(tp.get(\"val_frac\", 0.15))\n",
    "    test_frac  = float(tp.get(\"test_frac\", 0.15))\n",
    "    eps        = float(tp.get(\"eps\", 1e-8))\n",
    "    seed       = int(tp.get(\"seed\", 0))\n",
    "    device     = tp.get(\"device\", \"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "    plot_every = int(tp.get(\"plot_every\", 0) or 0)\n",
    "    plot_points= int(tp.get(\"plot_points\", 2048))\n",
    "    plot_chunk = int(tp.get(\"plot_chunk\", 4096))\n",
    "    print_stats= bool(tp.get(\"print_stats\", True))\n",
    "\n",
    "    # ---- data ----\n",
    "    dt = str(dataset_type).lower()\n",
    "    if dt == \"mdvsp\":\n",
    "        X, y, gt = make_mdvsp_dataset(**dataset_params)\n",
    "    elif dt == \"assignment\":\n",
    "        X, y, gt = generate_bipartite_subset_matching_dataset(**dataset_params)\n",
    "    elif dt == \"quadratic\":\n",
    "        X, y, gt = generate_convex_quadratic_dataset(**dataset_params)\n",
    "    else:\n",
    "        raise ValueError(\"dataset_type must be: mdvsp | assignment | quadratic\")\n",
    "\n",
    "    X = torch.as_tensor(X, dtype=torch.float32)\n",
    "    y = torch.as_tensor(y, dtype=torch.float32).view(-1)\n",
    "\n",
    "    # ---- print a quick dataset stats ----\n",
    "    if print_stats:\n",
    "        with torch.no_grad():\n",
    "            xmn = X.mean(0)\n",
    "            xsd = X.std(0, unbiased=False)\n",
    "            print(\n",
    "                f\"\\n--- Dataset stats ({dataset_type}) ---\\n\"\n",
    "                f\"  X: shape={tuple(X.shape)}  mean(mean)={xmn.mean():.3g}  std(mean)={xsd.mean():.3g}  \"\n",
    "                f\"min={float(X.min()):.3g}  max={float(X.max()):.3g}\\n\"\n",
    "                f\"  y: shape={tuple(y.shape)}  mean={float(y.mean()):.3g}  std={float(y.std(unbiased=False)):.3g}  \"\n",
    "                f\"min={float(y.min()):.3g}  max={float(y.max()):.3g}\\n\"\n",
    "            )\n",
    "\n",
    "    (Xtr, ytr), (Xva, yva), (Xte, yte) = _split_train_val_test(X, y, val_frac=val_frac, test_frac=test_frac, seed=seed)\n",
    "\n",
    "    scaler = _fit_standardizer(Xtr, ytr, eps=eps)\n",
    "    XtrN, ytrN = _apply_standardizer(Xtr, ytr, scaler)\n",
    "    XvaN, yvaN = _apply_standardizer(Xva, yva, scaler)\n",
    "    XteN, yteN = _apply_standardizer(Xte, yte, scaler)\n",
    "\n",
    "    train_loader = DataLoader(TensorDataset(XtrN, ytrN), batch_size=batch_sz, shuffle=True)\n",
    "    val_loader   = DataLoader(TensorDataset(XvaN, yvaN), batch_size=batch_sz, shuffle=False)\n",
    "\n",
    "    # subset for plotting\n",
    "    Nv = int(XvaN.shape[0])\n",
    "    if plot_points <= 0 or plot_points >= Nv:\n",
    "        plot_idx = torch.arange(Nv)\n",
    "    else:\n",
    "        g_plot = torch.Generator().manual_seed(seed + 12345)\n",
    "        plot_idx = torch.randperm(Nv, generator=g_plot)[:plot_points]\n",
    "\n",
    "    # ---- model ----\n",
    "    mt = str(model_type)\n",
    "    mp = dict(model_params)\n",
    "\n",
    "    if mt == \"DFN\":\n",
    "        model = DFN(**mp)\n",
    "        extra = f\"layers={mp.get('layer_sizes')} p_list={mp.get('p_list')} alpha={getattr(model,'alpha',None)} beta={getattr(model,'beta',None)}\"\n",
    "    else:\n",
    "        mp.pop(\"alpha\", None)\n",
    "        mp.pop(\"beta\", None)\n",
    "        if mt == \"MLP\":\n",
    "            model = MLP(**mp)\n",
    "            extra = f\"hidden={mp.get('hidden_dims')}\"\n",
    "        elif mt == \"MaxAffine\":\n",
    "            model = MaxAffine(**mp)\n",
    "            extra = f\"n_pieces={mp.get('n_pieces')}\"\n",
    "        elif mt == \"LSET\":\n",
    "            model = LSET(**mp)\n",
    "            extra = f\"n_pieces={mp.get('n_pieces')} T={mp.get('T')}\"\n",
    "        else:\n",
    "            raise ValueError(\"model_type must be: DFN | MLP | MaxAffine | LSET\")\n",
    "\n",
    "    model = model.to(device)\n",
    "    opt = torch.optim.Adam(model.parameters(), lr=lr, weight_decay=wd)\n",
    "\n",
    "    n_params = sum(p.numel() for p in model.parameters())\n",
    "    n_trainable = sum(p.numel() for p in model.parameters() if p.requires_grad)\n",
    "    print(\n",
    "        f\"\\n=== Run: {dataset_type} | {model_type} ===\\n\"\n",
    "        f\"  data: N={len(X)}  train/val/test={len(Xtr)}/{len(Xva)}/{len(Xte)}  dim={X.shape[1]}\\n\"\n",
    "        f\"  model: params={n_params:,} {extra}\\n\"\n",
    "        f\"  train: device={device}  epochs={epochs}  batch={batch_sz}  lr={lr:g}  wd={wd:g}  seed={seed}\\n\"\n",
    "    )\n",
    "\n",
    "    history = {\"train_mse_norm\": [], \"val_mse_norm\": []}\n",
    "    best_val, best_ep, best_state = float(\"inf\"), 0, None\n",
    "\n",
    "    live = display(None, display_id=True) if plot_every > 0 else None\n",
    "\n",
    "    for ep in range(1, epochs + 1):\n",
    "        model.train()\n",
    "        for xb, yb in train_loader:\n",
    "            xb, yb = xb.to(device), yb.to(device)\n",
    "            loss = F.mse_loss(model(xb).squeeze(-1), yb)\n",
    "            opt.zero_grad(set_to_none=True)\n",
    "            loss.backward()\n",
    "            opt.step()\n",
    "\n",
    "        tr_mse = _mse_norm(model, train_loader, device)\n",
    "        va_mse = _mse_norm(model, val_loader, device)\n",
    "        history[\"train_mse_norm\"].append(tr_mse)\n",
    "        history[\"val_mse_norm\"].append(va_mse)\n",
    "\n",
    "        if va_mse < best_val:\n",
    "            best_val, best_ep = va_mse, ep\n",
    "            best_state = {k: v.detach().cpu().clone() for k, v in model.state_dict().items()}\n",
    "\n",
    "        if plot_every > 0 and (ep == 1 or ep % plot_every == 0 or ep == epochs):\n",
    "            model.eval()\n",
    "            x_plot = XvaN[plot_idx].to(device)\n",
    "            y_true = yvaN[plot_idx].to(device)\n",
    "            y_pred = _predict_in_chunks(model, x_plot, chunk=plot_chunk)\n",
    "\n",
    "            fig, ax = plt.subplots(1, 2, figsize=(10, 4))\n",
    "            ax[0].plot(history[\"train_mse_norm\"], label=\"train\")\n",
    "            ax[0].plot(history[\"val_mse_norm\"], label=\"val\")\n",
    "            ax[0].set_yscale(\"log\")\n",
    "            ax[0].set_title(f\"Epoch {ep}/{epochs} | val MSE={va_mse:.3e} (norm)\")\n",
    "            ax[0].legend()\n",
    "\n",
    "            yt = y_true.detach().cpu().numpy()\n",
    "            yp = y_pred.detach().cpu().numpy()\n",
    "            ax[1].scatter(yt, yp, s=10)\n",
    "            lo = float(min(yt.min(), yp.min()))\n",
    "            hi = float(max(yt.max(), yp.max()))\n",
    "            ax[1].plot([lo, hi], [lo, hi])\n",
    "            ax[1].set_xlabel(\"y_true (norm)\")\n",
    "            ax[1].set_ylabel(\"y_pred (norm)\")\n",
    "            ax[1].set_title(f\"Val scatter (n={len(yt)})\")\n",
    "\n",
    "            plt.tight_layout()\n",
    "            live.update(fig)\n",
    "            plt.close(fig)\n",
    "\n",
    "    if best_state is not None:\n",
    "        model.load_state_dict(best_state)\n",
    "\n",
    "    if print_stats:\n",
    "        print(f\"[DONE] best val MSE (norm) = {best_val:.3e} @ epoch {best_ep}\\n\")\n",
    "\n",
    "    data = {\n",
    "        \"raw\":  {\"Xtr\": Xtr,  \"ytr\": ytr,  \"Xva\": Xva,  \"yva\": yva,  \"Xte\": Xte,  \"yte\": yte},\n",
    "        \"norm\": {\"Xtr\": XtrN, \"ytr\": ytrN, \"Xva\": XvaN, \"yva\": yvaN, \"Xte\": XteN, \"yte\": yteN},\n",
    "        \"scaler\": scaler,\n",
    "        \"best_val_mse_norm\": float(best_val),\n",
    "        \"best_epoch\": int(best_ep),\n",
    "        \"stats\": {\"n_params\": int(n_params), \"n_trainable\": int(n_trainable)},\n",
    "        \"true\": gt,\n",
    "        \"device\": device,\n",
    "    }\n",
    "\n",
    "    spec = {\n",
    "        \"model\": model_type,\n",
    "        \"extra\": extra,\n",
    "        \"n_params\": int(n_params),\n",
    "        \"n_trainable\": int(n_trainable),\n",
    "        \"model_params\": dict(model_params),\n",
    "        \"train_params\": {\n",
    "            \"epochs\": int(epochs),\n",
    "            \"batch_size\": int(batch_sz),\n",
    "            \"lr\": float(lr),\n",
    "            \"weight_decay\": float(wd),\n",
    "            \"seed\": int(seed),\n",
    "            \"device\": str(device),\n",
    "            \"val_frac\": float(val_frac),\n",
    "            \"test_frac\": float(test_frac),\n",
    "        },\n",
    "    }\n",
    "\n",
    "    return model, data, history, spec\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ace7f63f",
   "metadata": {},
   "source": [
    "## Optimization Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "38416530",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def eval_true_obj(gt, x):\n",
    "    if not isinstance(gt, dict):\n",
    "        return np.nan\n",
    "\n",
    "    x = np.asarray(x).reshape(-1)\n",
    "    t = gt.get(\"type\", None)\n",
    "\n",
    "    if t == \"quadratic\":\n",
    "        Q = np.asarray(gt[\"Q\"], float)\n",
    "        xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "        d = x.astype(float) - xs\n",
    "        return float(d @ Q @ d)\n",
    "\n",
    "    if t == \"assignment\":\n",
    "        C = np.asarray(gt[\"C\"], float)\n",
    "        idx = np.flatnonzero(x > 0.5)\n",
    "        if idx.size == 0:\n",
    "            return 0.0\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        return float(C[idx, :][r, c].sum())\n",
    "\n",
    "    if t == \"mdvsp\":\n",
    "        N, SS, TT = int(gt[\"N\"]), int(gt[\"SS\"]), int(gt[\"TT\"])\n",
    "        src = np.asarray(gt[\"src\"], np.int64)\n",
    "        dst = np.asarray(gt[\"dst\"], np.int64)\n",
    "        cost = np.asarray(gt[\"cost\"], float)\n",
    "        cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "        idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "        idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "        cap = cap0.copy()\n",
    "        cap[idxSS] = x.astype(float)\n",
    "        cap[idxTT] = x.astype(float)\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, float)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "        return float(lemon_mcf.solve_mcf(N, src, dst, cost, cap, supply)[\"total_cost\"])\n",
    "\n",
    "    return np.nan\n",
    "\n",
    "\n",
    "def local_search_l1_int(\n",
    "    f,\n",
    "    x0,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    delta: int,\n",
    "    sum_eq=None,\n",
    "    max_iters: int = 10_000,\n",
    "    print_every: int = 1,\n",
    "):\n",
    "    x  = np.asarray(x0,    int).ravel()\n",
    "    lo = np.asarray(x_min, int).ravel()\n",
    "    hi = np.asarray(x_max, int).ravel()\n",
    "    n = int(x.size)\n",
    "    delta = int(delta)\n",
    "\n",
    "    assert lo.size == n and hi.size == n\n",
    "    assert np.all(x >= lo) and np.all(x <= hi)\n",
    "    if sum_eq is not None:\n",
    "        sum_eq = int(sum_eq)\n",
    "        assert int(x.sum()) == sum_eq\n",
    "\n",
    "    def eval_batch(X):\n",
    "        y = np.asarray(f(X), float).reshape(-1)\n",
    "        return y\n",
    "\n",
    "    def ok(z):\n",
    "        if np.any(z < lo) or np.any(z > hi):\n",
    "            return False\n",
    "        if sum_eq is not None and int(z.sum()) != sum_eq:\n",
    "            return False\n",
    "        return True\n",
    "\n",
    "    # integer deltas with exact L1 = k\n",
    "    def deltas_exact(k):\n",
    "        d = np.zeros(n, int)\n",
    "\n",
    "        def rec(i, rem):\n",
    "            if i == n:\n",
    "                if rem == 0:\n",
    "                    yield d.copy()\n",
    "                return\n",
    "            for t in range(rem + 1):\n",
    "                if t == 0:\n",
    "                    d[i] = 0\n",
    "                    yield from rec(i + 1, rem)\n",
    "                else:\n",
    "                    d[i] = +t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "                    d[i] = -t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "            d[i] = 0\n",
    "\n",
    "        yield from rec(0, k)\n",
    "\n",
    "    t0 = time.perf_counter()\n",
    "    y = float(eval_batch(x[None, :])[0])\n",
    "    hist = [{\"iter\": 0, \"t\": 0.0, \"best_y\": y, \"x\": x.copy()}]\n",
    "    print(f\"iter=0  t=0.00s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    for it in range(1, int(max_iters) + 1):\n",
    "        cand = []\n",
    "        for k in range(1, delta + 1):\n",
    "            for dlt in deltas_exact(k):\n",
    "                z = x + dlt\n",
    "                if ok(z):\n",
    "                    cand.append(z)\n",
    "\n",
    "        if not cand:\n",
    "            print(f\"STOP: no feasible neighbors. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        Y = eval_batch(np.stack(cand, 0))\n",
    "        j = int(np.argmin(Y))\n",
    "        if float(Y[j]) >= y:\n",
    "            print(f\"STOP: local minimum. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        x, y = cand[j], float(Y[j])\n",
    "        hist.append({\"iter\": it, \"t\": time.perf_counter() - t0, \"best_y\": y, \"x\": x.copy()})\n",
    "        if it % max(1, int(print_every)) == 0:\n",
    "            print(f\"iter={it}  t={hist[-1]['t']:.2f}s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    return x, y, hist\n",
    "\n",
    "\n",
    "def _scaler_np(scaler):\n",
    "    xm = scaler[\"x_mean\"].detach().cpu().numpy().reshape(-1)\n",
    "    xs = scaler[\"x_std\"].detach().cpu().numpy().reshape(-1)\n",
    "    ym = float(scaler[\"y_mean\"].detach().cpu())\n",
    "    ys = float(scaler[\"y_std\"].detach().cpu())\n",
    "    return xm, xs, ym, ys\n",
    "\n",
    "\n",
    "def solve_dfn_ip_gurobi(dfn, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    x_mean, x_std, y_mean, y_std = _scaler_np(scaler)\n",
    "\n",
    "    cost = np.r_[dfn.cost_raw.detach().cpu().numpy(), dfn.cost_fixed.detach().cpu().numpy()]\n",
    "    cap  = np.r_[torch.round(F.softplus(dfn.cap_raw.detach())).cpu().numpy(), dfn.cap_fixed.detach().cpu().numpy()]\n",
    "\n",
    "    A = dfn.A.detach().cpu().numpy()\n",
    "    if isinstance(dfn.A, torch.nn.Parameter):\n",
    "        A = np.round(A)\n",
    "    b = np.round(dfn.b_raw.detach().cpu().numpy())\n",
    "\n",
    "    src = dfn.src.detach().cpu().numpy().astype(int)\n",
    "    dst = dfn.dst.detach().cpu().numpy().astype(int)\n",
    "    boundary = dfn.boundary.detach().cpu().numpy().astype(int)\n",
    "    fix = int(dfn.fix_node)\n",
    "    n = int(dfn.n)\n",
    "    m = int(src.size)\n",
    "    alpha = float(dfn.alpha)\n",
    "    beta  = float(dfn.beta)\n",
    "\n",
    "    out = [[] for _ in range(n)]\n",
    "    inn = [[] for _ in range(n)]\n",
    "    for e in range(m):\n",
    "        out[src[e]].append(e)\n",
    "        inn[dst[e]].append(e)\n",
    "\n",
    "    M = gp.Model(\"DFN_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    f = M.addVars(m, lb=0.0, ub=cap.tolist(), vtype=GRB.CONTINUOUS, name=\"f\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq)\n",
    "\n",
    "    xm_over_xs = x_mean / x_std\n",
    "    s = [0] * n\n",
    "    s_boundary = []\n",
    "    for r, v in enumerate(boundary):\n",
    "        const = float(b[r] - (A[r] * xm_over_xs).sum())\n",
    "        expr = const + gp.quicksum((A[r, j] / x_std[j]) * x[j] for j in range(d) if A[r, j] != 0)\n",
    "        s[v] = expr\n",
    "        s_boundary.append(expr)\n",
    "    s[fix] = -gp.quicksum(s_boundary)\n",
    "\n",
    "    for v in range(n):\n",
    "        M.addConstr(gp.quicksum(f[e] for e in out[v]) - gp.quicksum(f[e] for e in inn[v]) == s[v])\n",
    "\n",
    "    flow_cost = gp.quicksum(cost[e] * f[e] for e in range(m))\n",
    "    M.setObjective((alpha * flow_cost + beta) * y_std + y_mean, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No solution (Gurobi status {M.Status})\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None)}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_mlp_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    if not hasattr(base, \"net\"):\n",
    "        raise ValueError(\"Expected an MLP with attribute .net (nn.Sequential).\")\n",
    "    linears = [L for L in base.net if isinstance(L, torch.nn.Linear)]\n",
    "    if not linears:\n",
    "        raise ValueError(\"No Linear layers found in base.net\")\n",
    "\n",
    "    W = [L.weight.detach().cpu().numpy().astype(float) for L in linears]\n",
    "    b = [L.bias.detach().cpu().numpy().astype(float) for L in linears]\n",
    "\n",
    "    W[0] = W[0] / xs[None, :]\n",
    "    b[0] = b[0] - W[0] @ xm\n",
    "\n",
    "    W[-1] *= a_out\n",
    "    b[-1] = a_out * b[-1] + b_out\n",
    "\n",
    "    u = np.maximum(np.abs(x_min), np.abs(x_max))\n",
    "    preLU = []\n",
    "    for k in range(len(W) - 1):\n",
    "        U = np.abs(W[k]) @ u + np.abs(b[k])\n",
    "        preLU.append((-U, U))\n",
    "        u = np.maximum(0.0, U)\n",
    "\n",
    "    M = gp.Model(\"MLP_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    prev = [x[i] for i in range(d)]\n",
    "\n",
    "    for k in range(len(W) - 1):\n",
    "        Lk, Uk = preLU[k]\n",
    "        h = W[k].shape[0]\n",
    "\n",
    "        a = [M.addVar(lb=float(Lk[j]), ub=float(Uk[j]), name=f\"a{k}_{j}\") for j in range(h)]\n",
    "        z = [M.addVar(lb=0.0, ub=float(max(0.0, Uk[j])), name=f\"z{k}_{j}\") for j in range(h)]\n",
    "\n",
    "        for j in range(h):\n",
    "            M.addConstr(a[j] == b[k][j] + gp.quicksum(W[k][j, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "            Lj, Uj = float(Lk[j]), float(Uk[j])\n",
    "            if Uj <= 0.0:\n",
    "                M.addConstr(z[j] == 0.0)\n",
    "            elif Lj >= 0.0:\n",
    "                M.addConstr(z[j] == a[j])\n",
    "            else:\n",
    "                s = M.addVar(vtype=GRB.BINARY, name=f\"s{k}_{j}\")\n",
    "                M.addConstr(z[j] >= a[j])\n",
    "                M.addConstr(z[j] >= 0.0)\n",
    "                M.addConstr(z[j] <= Uj * s)\n",
    "                M.addConstr(z[j] <= a[j] - Lj * (1 - s))\n",
    "\n",
    "        prev = z\n",
    "\n",
    "    if W[-1].shape[0] != 1:\n",
    "        raise ValueError(f\"Expected scalar output, got out_dim={W[-1].shape[0]}\")\n",
    "\n",
    "    y_norm = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == b[-1][0] + gp.quicksum(W[-1][0, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "    M.setObjective(ys * y_norm + ym, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_maxaffine_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    W = base.W.detach().cpu().numpy().astype(float)\n",
    "    b = base.b.detach().cpu().numpy().astype(float)\n",
    "\n",
    "    Weff = W / xs[None, :]\n",
    "    beff = b - (Weff @ xm)\n",
    "\n",
    "    Weff *= a_out\n",
    "    beff  = a_out * beff + b_out\n",
    "\n",
    "    K = int(Weff.shape[0])\n",
    "\n",
    "    M = gp.Model(\"MaxAffine_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    t = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"t_norm\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(t >= beff[k] + gp.quicksum(Weff[k, j] * x[j] for j in range(d) if Weff[k, j] != 0.0))\n",
    "\n",
    "    M.setObjective(ys * t + ym, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_lset_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    A = model.A.detach().cpu().numpy().astype(float)\n",
    "    b = model.b.detach().cpu().numpy().astype(float)\n",
    "    T = float(model.T)\n",
    "    if T == 0.0:\n",
    "        raise ValueError(\"T must be nonzero\")\n",
    "\n",
    "    Aeff = A / xs[None, :]\n",
    "    beff = b - (Aeff @ xm)\n",
    "    K = int(Aeff.shape[0])\n",
    "\n",
    "    lin_lo = np.empty(K, dtype=float)\n",
    "    lin_hi = np.empty(K, dtype=float)\n",
    "    for k in range(K):\n",
    "        a = Aeff[k]\n",
    "        lo = beff[k]\n",
    "        hi = beff[k]\n",
    "        pos = a >= 0\n",
    "        lo += (a[pos] * x_min[pos]).sum() + (a[~pos] * x_max[~pos]).sum()\n",
    "        hi += (a[pos] * x_max[pos]).sum() + (a[~pos] * x_min[~pos]).sum()\n",
    "        lin_lo[k], lin_hi[k] = lo, hi\n",
    "\n",
    "    z_lo = lin_lo / T\n",
    "    z_hi = lin_hi / T\n",
    "    m_lo = float(np.max(z_lo))\n",
    "    m_hi = float(np.max(z_hi))\n",
    "\n",
    "    w_lo = float(np.min(z_lo - m_hi))  # <= 0\n",
    "    u_lo = float(np.exp(max(-700.0, w_lo)))\n",
    "    s_lo = max(1e-12, K * u_lo)\n",
    "    s_hi = float(K * 1.0)\n",
    "    v_lo = float(np.log(s_lo))\n",
    "    v_hi = float(np.log(s_hi))\n",
    "\n",
    "    yN_lo = float(T * (m_lo + v_lo))\n",
    "    yN_hi = float(T * (m_hi + v_hi))\n",
    "    y_lo  = float(ys * yN_lo + ym)\n",
    "    y_hi  = float(ys * yN_hi + ym)\n",
    "\n",
    "    M = gp.Model(\"lset_ip_stable_bounded\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    M.Params.FuncNonlinear = 1\n",
    "    M.Params.FeasibilityTol = 1e-9\n",
    "    M.Params.OptimalityTol  = 1e-9\n",
    "    M.Params.IntFeasTol     = 1e-9\n",
    "    M.Params.NumericFocus   = 3\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    z = M.addVars(K, lb=z_lo.tolist(), ub=z_hi.tolist(), vtype=GRB.CONTINUOUS, name=\"z\")\n",
    "    for k in range(K):\n",
    "        lin = beff[k] + gp.quicksum(Aeff[k, j] * x[j] for j in range(d) if Aeff[k, j] != 0.0)\n",
    "        M.addConstr(z[k] == lin / T, name=f\"zdef_{k}\")\n",
    "\n",
    "    m = M.addVar(lb=m_lo, ub=m_hi, vtype=GRB.CONTINUOUS, name=\"m\")\n",
    "    w = M.addVars(K, lb=w_lo, ub=0.0, vtype=GRB.CONTINUOUS, name=\"w\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(m >= z[k], name=f\"m_ge_z_{k}\")\n",
    "        M.addConstr(w[k] == z[k] - m, name=f\"wdef_{k}\")\n",
    "\n",
    "    u = M.addVars(K, lb=0.0, ub=1.0, vtype=GRB.CONTINUOUS, name=\"u\")\n",
    "    for k in range(K):\n",
    "        M.addGenConstrExp(w[k], u[k], name=f\"exp_{k}\")\n",
    "\n",
    "    s = M.addVar(lb=s_lo, ub=s_hi, vtype=GRB.CONTINUOUS, name=\"s\")\n",
    "    M.addConstr(s == gp.quicksum(u[k] for k in range(K)), name=\"sumexp_shifted\")\n",
    "\n",
    "    v = M.addVar(lb=v_lo, ub=v_hi, vtype=GRB.CONTINUOUS, name=\"v\")\n",
    "    M.addGenConstrLog(s, v, name=\"log_shifted\")\n",
    "\n",
    "    y_norm = M.addVar(lb=yN_lo, ub=yN_hi, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == T * (m + v), name=\"y_norm_def\")\n",
    "\n",
    "    y_raw = M.addVar(lb=y_lo, ub=y_hi, vtype=GRB.CONTINUOUS, name=\"y_raw\")\n",
    "    M.addConstr(y_raw == ys * y_norm + ym, name=\"y_raw_def\")\n",
    "\n",
    "    M.setObjective(y_raw, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution found. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], dtype=float)\n",
    "    y_star = float(y_raw.X)\n",
    "\n",
    "    info = {\n",
    "        \"status\": M.Status,\n",
    "        \"runtime\": M.Runtime,\n",
    "        \"gap\": getattr(M, \"MIPGap\", None),\n",
    "        \"sol_count\": M.SolCount,\n",
    "        \"obj_gurobi\": float(M.ObjVal),\n",
    "        \"obj_bound\": float(getattr(M, \"ObjBound\", float(\"nan\"))),\n",
    "    }\n",
    "    return x_star, y_star, info\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d4858979",
   "metadata": {},
   "source": [
    "## Test Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "569f2165",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def solve_ip(model_type, model, scaler, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    if model_type == \"DFN\":\n",
    "        return solve_dfn_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MLP\":\n",
    "        return solve_mlp_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MaxAffine\":\n",
    "        return solve_maxaffine_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"LSET\":\n",
    "        return solve_lset_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise ValueError(model_type)\n",
    "\n",
    "\n",
    "def suppress_stdout(fn, *args, silence: bool = False, **kwargs):\n",
    "    if not silence:\n",
    "        return fn(*args, **kwargs), \"\"\n",
    "    buf = io.StringIO()\n",
    "    with contextlib.redirect_stdout(buf):\n",
    "        out = fn(*args, **kwargs)\n",
    "    return out, buf.getvalue()\n",
    "\n",
    "\n",
    "def make_obj(model, scaler, device, chunk: int = 4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    @torch.no_grad()\n",
    "    def obj(Xraw):\n",
    "        Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "        if Xraw.dim() == 1:\n",
    "            Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "        outs = []\n",
    "        B = int(Xraw.shape[0])\n",
    "        for i in range(0, B, int(chunk)):\n",
    "            Xb = Xraw[i:i+chunk]\n",
    "            try:\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)  \n",
    "                y  = (yn * ys + ym).reshape(-1).detach().cpu() \n",
    "                if y.numel() != Xb.shape[0]:\n",
    "                    y = y.expand(Xb.shape[0]).contiguous()\n",
    "                outs.append(y)\n",
    "            except Exception as e:\n",
    "                obj._err_count += int(Xb.shape[0])\n",
    "                if obj._err_first is None:\n",
    "                    obj._err_first = repr(e)\n",
    "                outs.append(torch.full((int(Xb.shape[0]),), float(\"inf\"), device=\"cpu\"))\n",
    "\n",
    "        return torch.cat(outs, 0).numpy()\n",
    "\n",
    "    obj._err_count = 0\n",
    "    obj._err_first = None\n",
    "    return obj\n",
    "\n",
    "\n",
    "def safe_raw_mse(obj, Xraw, yraw):\n",
    "    yp = np.asarray(obj(Xraw), dtype=float).reshape(-1)\n",
    "    yt = yraw.detach().cpu().numpy().reshape(-1) if torch.is_tensor(yraw) else np.asarray(yraw, float).reshape(-1)\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "    mask = np.isfinite(yp)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions were non-finite (inf/nan)\"\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def safe_norm_mse(model, scaler, device, Xraw, yraw, *, chunk=4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "    if Xraw.dim() == 1:\n",
    "        Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "    yraw_t = yraw\n",
    "    if torch.is_tensor(yraw_t):\n",
    "        yraw_t = yraw_t.to(device=device, dtype=torch.float32)\n",
    "    else:\n",
    "        yraw_t = torch.as_tensor(yraw_t, dtype=torch.float32, device=device)\n",
    "    yraw_t = yraw_t.reshape(-1)\n",
    "\n",
    "    preds = []\n",
    "    B = int(Xraw.shape[0])\n",
    "    try:\n",
    "        with torch.no_grad():\n",
    "            for i in range(0, B, int(chunk)):\n",
    "                Xb = Xraw[i:i+chunk]\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)\n",
    "                if yn.numel() != Xb.shape[0]:\n",
    "                    yn = yn.expand(Xb.shape[0]).contiguous()\n",
    "                preds.append(yn.detach().cpu())\n",
    "    except Exception as e:\n",
    "        return np.nan, f\"model forward failed in safe_norm_mse: {repr(e)}\"\n",
    "\n",
    "    yp = torch.cat(preds, 0).numpy().reshape(-1)\n",
    "    yt = ((yraw_t - ym) / ys).detach().cpu().numpy().reshape(-1)\n",
    "\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "\n",
    "    mask = np.isfinite(yp) & np.isfinite(yt)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions or targets were non-finite (inf/nan)\"\n",
    "\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def t_to_best(hist, y_best):\n",
    "    for r in hist:\n",
    "        if abs(float(r.get(\"best_y\", np.inf)) - float(y_best)) < 1e-12:\n",
    "            return float(r.get(\"t\", float(\"nan\")))\n",
    "    return float(\"nan\")\n",
    "\n",
    "\n",
    "def check_ip_matches_obj(name, obj, x_ip, y_ip, *, strict: bool, tol: float):\n",
    "    y_obj = float(np.asarray(obj(np.asarray(x_ip)), dtype=float).reshape(-1)[0])\n",
    "    y_ip  = float(y_ip)\n",
    "    rel = abs(y_obj - y_ip) / (abs(y_obj) + 1e-12)\n",
    "    print(f\"[CHECK {name}] obj(x_ip)={y_obj:.6g}  ip_y={y_ip:.6g}  rel_err={rel:.3e}\")\n",
    "    if strict and rel > tol:\n",
    "        raise RuntimeError(f\"{name}: IP objective != obj() (rel_err={rel:.3e})\")\n",
    "    return rel\n",
    "\n",
    "\n",
    "def mean_se(x):\n",
    "    x = pd.to_numeric(pd.Series(x), errors=\"coerce\").to_numpy()\n",
    "    x = x[np.isfinite(x)]\n",
    "    n = int(x.shape[0])\n",
    "    if n == 0:\n",
    "        return np.nan, np.nan\n",
    "    m = float(x.mean())\n",
    "    se = float(x.std(ddof=1) / np.sqrt(n)) if n > 1 else 0.0\n",
    "    return m, se\n",
    "\n",
    "\n",
    "def fmt_mean_se(m, se):\n",
    "    if not np.isfinite(m):\n",
    "        return \"nan\"\n",
    "    if not np.isfinite(se):\n",
    "        return f\"{m:.6g}\"\n",
    "    return f\"{m:.6g} ± {se:.3g}\"\n",
    "\n",
    "\n",
    "def repr_solution(xs: pd.Series, seeds=None):\n",
    "    xs = xs.dropna().astype(str)\n",
    "    xs = xs[xs != \"None\"]\n",
    "    if xs.empty:\n",
    "        return None\n",
    "\n",
    "    # With seed information: show per-seed if solutions differ\n",
    "    if seeds is not None:\n",
    "        df = pd.DataFrame({\"seed\": pd.Series(seeds), \"x\": xs})\n",
    "        df = df.dropna()\n",
    "        df[\"x\"] = df[\"x\"].astype(str)\n",
    "        if df.empty:\n",
    "            return None\n",
    "        if df[\"x\"].nunique() == 1:\n",
    "            return df[\"x\"].iloc[0]\n",
    "        df = df.sort_values(\"seed\")\n",
    "        return \"\\n\".join([f\"seed={int(r.seed)}: {r.x}\" for r in df.itertuples(index=False)])\n",
    "\n",
    "    # No seed info: keep compact\n",
    "    if xs.nunique() == 1:\n",
    "        return xs.iloc[0]\n",
    "    vc = xs.value_counts()\n",
    "    top = vc.index[0]\n",
    "    n_unique = int(vc.shape[0])\n",
    "    return f\"{top} (+{n_unique-1} other)\"\n",
    "    \n",
    "\n",
    "# -----------------------------\n",
    "# Ground-truth optimum solvers\n",
    "# -----------------------------\n",
    "\n",
    "def solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    Q  = np.asarray(gt[\"Q\"], float)\n",
    "    xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "    n  = int(xs.shape[0])\n",
    "    assert Q.shape == (n, n)\n",
    "\n",
    "    m = gp.Model(\"gt_quadratic\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    x = m.addVars(n, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(n):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(n)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    expr = gp.quicksum(float(Q[i, j]) * (x[i] - float(xs[i])) * (x[j] - float(xs[j])) for i in range(n) for j in range(n))\n",
    "    m.setObjective(expr, GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT quadratic solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(n)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_assignment(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    C = np.asarray(gt[\"C\"], float)\n",
    "    n_rows, n_cols = C.shape\n",
    "    k = int(sum_eq)\n",
    "    if k > n_cols:\n",
    "        raise RuntimeError(f\"sum_eq={k} exceeds number of columns={n_cols} (infeasible)\")\n",
    "\n",
    "    m = gp.Model(\"gt_assignment\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    z = m.addVars(n_rows, vtype=GRB.BINARY, name=\"z\")  # select row i\n",
    "    y = m.addVars(n_rows, n_cols, vtype=GRB.BINARY, name=\"y\")  # assign row i to col j\n",
    "\n",
    "    m.addConstr(gp.quicksum(z[i] for i in range(n_rows)) == k, name=\"k_rows\")\n",
    "    for i in range(n_rows):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for j in range(n_cols)) == z[i], name=f\"assign_row_{i}\")\n",
    "    for j in range(n_cols):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for i in range(n_rows)) <= 1, name=f\"assign_col_{j}\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(C[i, j]) * y[i, j] for i in range(n_rows) for j in range(n_cols)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT assignment solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(z[i].X)) for i in range(n_rows)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    N  = int(gt[\"N\"])\n",
    "    SS = int(gt[\"SS\"])\n",
    "    TT = int(gt[\"TT\"])\n",
    "    src  = np.asarray(gt[\"src\"], np.int64)\n",
    "    dst  = np.asarray(gt[\"dst\"], np.int64)\n",
    "    cost = np.asarray(gt[\"cost\"], float)\n",
    "    cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "    idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "    idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "    E = int(src.shape[0])\n",
    "    k = int(idxSS.shape[0])\n",
    "    assert idxTT.shape[0] == k, \"Expect idxSS and idxTT to be same length\"\n",
    "\n",
    "    out_edges = [[] for _ in range(N)]\n",
    "    in_edges  = [[] for _ in range(N)]\n",
    "    for e in range(E):\n",
    "        out_edges[int(src[e])].append(e)\n",
    "        in_edges[int(dst[e])].append(e)\n",
    "\n",
    "    idxSS = idxSS.astype(int)\n",
    "    idxTT = idxTT.astype(int)\n",
    "    var_arc_to_i = {int(idxSS[i]): i for i in range(k)}\n",
    "    var_arc_to_i.update({int(idxTT[i]): i for i in range(k)})\n",
    "    var_arcs = sorted(var_arc_to_i.keys())\n",
    "\n",
    "    m = gp.Model(\"gt_mdvsp_true_opt\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    m.Params.NonConvex = 2\n",
    "\n",
    "    x = m.addVars(k, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(k):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(k)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    f = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, name=\"f\")\n",
    "\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            m.addConstr(f[e] <= x[i], name=f\"cap_var_{e}\")\n",
    "        else:\n",
    "            m.addConstr(f[e] <= float(cap0[e]), name=f\"cap_fix_{e}\")\n",
    "\n",
    "    F = m.addVar(vtype=GRB.CONTINUOUS, lb=0.0, name=\"F\")\n",
    "\n",
    "    # flow conservation: out - in = F at SS, = -F at TT, else 0\n",
    "    for v in range(N):\n",
    "        out_sum = gp.quicksum(f[e] for e in out_edges[v])\n",
    "        in_sum  = gp.quicksum(f[e] for e in in_edges[v])\n",
    "        rhs = F if v == SS else (-F if v == TT else 0.0)\n",
    "        m.addConstr(out_sum - in_sum == rhs, name=f\"flow_{v}\")\n",
    "\n",
    "    y = m.addVars(N, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"y\")\n",
    "    m.addConstr(y[SS] == 1.0, name=\"ySS\")\n",
    "    m.addConstr(y[TT] == 0.0, name=\"yTT\")\n",
    "\n",
    "    z = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"z\")\n",
    "    for e in range(E):\n",
    "        m.addConstr(z[e] >= y[int(src[e])] - y[int(dst[e])], name=f\"dual_{e}\")\n",
    "\n",
    "    dual_obj = gp.QuadExpr()\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            # bilinear term: x_i * z_e\n",
    "            dual_obj += x[i] * z[e]\n",
    "        else:\n",
    "            dual_obj += float(cap0[e]) * z[e]\n",
    "    m.addConstr(F == dual_obj, name=\"strong_duality\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(cost[e]) * f[e] for e in range(E)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT mdvsp true-opt failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(k)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status), \"F\": float(F.X)}\n",
    "    \n",
    "\n",
    "\n",
    "def solve_true_opt(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    \"\"\"Top-level dispatcher for ground-truth optimum.\"\"\"\n",
    "    if not isinstance(gt, dict):\n",
    "        raise RuntimeError(\"No ground-truth structure found (gt must be a dict).\")\n",
    "    t = gt.get(\"type\", None)\n",
    "    if t == \"quadratic\":\n",
    "        return solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"assignment\":\n",
    "        return solve_true_opt_assignment(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"mdvsp\":\n",
    "        return solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise RuntimeError(f\"Unknown gt type: {t}\")\n",
    "\n",
    "\n",
    "# -----------------------------\n",
    "# Benchmark runner\n",
    "# -----------------------------\n",
    "def run_benchmark(\n",
    "    *,\n",
    "    dataset_type: str,\n",
    "    dataset_params: dict,\n",
    "    runs: list[tuple[str, str, dict]],\n",
    "    train_base: dict,\n",
    "    lr_map: dict,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    n_seeds: int = 1,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    allow_plots_multi_seed: bool = True,\n",
    "    time_limit=None,\n",
    "):\n",
    "    seeds = list(range(int(n_seeds)))\n",
    "\n",
    "    learn_rows, opt_rows, spec_rows, fail_rows = [], [], [], []\n",
    "    gt_rows = []\n",
    "\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    for seed in seeds:\n",
    "        print(f\"\\n\\n===================== SEED {seed} =====================\")\n",
    "\n",
    "        for name, model_type, model_params_base in runs:\n",
    "            # ---- per-run params ----\n",
    "            dp = dict(dataset_params)\n",
    "            if vary_dataset_seed and \"seed\" in dp:\n",
    "                dp[\"seed\"] = int(seed)\n",
    "\n",
    "            mp = dict(model_params_base)\n",
    "            if vary_model_init_seed and \"seed\" in mp:\n",
    "                mp[\"seed\"] = int(seed)\n",
    "\n",
    "            tp = dict(train_base)\n",
    "            tp[\"seed\"] = int(seed)\n",
    "            tp[\"lr\"] = float(lr_map[model_type])\n",
    "\n",
    "            # avoid plot spam unless explicitly allowed\n",
    "            if (n_seeds > 1) and (tp.get(\"plot_every\", 0) not in (0, None)) and (not allow_plots_multi_seed):\n",
    "                tp[\"plot_every\"] = 0\n",
    "\n",
    "            # ---- TRAIN ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                out = generate_and_train_simple(dataset_type, dp, model_type, mp, tp)\n",
    "            except Exception as e:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=repr(e)))\n",
    "                learn_rows.append(dict(seed=seed, model=name, train_time=np.nan, best_epoch=np.nan,\n",
    "                                       best_val=np.nan, test=np.nan, train_err=repr(e)))\n",
    "                continue\n",
    "            train_time = time.perf_counter() - t0\n",
    "\n",
    "            if isinstance(out, tuple) and len(out) == 4:\n",
    "                model, data, hist, spec = out\n",
    "            elif isinstance(out, tuple) and len(out) == 3:\n",
    "                model, data, hist = out\n",
    "                n_params = sum(p.numel() for p in model.parameters())\n",
    "                spec = dict(n_params=int(n_params), extra=\"\", train_params=tp)\n",
    "            else:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=f\"Unexpected return: {type(out)}\"))\n",
    "                continue\n",
    "\n",
    "            # ---- model spec (seed 0 only) ----\n",
    "            if seed == seeds[0]:\n",
    "                spec_rows.append(dict(\n",
    "                    model=name,\n",
    "                    n_params=int(spec.get(\"n_params\", np.nan)),\n",
    "                    details=str(spec.get(\"extra\", \"\")),\n",
    "                    lr=float(spec.get(\"train_params\", {}).get(\"lr\", tp[\"lr\"])),\n",
    "                    batch_size=int(spec.get(\"train_params\", {}).get(\"batch_size\", tp[\"batch_size\"])),\n",
    "                    epochs=int(spec.get(\"train_params\", {}).get(\"epochs\", tp[\"epochs\"])),\n",
    "                ))\n",
    "\n",
    "            device = data[\"device\"]\n",
    "            scaler = data[\"scaler\"]\n",
    "            obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "            gt = data.get(\"true\", None)\n",
    "\n",
    "            # ---- compute GT optimum (once per seed) ----\n",
    "            if seed not in gt_cache_by_seed:\n",
    "                t_gt0 = time.perf_counter()\n",
    "                try:\n",
    "                    x_gt, y_gt, gt_info = solve_true_opt(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=False)\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    # evaluate with the existing helper to be extra sure\n",
    "                    y_gt_check = float(eval_true_obj(gt, x_gt))\n",
    "                    if np.isfinite(y_gt_check) and abs(y_gt_check - float(y_gt)) / (abs(float(y_gt_check)) + 1e-12) > 1e-6:\n",
    "                        fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT_CHECK\",\n",
    "                                              error=f\"gt solver mismatch: solver={y_gt:.6g} eval_true_obj={y_gt_check:.6g}\"))\n",
    "                    gt_cache_by_seed[seed] = dict(x=str(np.asarray(x_gt, int).tolist()),\n",
    "                                                  true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                                                  runtime=float(gt_time),\n",
    "                                                  err=None)\n",
    "                except Exception as e:\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "            # ---- learning metrics: best val (normalized) + test loss (normalized) ----\n",
    "            test_norm, err_te = safe_norm_mse(model, scaler, device, data['raw']['Xte'], data['raw']['yte'], chunk=int(tp.get('plot_chunk', 4096)))\n",
    "            if err_te:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"EVAL_TEST\", error=err_te))\n",
    "\n",
    "            if obj._err_count > 0:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                      error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "            val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "            best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "            best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "            learn_rows.append(dict(\n",
    "                seed=seed, model=name,\n",
    "                train_time=float(train_time),\n",
    "                best_epoch=best_ep,\n",
    "                best_val=float(best_val),\n",
    "                test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "                train_err=(err_te),\n",
    "            ))\n",
    "\n",
    "            # ---- LOCAL SEARCH ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "                (ls_out, _ls_log) = suppress_stdout(\n",
    "                    local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                    delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                    silence=silence_local_search\n",
    "                )\n",
    "                x_best, y_best, ls_hist = ls_out\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=str(np.asarray(x_best, int).tolist()),\n",
    "                    y=float(y_best),\n",
    "                    true_y=float(eval_true_obj(gt, x_best)),\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                    iters=int(len(ls_hist) - 1),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=np.nan, iters=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "            # ---- IP SOLVE (learned objective) ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                x_ip, y_ip, info = solve_ip(\n",
    "                    model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                    time_limit=time_limit, verbose=True\n",
    "                )\n",
    "                ip_time = time.perf_counter() - t0\n",
    "\n",
    "                rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "                if rel_err > ip_check_tol:\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                          error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=str(np.asarray(x_ip, int).tolist()),\n",
    "                    y=float(y_ip),\n",
    "                    true_y=float(eval_true_obj(gt, x_ip)),\n",
    "                    runtime=float(ip_time),\n",
    "                    status=(info.get(\"status\") if isinstance(info, dict) else None),\n",
    "                    gap=(info.get(\"gap\") if isinstance(info, dict) else None),\n",
    "                    ip_rel_err=float(rel_err),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ip_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ip_time),\n",
    "                    status=None, gap=None, ip_rel_err=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    spec_df  = pd.DataFrame(spec_rows).drop_duplicates(\"model\").sort_values(\"model\").reset_index(drop=True)\n",
    "    learn_df = pd.DataFrame(learn_rows).sort_values([\"model\", \"seed\"]).reset_index(drop=True)\n",
    "    opt_df   = pd.DataFrame(opt_rows).sort_values([\"model\", \"seed\", \"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # ---- per-seed gaps ----\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\", \"model\", \"ls_vs_ip_pct\", \"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        # IP true vs GT optimum true\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    for model in sorted(learn_df[\"model\"].unique()):\n",
    "        sub = learn_df[learn_df[\"model\"] == model]\n",
    "        m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "        learn_sum_rows.append(dict(model=model, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- OPTIMIZATION SUMMARY ----\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model]\n",
    "        row = {\"model\": model}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== MODEL SPECS (from seed 0 run) ===\")\n",
    "    if not spec_df.empty:\n",
    "        print(spec_df[[\"model\", \"n_params\", \"details\", \"lr\", \"batch_size\", \"epochs\"]].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not learn_summary_df.empty:\n",
    "        print(learn_summary_df.to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    return spec_df, learn_df, opt_df, fail_df, learn_summary_df, opt_summary_df, gt_df\n",
    "\n",
    "# ===========================\n",
    "# Separate Learning & Optimization helpers (no retraining needed to re-optimize)\n",
    "# ===========================\n",
    "\n",
    "def run_learning_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",
    "    n_seeds: int = 1,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    allow_plots_multi_seed: bool = True,\n",
    "):\n",
    "    \"\"\"\n",
    "    Trains all models (for all seeds) and returns a bundle you can reuse for optimization in later cells.\n",
    "\n",
    "    Returns:\n",
    "        bundle: dict with trained models + metadata\n",
    "        spec_df, learn_df, fail_df, learn_summary_df: dataframes\n",
    "    \"\"\"\n",
    "    seeds = list(range(int(n_seeds)))\n",
    "\n",
    "    learn_rows, spec_rows, fail_rows = [], [], []\n",
    "    trained = {}  # (seed, model_name) -> dict\n",
    "\n",
    "    for seed in seeds:\n",
    "        print(f\"\\n\\n===================== LEARNING | 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",
    "            # store trained artifacts for later optimization\n",
    "            device = data[\"device\"]\n",
    "            scaler = data[\"scaler\"]\n",
    "            gt = data.get(\"true\", None)\n",
    "            model.eval()\n",
    "\n",
    "            trained[(int(seed), str(name))] = dict(\n",
    "                seed=int(seed),\n",
    "                name=str(name),\n",
    "                model_type=str(model_type),\n",
    "                model=model,\n",
    "                scaler=scaler,\n",
    "                device=device,\n",
    "                gt=gt,\n",
    "                train_params=tp,\n",
    "                dataset_params=dp,\n",
    "                hist=hist,\n",
    "            )\n",
    "\n",
    "            # ---- learning metrics ----\n",
    "            test_norm, err_te = safe_norm_mse(\n",
    "                model, scaler, device, data['raw']['Xte'], data['raw']['yte'],\n",
    "                chunk=int(tp.get('plot_chunk', 4096))\n",
    "            )\n",
    "            if err_te:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"EVAL_TEST\", error=err_te))\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",
    "    # ---- 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",
    "\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",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    if not learn_df.empty and (\"model\" in learn_df.columns):\n",
    "        for model_name in sorted(learn_df[\"model\"].unique()):\n",
    "            sub = learn_df[learn_df[\"model\"] == model_name]\n",
    "            m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "            m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "            m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "            learn_sum_rows.append(dict(model=model_name, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- 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=== LEARNING 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",
    "    bundle = dict(\n",
    "        dataset_type=str(dataset_type),\n",
    "        dataset_params=dict(dataset_params),\n",
    "        runs=list(runs),\n",
    "        seeds=seeds,\n",
    "        trained=trained,\n",
    "    )\n",
    "    return bundle, spec_df, learn_df, fail_df, learn_summary_df\n",
    "\n",
    "\n",
    "def run_optimization_benchmark(\n",
    "    bundle: dict,\n",
    "    *,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    time_limit=None,\n",
    "):\n",
    "    \"\"\"\n",
    "    Runs optimization on an already-trained bundle (no retraining).\n",
    "\n",
    "    Returns:\n",
    "        opt_df, fail_df, opt_summary_df, gt_df, gap_seed_df\n",
    "    \"\"\"\n",
    "    if not isinstance(bundle, dict) or \"trained\" not in bundle:\n",
    "        raise ValueError(\"bundle must be the output of run_learning_benchmark(...).\")\n",
    "\n",
    "    trained = bundle[\"trained\"]\n",
    "    if not trained:\n",
    "        raise ValueError(\"No trained models found in bundle. Run learning first.\")\n",
    "\n",
    "    opt_rows, fail_rows = [], []\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    # compute GT optimum once per seed (if gt available)\n",
    "    # NOTE: Uses current xmin/xmax/sum_eq/time_limit, so changing constraints changes GT too.\n",
    "    for (seed, _name), info in trained.items():\n",
    "        seed = int(seed)\n",
    "        if seed in gt_cache_by_seed:\n",
    "            continue\n",
    "        gt = info.get(\"gt\", None)\n",
    "        if gt is None:\n",
    "            gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=np.nan, err=\"No gt object in trained data\")\n",
    "            continue\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",
    "            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=\"(GT)\", 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(\n",
    "                x=str(np.asarray(x_gt, int).tolist()),\n",
    "                true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                runtime=float(gt_time),\n",
    "                err=None,\n",
    "            )\n",
    "        except Exception as e:\n",
    "            gt_time = time.perf_counter() - t_gt0\n",
    "            gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "            fail_rows.append(dict(seed=seed, model=\"(GT)\", stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "    # ---- per-model optimization (LS + IP) ----\n",
    "    # stable iteration order\n",
    "    keys = sorted(trained.keys(), key=lambda k: (int(k[0]), str(k[1])))\n",
    "    for (seed, name) in keys:\n",
    "        info = trained[(seed, name)]\n",
    "        model_type = info[\"model_type\"]\n",
    "        model = info[\"model\"]\n",
    "        scaler = info[\"scaler\"]\n",
    "        device = info[\"device\"]\n",
    "        gt = info.get(\"gt\", None)\n",
    "        tp = info.get(\"train_params\", {})\n",
    "\n",
    "        print(f\"\\n\\n===================== OPTIMIZATION | SEED {seed} | {name} =====================\")\n",
    "\n",
    "        obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "\n",
    "        if obj._err_count > 0:\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                  error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "        # ---- LOCAL SEARCH ----\n",
    "        t0 = time.perf_counter()\n",
    "        try:\n",
    "            obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "            (ls_out, _ls_log) = suppress_stdout(\n",
    "                local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                silence=silence_local_search\n",
    "            )\n",
    "            x_best, y_best, ls_hist = ls_out\n",
    "            ls_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"LS\",\n",
    "                x=str(np.asarray(x_best, int).tolist()),\n",
    "                y=float(y_best),\n",
    "                true_y=float(eval_true_obj(gt, x_best)) if gt is not None else np.nan,\n",
    "                runtime=float(ls_time),\n",
    "                t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                iters=int(len(ls_hist) - 1),\n",
    "                err=None,\n",
    "            ))\n",
    "        except Exception as e:\n",
    "            ls_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"LS\",\n",
    "                x=None, y=np.nan, true_y=np.nan,\n",
    "                runtime=float(ls_time),\n",
    "                t_best=np.nan, iters=np.nan,\n",
    "                err=repr(e),\n",
    "            ))\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "        # ---- IP SOLVE (learned objective) ----\n",
    "        t0 = time.perf_counter()\n",
    "        try:\n",
    "            x_ip, y_ip, info_ip = solve_ip(\n",
    "                model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                time_limit=time_limit, verbose=True\n",
    "            )\n",
    "            ip_time = time.perf_counter() - t0\n",
    "\n",
    "            rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "            if rel_err > ip_check_tol:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                      error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"IP\",\n",
    "                x=str(np.asarray(x_ip, int).tolist()),\n",
    "                y=float(y_ip),\n",
    "                true_y=float(eval_true_obj(gt, x_ip)) if gt is not None else np.nan,\n",
    "                runtime=float(ip_time),\n",
    "                status=(info_ip.get(\"status\") if isinstance(info_ip, dict) else None),\n",
    "                gap=(info_ip.get(\"gap\") if isinstance(info_ip, dict) else None),\n",
    "                ip_rel_err=float(rel_err),\n",
    "                err=None,\n",
    "            ))\n",
    "        except Exception as e:\n",
    "            ip_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"IP\",\n",
    "                x=None, y=np.nan, true_y=np.nan,\n",
    "                runtime=float(ip_time),\n",
    "                status=None, gap=None, ip_rel_err=np.nan,\n",
    "                err=repr(e),\n",
    "            ))\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\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",
    "    # ---- 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_name in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model_name]\n",
    "        row = {\"model\": model_name}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model_name] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\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=== OPTIMIZATION 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 opt_df, fail_df, opt_summary_df, gt_df, gap_seed_df\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e88e7a1f",
   "metadata": {},
   "source": [
    "## Tests"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8edf3f97",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "===================== SEED 0 =====================\n",
      "\n",
      "--- Dataset stats (quadratic) ---\n",
      "  X: shape=(5000, 10)  mean(mean)=-0.244  std(mean)=57.9  min=-100  max=100\n",
      "  y: shape=(5000,)  mean=9.28e+05  std=3.41e+05  min=1.06e+05  max=2.57e+06\n",
      "\n",
      "\n",
      "=== Run: quadratic | DFN ===\n",
      "  data: N=5000  train/val/test=3500/750/750  dim=10\n",
      "  model: params=16,725 layers=[16, 128, 16] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[DONE] best val MSE (norm) = 3.774e-03 @ epoch 612\n",
      "\n",
      "Set parameter Username\n",
      "Set parameter LicenseID to value 2685751\n",
      "Academic license - for non-commercial use only - expires 2026-07-09\n",
      "iter=0  t=0.00s  best_y=603428  x=[15, -23, 12, -10, 17, 30, 0, 21, -33, -1]\n",
      "iter=1  t=0.04s  best_y=601888  x=[15, -23, 12, -10, 17, 30, 0, 21, -32, -2]\n",
      "iter=2  t=0.08s  best_y=600986  x=[15, -23, 12, -10, 17, 30, 0, 21, -31, -3]\n",
      "iter=3  t=0.12s  best_y=600746  x=[15, -23, 12, -10, 17, 30, 0, 21, -30, -4]\n",
      "iter=4  t=0.16s  best_y=600619  x=[15, -23, 12, -10, 17, 30, 0, 21, -29, -5]\n",
      "iter=5  t=0.20s  best_y=600522  x=[15, -23, 12, -10, 17, 30, 0, 21, -28, -6]\n",
      "iter=6  t=0.24s  best_y=600444  x=[15, -23, 12, -10, 17, 30, 0, 21, -27, -7]\n",
      "iter=7  t=0.28s  best_y=600366  x=[15, -23, 12, -10, 17, 30, 0, 21, -26, -8]\n",
      "iter=8  t=0.32s  best_y=600288  x=[15, -23, 12, -10, 17, 30, 0, 21, -25, -9]\n",
      "iter=9  t=0.36s  best_y=600210  x=[15, -23, 12, -10, 17, 30, 0, 21, -24, -10]\n",
      "iter=10  t=0.40s  best_y=600133  x=[15, -23, 12, -10, 17, 30, 0, 21, -23, -11]\n",
      "iter=11  t=0.44s  best_y=600055  x=[15, -23, 12, -10, 17, 30, 0, 21, -22, -12]\n",
      "iter=12  t=0.48s  best_y=599977  x=[15, -23, 12, -10, 17, 30, 0, 21, -21, -13]\n",
      "iter=13  t=0.52s  best_y=599899  x=[15, -23, 12, -10, 17, 30, 0, 21, -20, -14]\n",
      "iter=14  t=0.57s  best_y=599821  x=[15, -23, 12, -10, 17, 30, 0, 21, -19, -15]\n",
      "iter=15  t=0.61s  best_y=599743  x=[15, -23, 12, -10, 17, 30, 0, 21, -18, -16]\n",
      "iter=16  t=0.65s  best_y=599712  x=[15, -23, 12, -10, 17, 30, 0, 21, -17, -17]\n",
      "STOP: local minimum. best_y=599712 x=[15, -23, 12, -10, 17, 30, 0, 21, -17, -17]\n",
      "Set parameter OutputFlag to value 1\n",
      "Set parameter TimeLimit to value 300\n",
      "Gurobi Optimizer version 12.0.3 build v12.0.3rc0 (mac64[arm] - Darwin 25.2.0 25C56)\n",
      "\n",
      "CPU model: Apple M4 Max\n",
      "Thread count: 14 physical cores, 14 logical processors, using up to 14 threads\n",
      "\n",
      "Non-default parameters:\n",
      "TimeLimit  300\n",
      "\n",
      "Optimize a model with 161 rows, 8714 columns and 17681 nonzeros\n",
      "Model fingerprint: 0xe5bbb88e\n",
      "Variable types: 8704 continuous, 10 integer (0 binary)\n",
      "Coefficient statistics:\n",
      "  Matrix range     [2e-02, 1e+00]\n",
      "  Objective range  [5e+00, 2e+09]\n",
      "  Bounds range     [1e+00, 1e+06]\n",
      "  RHS range        [3e-02, 4e+01]\n",
      "Warning: Model contains large objective coefficients\n",
      "         Consider reformulating model or setting NumericFocus parameter\n",
      "         to avoid numerical issues.\n",
      "Presolve removed 0 rows and 1357 columns\n",
      "Presolve time: 0.00s\n",
      "Presolved: 161 rows, 7357 columns, 14960 nonzeros\n",
      "Variable types: 7347 continuous, 10 integer (0 binary)\n",
      "Found heuristic solution: objective 517133.93069\n",
      "\n",
      "Root relaxation: objective -4.350562e+04, 115 iterations, 0.00 seconds (0.01 work units)\n",
      "\n",
      "    Nodes    |    Current Node    |     Objective Bounds      |     Work\n",
      " Expl Unexpl |  Obj  Depth IntInf | Incumbent    BestBd   Gap | It/Node Time\n",
      "\n",
      "     0     0 -43505.617    0    9 517133.931 -43505.617   108%     -    0s\n",
      "H    0     0                    -2766.540319 -43505.617  1473%     -    0s\n",
      "     0     0 -43485.679    0    9 -2766.5403 -43485.679  1472%     -    0s\n",
      "H    0     0                    -41706.34510 -43452.104  4.19%     -    0s\n",
      "     0     2 -43452.104    0    9 -41706.345 -43452.104  4.19%     -    0s\n",
      "*  103   158              12    -42509.20642 -43351.141  1.98%   2.3    0s\n",
      "*  135   158              10    -42776.26074 -43351.141  1.34%   2.3    0s\n",
      "*  152   158              12    -42894.34894 -43351.141  1.06%   2.4    0s\n",
      "*  305   127              11    -42919.95520 -43275.774  0.83%   2.1    0s\n",
      "\n",
      "Explored 554 nodes (1172 simplex iterations) in 0.13 seconds (0.23 work units)\n",
      "Thread count was 14 (of 14 available processors)\n",
      "\n",
      "Solution count 7: -42920 -42894.3 -42776.3 ... 517134\n",
      "No other solutions better than -42920\n",
      "\n",
      "Optimal solution found (tolerance 1.00e-04)\n",
      "Best objective -4.291995520174e+04, best bound -4.291995520174e+04, gap 0.0000%\n",
      "[CHECK DFN] obj(x_ip)=-42919.9  ip_y=-42920  rel_err=4.124e-07\n",
      "\n",
      "--- Dataset stats (quadratic) ---\n",
      "  X: shape=(5000, 10)  mean(mean)=-0.244  std(mean)=57.9  min=-100  max=100\n",
      "  y: shape=(5000,)  mean=9.28e+05  std=3.41e+05  min=1.06e+05  max=2.57e+06\n",
      "\n",
      "\n",
      "=== Run: quadratic | DFN ===\n",
      "  data: N=5000  train/val/test=3500/750/750  dim=10\n",
      "  model: params=17,610 layers=[5, 400, 6] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ===================== Quadratic DATASET =====================\n",
    "N_SEEDS = 10\n",
    "\n",
    "VARY_DATASET_SEED = False\n",
    "VARY_MODEL_INIT_SEED = True\n",
    "STRICT_IP_CHECK = False\n",
    "IP_CHECK_TOL = 1e-4\n",
    "\n",
    "SILENCE_LOCAL_SEARCH = False\n",
    "ALLOW_PLOTS_MULTI_SEED = True\n",
    "\n",
    "dataset_type = \"quadratic\"\n",
    "dataset_params = dict(\n",
    "    K=5000, dim=10, eigen_min=1, eigen_max=20.0,\n",
    "    x_min=-100, x_max=100, noise_std=0.0, seed=0\n",
    ")\n",
    "in_dim = int(dataset_params[\"dim\"])\n",
    "\n",
    "train_base = dict(\n",
    "    epochs=1000,\n",
    "    batch_size=8,\n",
    "    val_frac=0.15,\n",
    "    test_frac=0.15,\n",
    "    seed=0,\n",
    "    device=(\"cuda\" if torch.cuda.is_available() else \"cpu\"),\n",
    "    eps=1e-8,\n",
    "    weight_decay=0.0,\n",
    "    plot_every=10,\n",
    "    plot_points=256,\n",
    "    plot_chunk=256,\n",
    ")\n",
    "\n",
    "# ---- DFN (learnable A) ----\n",
    "dfn_params = dict(\n",
    "    input_dim=in_dim, layer_sizes=[16, 128, 16], p_list=[1, 1],\n",
    "    seed=0, alpha=5e-3, beta=-2.0\n",
    ")\n",
    "\n",
    "# ---- DFN (fixed A = I) ----\n",
    "dfnA_layer_sizes = [5, 400, 6]\n",
    "assert dfnA_layer_sizes[0] + dfnA_layer_sizes[-1] == in_dim + 1, \"Need |L1|+|LK| = dim+1 for A_fixed=I\"\n",
    "\n",
    "dfn_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 ----\n",
    "mlp_params  = dict(in_dim=in_dim, hidden_dims=[128, 128], out_dim=1)\n",
    "maff_params = dict(in_dim=in_dim, n_pieces=1600)\n",
    "lset_params = dict(in_dim=in_dim, n_pieces=1600, T=0.05)\n",
    "\n",
    "lr_map = dict(DFN=1e-1, MLP=1e-3, MaxAffine=1e-3, LSET=1e-3)\n",
    "\n",
    "time_limit = 300\n",
    "runs = [\n",
    "    (\"DFN\",        \"DFN\", dfn_params),\n",
    "    (\"DFN_AfixI\",  \"DFN\", dfn_Afix_params),\n",
    "    (\"MLP\",        \"MLP\", mlp_params),\n",
    "    (\"MaxAffine\",  \"MaxAffine\", maff_params),\n",
    "    (\"LSET\",       \"LSET\", lset_params),\n",
    "]\n",
    "\n",
    "# ---- LEARNING (train once) ----\n",
    "learn_bundle, spec_df, learn_df, learn_fail_df, learn_summary_df = run_learning_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",
    "    n_seeds=N_SEEDS,\n",
    "    vary_dataset_seed=VARY_DATASET_SEED,\n",
    "    vary_model_init_seed=VARY_MODEL_INIT_SEED,\n",
    "    allow_plots_multi_seed=ALLOW_PLOTS_MULTI_SEED,\n",
    ")\n",
    "\n",
    "# Convenient notebook-friendly displays:\n",
    "try:\n",
    "    display(spec_df)\n",
    "    display(learn_summary_df)\n",
    "except NameError:\n",
    "    pass\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0c4398be",
   "metadata": {},
   "source": [
    "### Optimization (rerunnable)\n",
    "\n",
    "Run this cell as many times as you want with different constraints **without retraining**.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e1000bc1-6ad1-4e81-a1e5-e8b71262b2ae",
   "metadata": {},
   "outputs": [],
   "source": [
    "x0    = np.array([15,-23,12,-10,17,30,0,21,-33,-1], dtype=int)\n",
    "xmin  = np.full(in_dim, -100, dtype=int)\n",
    "xmax  = np.full(in_dim,  100, dtype=int)\n",
    "delta = 2\n",
    "sum_eq = int(x0.sum())\n",
    "\n",
    "# ---- OPTIMIZATION (rerunnable; does NOT retrain) ----\n",
    "# You can change xmin/xmax/x0/delta/sum_eq/time_limit below and rerun this cell.\n",
    "opt_df, opt_fail_df, opt_summary_df, gt_df, gap_seed_df = run_optimization_benchmark(\n",
    "    learn_bundle,\n",
    "    x0=x0, xmin=xmin, xmax=xmax,\n",
    "    delta=delta, sum_eq=sum_eq,\n",
    "    strict_ip_check=STRICT_IP_CHECK,\n",
    "    ip_check_tol=IP_CHECK_TOL,\n",
    "    silence_local_search=SILENCE_LOCAL_SEARCH,\n",
    "    time_limit=time_limit,\n",
    ")\n",
    "\n",
    "try:\n",
    "    display(opt_summary_df)\n",
    "except NameError:\n",
    "    pass\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:dfn]",
   "language": "python",
   "name": "conda-env-dfn-py"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
