{
 "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": "3ac79bcb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[ASSIGNMENT] inferred in_dim=20 from Xtmp shape=(5000, 20)\n",
      "\n",
      "\n",
      "===================== SEED 0 =====================\n",
      "\n",
      "--- Dataset stats (assignment) ---\n",
      "  X: shape=(5000, 20)  mean(mean)=0.5  std(mean)=0.5  min=0  max=1\n",
      "  y: shape=(5000,)  mean=11.5  std=3.13  min=2  max=23\n",
      "\n",
      "\n",
      "=== Run: assignment | DFN ===\n",
      "  data: N=5000  train/val/test=3500/750/750  dim=20\n",
      "  model: params=20,439 layers=[30, 80, 30] 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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DoBBt4W8U6EMtpX6CS+Mbsj+YCduLhpgqYcmIrIuWRYx6rl1bolckrFVq1Z48OABtmzZ8mrf48ePsWHDBnzwwQcICwtDw4YNlUD78OHDqFevHho3boxr1zgliIiISC3HbjzBuz/txJ//vTzedq2SRwm4i2T1cahOd7F2A+xJy/fewx/3v0TnG18i+7m5uPdvAWR6p7+1m0VElDLR4cB3Vppr9dktwM37rXdLnz496tevj/nz56N27drKvsWLFyv7ZVun0yEoKOjV/UeNGoXly5dj1apV6NOnj6ovgYiIyNHoDUZM23oBk/49jxiDEZl93DG+VTCqFMgAR8SRbksvJda5N+al7aps++/8GqFH11jyKYiIKB4yor106VJERkYq23/++Sfatm2rBNzPnz/HJ598gsDAQPj5+Skp5mfOnOFINxERkYVdfxSurLv948ZzSsDdsHgWbBhQzWEDbsGRbgtzd9GhUY8xWDPpGt6N+QduK7oh0n8D3HP8b4SFiEhTXL1ejjhb67kTSdLFZU732rVrlTnbO3bswIQJE5TfDR06VEk1//HHH5E/f354enqiZcuWiIqKUrHxREREjrUU2LJDNzFi1UmERcYgjbsLRr5XVFl/WwYnHRmDbhWkS+OOwG6/4L/p76K88QQez24B177b4exr36XwichOyYEyESne1iaBdPPmzZUR7gsXLqBgwYIoXbq08jsJwDt37oxmzZop2zLH+8qVK1ZuMRERkX14Eh6Fz5efwNrjt5XtMrnSYWKbYASkT/zFc3vG9HKV5M2SDk6t5+KCMRvSxdzHvZnNXhYFIiIiVVPMZaT7999/R/v27V/tl9HtZcuW4ciRIzh69CjatWv3RqVzIiIiSrqd5x+g3qTtSsDt4uyEIXULYuFHFRhwm2HQraJygXlxttaveGhMiyzPz+DW7x0AnuQREammVq1aSvG0s2fPKoG1ycSJE5EuXTpUqlRJSUOX6uWlSpXiO0FERJRMEdF6jFpzCu1/+w93n0YibwZvLP24EvrUKgAXHcNMc0wvV1mj6pUx7954tDrRC9nubMLtpZ8ia6txaj8tEZFDkqJpt269Of88d+7c2Lx582v7evfu/do2082JiIgS58ydpxiw8AjO3HmmbH9QPic+b1QEXm4ML+PCSxCpoF2L1vgzy6fKv7OenIkH22amxtMSERERERFZjMFg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ztrURGS7eOmXiIjI/MDo7ITGQdnw76DqGPleUfh7u+HwIzc03ZVTmSdMtmfk6pOYtvVlwN2mTADW9a/KgJscbm1qIrJdHOkmIiKKg5uLMzpVyo0WpXPgl+2X8M+pu2hYPCv7ygb1rJ5PmY//WcPCqF+M7xGpuzY1q2ATUVIx6CYiIkpAGncXDKxTEP1rF1BGwcn2BKT3wubB1eHCudvkwGtTE5HtYno5ERFRYg6YDLhtGgNusiQZzZY53OZkbWqOchNRcnCkm4iIiIgoFimaVq9oFiWlXEa4GXATUXIx6CYiIiIiioME2gy2iSilmF5OREREREREpBIG3UREREREREQqYdBNREREREREpBIG3UREREREREQqYdBNREREREREpBIG3UREREREREQqYdBNREREREREpBIG3UREREREREQqcYHGGI1G5efTp0+t3RQiIqJkMR3DTMc0R8TjOREROcrxXHNB97Nnz5SfAQEB1m4KERFRio9pvr6+DtmLPJ4TEZGjHM+djBq7zG4wGHDr1i2kTZsWTk5OFrk6IQH89evX4ePjY5E2Evs2NfCzy/7VMkf//MqhVw7Q2bJlg7OzY870svTxXGsc/W8gPuwX9gs/K/wb0tJ3S2KP55ob6ZYXkyNHDos/rnQ+D3rqYN+qi/3L/tUyR/78OuoIt9rHc61x5L+BhLBf2C/8rPBvSCvfLYk5njvm5XUiIiIiIiKiVMCgm4iIiIiIiEglDh90u7u7Y8SIEcpPsiz2rbrYv+xfLePnlxwd/wbYL/y88G+I3y2O852ruUJqRERERERERFrh8CPdRERERERERGph0E1ERERERESkEgbdRERERERERCpx6KB76tSpyJMnDzw8PFC6dGns2LHD2k2yC2PGjEHZsmWRNm1aZMqUCU2bNsXZs2et3Sy77m8nJycMGDDA2k2xGzdv3kT79u3h7+8PLy8vBAcH4+DBg9Zull2IiYnBF198oXz3enp6Im/evPjmm29gMBis3TQiq7ly5Qq6du366u8iX758SsGfqKgoh35XRo8ejUqVKinfw35+fnBUPF990/bt29G4cWNky5ZNOQdasWIFHB3Pv980bdo0lChR4tXa3BUrVsTff/8Na3DYoHvRokVKkPL555/j8OHDqFq1Kho0aIBr165Zu2mat23bNvTu3Rt79+7FP//8o5xk161bF8+f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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ===================== ASSIGNMENT DATASET =====================\n",
    "N_SEEDS = 5\n",
    "VARY_DATASET_SEED = False\n",
    "VARY_MODEL_INIT_SEED = True\n",
    "STRICT_IP_CHECK = False\n",
    "IP_CHECK_TOL = 1e-4\n",
    "SILENCE_LOCAL_SEARCH = False\n",
    "ALLOW_PLOTS_MULTI_SEED = True\n",
    "\n",
    "dataset_type = \"assignment\"\n",
    "dataset_params = dict(\n",
    "    K=5000,\n",
    "    num_nodes=20,\n",
    "    c_min=1,\n",
    "    c_max=10,\n",
    "    noise_std=0.0,\n",
    "    seed=0,\n",
    ")\n",
    "\n",
    "Xtmp, _, _ = generate_bipartite_subset_matching_dataset(**dataset_params)\n",
    "in_dim = int(np.asarray(Xtmp).shape[1])\n",
    "print(f\"[ASSIGNMENT] inferred in_dim={in_dim} from Xtmp shape={np.asarray(Xtmp).shape}\")\n",
    "\n",
    "train_base = dict(\n",
    "    epochs=1000,\n",
    "    batch_size=8,\n",
    "    val_frac=0.15,\n",
    "    test_frac=0.15,\n",
    "    seed=0,\n",
    "    device=(\"cuda\" if torch.cuda.is_available() else \"cpu\"),\n",
    "    eps=1e-8,\n",
    "    weight_decay=0.0,\n",
    "    plot_every=10,\n",
    "    plot_points=256,\n",
    "    plot_chunk=256,\n",
    ")\n",
    "\n",
    "# ---- DFN (learnable A) ----\n",
    "dfn_params = dict(\n",
    "    input_dim=in_dim, layer_sizes=[30, 80, 30], p_list=[1, 1],\n",
    "    seed=0, alpha=5e-3, beta=-2.0\n",
    ")\n",
    "\n",
    "# ---- DFN (fixed A = I) ----\n",
    "L1 = 10\n",
    "LK = 11\n",
    "dfn_Afix_params = dict(\n",
    "    input_dim=in_dim,\n",
    "    layer_sizes=[L1, 230, LK],\n",
    "    p_list=[1, 1],\n",
    "    seed=0,\n",
    "    alpha=5e-3,\n",
    "    beta=-2.0,\n",
    "    A_fixed=np.eye(in_dim, dtype=np.float32),\n",
    ")\n",
    "\n",
    "# ---- other models ----\n",
    "mlp_params  = dict(in_dim=in_dim, hidden_dims=[128, 128], out_dim=1)\n",
    "maff_params = dict(in_dim=in_dim, n_pieces=900)\n",
    "lset_params = dict(in_dim=in_dim, n_pieces=900, T=0.05)\n",
    "\n",
    "lr_map = dict(DFN=1e-1, MLP=1e-3, MaxAffine=1e-3, LSET=1e-3)\n",
    "time_limit = 300\n",
    "\n",
    "# feasible integer box + feasible x0 / sum_eq\n",
    "xmin  = np.full(in_dim, 0, dtype=int)\n",
    "xmax  = np.full(in_dim, 1, dtype=int)\n",
    "x0    = np.array([1, 1, 1, 1, 1, 1, 1, 1] + [0] * (in_dim - 8), 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",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "92dbc0ee-bbe1-4fdd-8782-44ece09dda4b",
   "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
}
