{
 "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"
   ]
  },
  {
   "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"
    }
   ],
   "source": [
    "# ===================== Quadratic DATASET =====================\n",
    "N_SEEDS = 5\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",
    "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",
    "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",
    "_ = run_benchmark(\n",
    "    dataset_type=dataset_type,\n",
    "    dataset_params=dataset_params,\n",
    "    runs=runs,\n",
    "    train_base=train_base,\n",
    "    lr_map=lr_map,\n",
    "    x0=x0, xmin=xmin, xmax=xmax,\n",
    "    delta=delta, sum_eq=sum_eq,\n",
    "    n_seeds=N_SEEDS,\n",
    "    vary_dataset_seed=VARY_DATASET_SEED,\n",
    "    vary_model_init_seed=VARY_MODEL_INIT_SEED,\n",
    "    strict_ip_check=STRICT_IP_CHECK,\n",
    "    ip_check_tol=IP_CHECK_TOL,\n",
    "    silence_local_search=SILENCE_LOCAL_SEARCH,\n",
    "    allow_plots_multi_seed=ALLOW_PLOTS_MULTI_SEED,\n",
    "    time_limit=time_limit,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e1000bc1-6ad1-4e81-a1e5-e8b71262b2ae",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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
}
