{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "f5b14808",
   "metadata": {},
   "source": [
    "# DFN"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2efb5921",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "ce8269a4",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "\n",
    "import cppimport\n",
    "\n",
    "import sys, time, math, io, contextlib\n",
    "from pathlib import Path\n",
    "from typing import Optional, List, Tuple, Dict, Any\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linear_sum_assignment\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "from IPython.display import display\n",
    "\n",
    "import gurobipy as gp\n",
    "from gurobipy import GRB\n",
    "\n",
    "_TOL = 1e-9"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "417583e0",
   "metadata": {},
   "source": [
    "## LEMON"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "62767112",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "repo = Path().resolve().parent\n",
    "if str(repo) not in sys.path:\n",
    "    sys.path.insert(0, str(repo))\n",
    "\n",
    "lemon_mcf = cppimport.imp(\"lemon_mcf\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "15e6bd1b",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "88381d28",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'status': 1, 'flow': array([1., 2., 0.]), 'potential': array([-2.,  0., -1.]), 'reduced_cost': array([0., 0., 4.]), 'at_capacity': array([False,  True, False]), 'total_cost': 4.0}\n",
      "{'value': 4.0, 'flow': array([2., 2., 2., 2.])}\n"
     ]
    }
   ],
   "source": [
    "n = 3\n",
    "src    = np.array([0, 0, 1], dtype=np.int64)\n",
    "dst    = np.array([1, 2, 2], dtype=np.int64)\n",
    "cost   = np.array([2.0, 1.0, 3.0], dtype=np.float64)\n",
    "cap    = np.array([5.0, 2.0, 4.0], dtype=np.float64)\n",
    "supply = np.array([3.0, -1.0, -2.0], dtype=np.float64)\n",
    "\n",
    "out_min_cost_flow = lemon_mcf.solve_mcf(n, src, dst, cost, cap, supply)\n",
    "print(out_min_cost_flow)\n",
    "\n",
    "n = 4\n",
    "src = np.array([0,0,1,2], dtype=np.int64)\n",
    "dst = np.array([1,2,3,3], dtype=np.int64)\n",
    "cap = np.array([3.0,2.0,2.0,4.0], dtype=np.float64)\n",
    "out_max_flow = lemon_mcf.max_flow(n, src, dst, cap, 0, 3)\n",
    "print(out_max_flow)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "75235271",
   "metadata": {},
   "source": [
    "## Dataset generators"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "67763cc2",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def make_mdvsp_dataset(\n",
    "    K: int,\n",
    "    filename: str,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    max_trips=None,\n",
    "    max_succ=None,\n",
    "):\n",
    "    \"\"\"Multiple-Depot Vehicle Scheduling (MDVSP) dataset.\n",
    "\n",
    "    Returns:\n",
    "      X: (K, m) integer-ish capacities (float32)\n",
    "      y: (K,) min-cost-flow objective values (float32)\n",
    "      gt: dict containing the fixed network pieces (for later evaluation)\n",
    "    \"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "\n",
    "    with open(filename) as f:\n",
    "        m, n, l = map(int, f.readline().split())\n",
    "        f.readline()  # blank line\n",
    "        trips = np.loadtxt(f, max_rows=n, dtype=np.int64)[:max_trips]\n",
    "        D = np.loadtxt(f, max_rows=l, dtype=np.int64)\n",
    "\n",
    "    p, s, q, e = trips.T\n",
    "    ntr = len(trips)\n",
    "\n",
    "    # node ids\n",
    "    SS = 0\n",
    "    depS = 1 + np.arange(m)\n",
    "    depT = 1 + m + np.arange(m)\n",
    "    trS  = 1 + 2*m + np.arange(ntr)\n",
    "    trT  = 1 + 2*m + ntr + np.arange(ntr)\n",
    "    TT = 1 + 2*m + 2*ntr\n",
    "    N = TT + 1\n",
    "\n",
    "    src, dst, cost, cap = [], [], [], []\n",
    "\n",
    "    # depot boundary arcs (capacity is what we learn/provide per sample)\n",
    "    for d in range(m):\n",
    "        src += [SS, int(depT[d])]\n",
    "        dst += [int(depS[d]), TT]\n",
    "        cost += [0.0, 0.0]\n",
    "        cap  += [0.0, 0.0]\n",
    "    idxSS = np.arange(0, 2*m, 2)  # arcs SS->depS\n",
    "    idxTT = np.arange(1, 2*m, 2)  # arcs depT->TT\n",
    "\n",
    "    # trip arcs (always cap=1)\n",
    "    src += trS.tolist()\n",
    "    dst += trT.tolist()\n",
    "    cost += [0.0] * ntr\n",
    "    cap  += [1.0] * ntr\n",
    "\n",
    "    # depot-to-trip and trip-to-depot arcs\n",
    "    for d in range(m):\n",
    "        # depS -> trS\n",
    "        src += [int(depS[d])] * ntr\n",
    "        dst += trS.tolist()\n",
    "        cost += (5000 + 10 * D[d, p]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "        # trT -> depT\n",
    "        src += trT.tolist()\n",
    "        dst += [int(depT[d])] * ntr\n",
    "        cost += (5000 + 10 * D[q, d]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "    # feasible trip successor arcs (trT -> next trS)\n",
    "    order = np.argsort(s)\n",
    "    p2, s2 = p[order], s[order]\n",
    "    max_succ_eff = ntr if max_succ is None else int(max_succ)\n",
    "\n",
    "    for i in range(ntr):\n",
    "        travel = D[q[i], p2]                          # time from trip i end depot -> next trip start depot\n",
    "        feas = np.flatnonzero(s2 >= e[i] + travel)[:max_succ_eff]\n",
    "        j = order[feas]\n",
    "        if j.size:\n",
    "            src += [int(trT[i])] * j.size\n",
    "            dst += trS[j].tolist()\n",
    "            cost += (8 * travel[feas] + 2 * (s[j] - e[i])).astype(float).tolist()\n",
    "            cap  += [1.0] * j.size\n",
    "\n",
    "    src = np.asarray(src, dtype=np.int64)\n",
    "    dst = np.asarray(dst, dtype=np.int64)\n",
    "    cost = np.asarray(cost, dtype=np.float64)\n",
    "    cap0 = np.asarray(cap, dtype=np.float64)\n",
    "\n",
    "    # sample capacities X and compute y via max-flow + min-cost-flow\n",
    "    X = rng.integers(x_min, np.asarray(x_max) + 1, size=(K, m)).astype(np.float64)\n",
    "    y = np.empty(K, dtype=np.float64)\n",
    "\n",
    "    for k in range(K):\n",
    "        cap_k = cap0.copy()\n",
    "        cap_k[idxSS] = X[k]\n",
    "        cap_k[idxTT] = X[k]\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap_k, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, dtype=np.float64)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "\n",
    "        y[k] = lemon_mcf.solve_mcf(N, src, dst, cost, cap_k, supply)[\"total_cost\"]\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = dict(\n",
    "        type=\"mdvsp\", N=int(N), SS=int(SS), TT=int(TT),\n",
    "        src=src, dst=dst, cost=cost, cap0=cap0, idxSS=idxSS, idxTT=idxTT\n",
    "    )\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n",
    "\n",
    "\n",
    "def generate_bipartite_subset_matching_dataset(\n",
    "    K: int, num_nodes: int, c_min: int, c_max: int, noise_std: float = 0.0, seed: int = 0\n",
    "):\n",
    "    \"\"\"Assignment-style dataset: choose a subset of left nodes, match to right nodes with min cost.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    C = rng.integers(c_min, c_max + 1, size=(num_nodes, num_nodes)).astype(np.float32)\n",
    "\n",
    "    X = np.zeros((K, num_nodes), dtype=np.float32)\n",
    "    y = np.zeros((K,), dtype=np.float32)\n",
    "\n",
    "    for k in range(K):\n",
    "        mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        while not mask.any():\n",
    "            mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        idx = np.flatnonzero(mask)\n",
    "        X[k, idx] = 1.0\n",
    "\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        y[k] = C[idx, :][r, c].sum()\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = {\"type\": \"assignment\", \"C\": C.astype(np.float32)}\n",
    "    return X, y, gt\n",
    "\n",
    "\n",
    "def generate_convex_quadratic_dataset(\n",
    "    K: int,\n",
    "    dim: int,\n",
    "    eigen_min: float,\n",
    "    eigen_max: float,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    x_star_zero: bool = False,\n",
    "):\n",
    "    \"\"\"y = (x - x*)^T Q (x - x*) + noise, with Q symmetric PSD.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    U, R = np.linalg.qr(rng.standard_normal((dim, dim)))\n",
    "    U *= np.sign(np.diag(R) + 1e-12)\n",
    "    Q = U @ np.diag(rng.uniform(eigen_min, eigen_max, dim)) @ U.T\n",
    "\n",
    "    x_star = np.zeros(dim, dtype=np.int64) if x_star_zero else rng.integers(x_min, x_max + 1, size=dim)\n",
    "\n",
    "    X = rng.integers(x_min, x_max + 1, size=(K, dim)).astype(np.float32)\n",
    "    d = X - x_star\n",
    "    y = np.einsum(\"bi,ij,bj->b\", d, Q, d) + noise_std * rng.normal(size=K)\n",
    "\n",
    "    gt = {\"type\": \"quadratic\", \"Q\": Q.astype(np.float32), \"x_star\": x_star.astype(np.int64)}\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1dfc62e4",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "89201b4a",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(50, 8) (50,) \n",
      " [[ 5.  5.  8. 10.  0.  1.  9. 10.]\n",
      " [ 2.  3.  9.  4.  3.  9.  2.  4.]\n",
      " [ 7.  6.  0.  0.  9.  8.  9.  5.]\n",
      " [ 8.  3.  4.  8.  1.  3.  1.  4.]\n",
      " [10.  1.  4.  4.  9.  2.  5.  2.]] \n",
      " [490251.12 366100.16 447590.56 326038.94 376371.84] \n",
      "\n",
      "(50, 10) (50,) \n",
      " [[0. 1. 1. 0. 0. 0. 0. 1. 0. 1.]\n",
      " [1. 0. 0. 0. 0. 0. 1. 1. 0. 0.]\n",
      " [0. 1. 0. 0. 1. 0. 1. 0. 1. 1.]\n",
      " [1. 1. 0. 0. 1. 1. 1. 1. 0. 1.]\n",
      " [0. 1. 0. 1. 1. 0. 0. 0. 1. 0.]] \n",
      " [ 9.  4. 13. 17. 11.] \n",
      "\n",
      "(50, 10) (50,) \n",
      " [[-47. -87.  -1.  69.  25. -87.  30. -31. -55. -14.]\n",
      " [ 75.  94. -72.  13.  53. -48. -46. -52. -58.  78.]\n",
      " [-57. -55. -75. -75.  56. -43.  61.  17.  72.  11.]\n",
      " [ 53.  62. -88.  12.  -9. -43. -10. -18.  -2.  64.]\n",
      " [ 65.  25.  42.  92.  28. -26. -84.  11. -54.  19.]] \n",
      " [1.5913905e+06 9.7035825e+05 7.8526031e+05 5.9535256e+05 1.0773938e+06] \n",
      "\n"
     ]
    }
   ],
   "source": [
    "# NOTE: MDVSP requires that `filename` exists on disk.\n",
    "X, y, _ = make_mdvsp_dataset(\n",
    "    K=50, filename=\"RN-8-3000-03.dat\", x_min=0, x_max=10, noise_std=1.0, seed=1, max_trips=200, max_succ=5\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")\n",
    "\n",
    "X, y, _ = generate_bipartite_subset_matching_dataset(\n",
    "    K=50, num_nodes=10, c_min=1, c_max=10, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")\n",
    "\n",
    "X, y, _ = generate_convex_quadratic_dataset(\n",
    "    K=50, dim=10, eigen_min=1.0, eigen_max=20.0, x_min=-100, x_max=100, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "70071f5f",
   "metadata": {},
   "source": [
    "## Models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "07278db8",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def _ste_round(x: torch.Tensor) -> torch.Tensor:\n",
    "    return x + (torch.round(x) - x).detach()\n",
    "\n",
    "class _MCFValue(torch.autograd.Function):\n",
    "    \n",
    "    @staticmethod\n",
    "    def forward(ctx, n_nodes, src, dst, cost, cap, supply):\n",
    "        n = int(n_nodes)\n",
    "\n",
    "        src = src.to(dtype=torch.int64).contiguous()\n",
    "        dst = dst.to(dtype=torch.int64).contiguous()\n",
    "\n",
    "        m = int(src.numel())\n",
    "        if (dst.numel() != m) or (cost.numel() != m) or (cap.numel() != m) or (supply.numel() != n):\n",
    "            raise ValueError(\"Bad shapes for MCF inputs.\")\n",
    "        if torch.abs(supply.double().sum()) > _TOL:\n",
    "            raise ValueError(\"Require sum(supply)=0\")\n",
    "\n",
    "        def as_np(t: torch.Tensor, dtype):\n",
    "            return t.detach().cpu().contiguous().view(-1).numpy().astype(dtype, copy=False)\n",
    "\n",
    "        out = lemon_mcf.solve_mcf(\n",
    "            n,\n",
    "            as_np(src, np.int64),\n",
    "            as_np(dst, np.int64),\n",
    "            as_np(cost, np.float64),\n",
    "            as_np(cap, np.float64),\n",
    "            as_np(supply, np.float64),\n",
    "            tol=_TOL,\n",
    "        )\n",
    "        if out[\"status\"] != 1:\n",
    "            raise RuntimeError(f\"LEMON failed (status={out['status']})\")\n",
    "\n",
    "        flow = out[\"flow\"]\n",
    "        pot = out[\"potential\"]\n",
    "        red = out[\"reduced_cost\"]\n",
    "        at = out.get(\"at_cap\", out.get(\"at_capacity\", None))\n",
    "        if at is None:\n",
    "            at = np.abs(flow - as_np(cap, np.float64)) <= _TOL\n",
    "\n",
    "        ctx.flow, ctx.pot, ctx.red, ctx.at = flow, pot, red, at\n",
    "        return cost.new_tensor(float(out[\"total_cost\"]))\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, g):\n",
    "        dev, dt = g.device, g.dtype\n",
    "        flow = torch.as_tensor(ctx.flow, device=dev, dtype=dt)\n",
    "        pot  = torch.as_tensor(ctx.pot,  device=dev, dtype=dt)\n",
    "        red  = torch.as_tensor(ctx.red,  device=dev, dtype=dt)\n",
    "        at   = torch.as_tensor(ctx.at,   device=dev, dtype=torch.bool)\n",
    "\n",
    "        grad_cost = flow\n",
    "        grad_cap  = torch.where(at, red, torch.zeros_like(red))\n",
    "        grad_sup  = pot.mean() - pot\n",
    "\n",
    "        return None, None, None, grad_cost * g, grad_cap * g, grad_sup * g\n",
    "\n",
    "\n",
    "class DFN(nn.Module):\n",
    "    def __init__(\n",
    "        self,\n",
    "        input_dim: int,\n",
    "        layer_sizes,\n",
    "        p_list,\n",
    "        big_cost: float = 1e6,\n",
    "        big_cap: float = 1e6,\n",
    "        seed: int = 0,\n",
    "        A_fixed=None,\n",
    "        alpha: float = 1e-6,\n",
    "        beta: float = -0.0,\n",
    "    ):\n",
    "        super().__init__()\n",
    "        self.alpha = float(alpha)\n",
    "        self.beta  = float(beta)\n",
    "\n",
    "        layer_sizes = list(map(int, layer_sizes))\n",
    "        if len(layer_sizes) < 2 or len(p_list) != len(layer_sizes) - 1:\n",
    "            raise ValueError(\"Need len(layer_sizes)>=2 and len(p_list)=len(layer_sizes)-1\")\n",
    "\n",
    "        self.n = int(sum(layer_sizes))\n",
    "        if self.n <= 0:\n",
    "            raise ValueError(\"sum(layer_sizes) must be > 0\")\n",
    "\n",
    "        # node indices per layer\n",
    "        layers, off = [], 0\n",
    "        for s in layer_sizes:\n",
    "            layers.append(torch.arange(off, off + s, dtype=torch.long))\n",
    "            off += s\n",
    "\n",
    "        L1, LK = layers[0], layers[-1]\n",
    "        if L1.numel() == 0 or LK.numel() == 0:\n",
    "            raise ValueError(\"First/last layer must be non-empty.\")\n",
    "\n",
    "        self.fix_node = int(LK[-1].item())\n",
    "        boundary = torch.cat([L1, LK[:-1]], 0)\n",
    "        self.register_buffer(\"boundary\", boundary)\n",
    "\n",
    "        gen = torch.Generator().manual_seed(int(seed))\n",
    "\n",
    "        def bipartite(U: torch.Tensor, V: torch.Tensor):\n",
    "            su, sv = int(U.numel()), int(V.numel())\n",
    "            return U.repeat_interleave(sv), V.repeat(su)\n",
    "\n",
    "        def sample_edges(U, V, p: float):\n",
    "            s, t = bipartite(U, V)\n",
    "            if p < 1.0:\n",
    "                keep = torch.rand(s.numel(), generator=gen) < float(p)\n",
    "                s, t = s[keep], t[keep]\n",
    "            return s, t\n",
    "\n",
    "        # learnable arcs between consecutive layers (both directions)\n",
    "        sf, tf, sb, tb = [], [], [], []\n",
    "        for i, p in enumerate(map(float, p_list)):\n",
    "            if not (0.0 <= p <= 1.0):\n",
    "                raise ValueError(\"p_list entries must be in [0,1]\")\n",
    "\n",
    "            s, t = sample_edges(layers[i], layers[i + 1], p)  # forward\n",
    "            sf.append(s); tf.append(t)\n",
    "\n",
    "            s, t = sample_edges(layers[i + 1], layers[i], p)  # backward\n",
    "            sb.append(s); tb.append(t)\n",
    "\n",
    "        src_param = torch.cat([torch.cat(sf, 0), torch.cat(sb, 0)], 0)\n",
    "        dst_param = torch.cat([torch.cat(tf, 0), torch.cat(tb, 0)], 0)\n",
    "        if src_param.numel() == 0:\n",
    "            raise ValueError(\"No learnable arcs (increase p_list / layer sizes).\")\n",
    "\n",
    "        s1, t1 = bipartite(L1, LK)\n",
    "        s2, t2 = bipartite(LK, L1)\n",
    "        src_fixed = torch.cat([s1, s2], 0)\n",
    "        dst_fixed = torch.cat([t1, t2], 0)\n",
    "        m_fixed = int(src_fixed.numel())\n",
    "\n",
    "        self.register_buffer(\"src\", torch.cat([src_param, src_fixed], 0))\n",
    "        self.register_buffer(\"dst\", torch.cat([dst_param, dst_fixed], 0))\n",
    "        self.register_buffer(\"cap_fixed\",  torch.full((m_fixed,), float(big_cap),  dtype=torch.float32))\n",
    "        self.register_buffer(\"cost_fixed\", torch.full((m_fixed,), float(big_cost), dtype=torch.float32))\n",
    "\n",
    "        nb = int(boundary.numel())\n",
    "        input_dim = int(input_dim)\n",
    "\n",
    "        m_param = int(src_param.numel())\n",
    "        self.cap_raw  = nn.Parameter(torch.zeros(m_param) + 0.542)\n",
    "        self.cost_raw = nn.Parameter(torch.randn(m_param) + 1.0)\n",
    "        self.b_raw    = nn.Parameter(torch.zeros(nb))\n",
    "\n",
    "        if A_fixed is None:\n",
    "            A = torch.zeros(nb, input_dim)\n",
    "            rows = torch.arange(nb)\n",
    "            A[rows, rows % input_dim] = 1.0\n",
    "            self.A = nn.Parameter(A)\n",
    "        else:\n",
    "            A_fixed = torch.as_tensor(A_fixed, dtype=torch.float32)\n",
    "            if A_fixed.shape != (nb, input_dim):\n",
    "                raise ValueError(f\"A_fixed must have shape {(nb, input_dim)}, got {tuple(A_fixed.shape)}\")\n",
    "            self.register_buffer(\"A\", A_fixed)\n",
    "\n",
    "    def forward(self, w: torch.Tensor) -> torch.Tensor:\n",
    "        capP  = _ste_round(F.softplus(self.cap_raw))\n",
    "        costP = self.cost_raw\n",
    "        b     = _ste_round(self.b_raw)\n",
    "        A     = _ste_round(self.A) if isinstance(self.A, nn.Parameter) else self.A\n",
    "\n",
    "        cap  = torch.cat([capP,  self.cap_fixed.to(w.device, w.dtype)], 0)\n",
    "        cost = torch.cat([costP, self.cost_fixed.to(w.device, w.dtype)], 0)\n",
    "\n",
    "        def one(w1: torch.Tensor) -> torch.Tensor:\n",
    "            supply = torch.zeros(self.n, device=w1.device, dtype=torch.float64)\n",
    "            supply[self.boundary] = (A.double() @ w1.double()) + b.double()\n",
    "            supply[self.fix_node] = -supply.sum()\n",
    "            return _MCFValue.apply(self.n, self.src, self.dst, cost, cap, supply)\n",
    "\n",
    "        out = one(w) if w.dim() == 1 else torch.stack([one(wi) for wi in w], 0)\n",
    "        return self.alpha * out + self.beta\n",
    "\n",
    "\n",
    "class MLP(nn.Module):\n",
    "    def __init__(self, in_dim, hidden_dims, out_dim):\n",
    "        super().__init__()\n",
    "        dims = [in_dim] + list(hidden_dims) + [out_dim]\n",
    "        layers = []\n",
    "        for a, b in zip(dims[:-2], dims[1:-1]):\n",
    "            layers += [nn.Linear(a, b), nn.ReLU()]\n",
    "        layers += [nn.Linear(dims[-2], dims[-1])]\n",
    "        self.net = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "\n",
    "class MaxAffine(nn.Module):\n",
    "    def __init__(self, in_dim: int, n_pieces: int):\n",
    "        super().__init__()\n",
    "        self.W = nn.Parameter(torch.randn(n_pieces, in_dim) / (in_dim**0.5))\n",
    "        self.b = nn.Parameter(torch.zeros(n_pieces))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        return (x @ self.W.T + self.b).max(dim=1).values\n",
    "\n",
    "\n",
    "class LSET(nn.Module):\n",
    "    def __init__(self, in_dim: int, n_pieces: int, T: float = 0.01):\n",
    "        super().__init__()\n",
    "        self.T = float(T)\n",
    "        if self.T == 0.0:\n",
    "            raise ValueError(\"T must be nonzero\")\n",
    "        self.A = nn.Parameter(torch.randn(n_pieces, in_dim) / (in_dim**0.5))\n",
    "        self.b = nn.Parameter(torch.zeros(n_pieces))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        z = (x @ self.A.t() + self.b) / self.T\n",
    "        return self.T * torch.logsumexp(z, dim=-1)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7cca0120",
   "metadata": {},
   "source": [
    "## Training Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0a7db872",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def _split_train_val_test(X: torch.Tensor, y: torch.Tensor, *, val_frac: float, test_frac: float, seed: int):\n",
    "    N = int(X.shape[0])\n",
    "    n_test = int(round(test_frac * N))\n",
    "    n_val  = int(round(val_frac  * N))\n",
    "    n_train = N - n_val - n_test\n",
    "    if n_train <= 0:\n",
    "        raise ValueError(\"splits too large; train set would be empty\")\n",
    "\n",
    "    g = torch.Generator().manual_seed(int(seed))\n",
    "    perm = torch.randperm(N, generator=g)\n",
    "    i_tr = perm[:n_train]\n",
    "    i_va = perm[n_train:n_train + n_val]\n",
    "    i_te = perm[n_train + n_val:]\n",
    "\n",
    "    return (X[i_tr], y[i_tr]), (X[i_va], y[i_va]), (X[i_te], y[i_te])\n",
    "\n",
    "\n",
    "def _fit_standardizer(Xtr: torch.Tensor, ytr: torch.Tensor, *, eps: float):\n",
    "    x_mean = Xtr.mean(0, keepdim=True)\n",
    "    x_std  = Xtr.std(0, unbiased=False, keepdim=True)\n",
    "    x_std  = torch.where(x_std < eps, torch.ones_like(x_std), x_std)\n",
    "\n",
    "    y_mean = ytr.mean()\n",
    "    y_std  = ytr.std(unbiased=False).clamp_min(eps)\n",
    "\n",
    "    return {\"x_mean\": x_mean, \"x_std\": x_std, \"y_mean\": y_mean, \"y_std\": y_std}\n",
    "\n",
    "\n",
    "def _apply_standardizer(X: torch.Tensor, y: torch.Tensor, scaler):\n",
    "    Xn = (X - scaler[\"x_mean\"]) / scaler[\"x_std\"]\n",
    "    yn = (y - scaler[\"y_mean\"]) / scaler[\"y_std\"]\n",
    "    return Xn, yn\n",
    "\n",
    "@torch.no_grad()\n",
    "def _mse_norm(model: nn.Module, loader: DataLoader, device: str):\n",
    "    model.eval()\n",
    "    tot, n = 0.0, 0\n",
    "    for xb, yb in loader:\n",
    "        xb, yb = xb.to(device), yb.to(device)\n",
    "        pred = model(xb).squeeze(-1)\n",
    "        tot += F.mse_loss(pred, yb, reduction=\"sum\").item()\n",
    "        n += yb.numel()\n",
    "    return tot / max(n, 1)\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def _predict_in_chunks(model: nn.Module, X: torch.Tensor, *, chunk: int):\n",
    "    out = []\n",
    "    for i in range(0, int(X.shape[0]), int(chunk)):\n",
    "        out.append(model(X[i:i + chunk]).squeeze(-1))\n",
    "    return torch.cat(out, 0)\n",
    "\n",
    "\n",
    "def generate_and_train_simple(dataset_type, dataset_params, model_type, model_params, train_params=None):\n",
    "    \n",
    "    tp = train_params or {}\n",
    "\n",
    "    epochs     = int(tp.get(\"epochs\", 200))\n",
    "    batch_sz   = int(tp.get(\"batch_size\", 32))\n",
    "    lr         = float(tp.get(\"lr\", 1e-3))\n",
    "    wd         = float(tp.get(\"weight_decay\", 0.0))\n",
    "    val_frac   = float(tp.get(\"val_frac\", 0.15))\n",
    "    test_frac  = float(tp.get(\"test_frac\", 0.15))\n",
    "    eps        = float(tp.get(\"eps\", 1e-8))\n",
    "    seed       = int(tp.get(\"seed\", 0))\n",
    "    device     = tp.get(\"device\", \"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "    plot_every = int(tp.get(\"plot_every\", 0) or 0)\n",
    "    plot_points= int(tp.get(\"plot_points\", 2048))\n",
    "    plot_chunk = int(tp.get(\"plot_chunk\", 4096))\n",
    "    print_stats= bool(tp.get(\"print_stats\", True))\n",
    "\n",
    "    # ---- data ----\n",
    "    dt = str(dataset_type).lower()\n",
    "    if dt == \"mdvsp\":\n",
    "        X, y, gt = make_mdvsp_dataset(**dataset_params)\n",
    "    elif dt == \"assignment\":\n",
    "        X, y, gt = generate_bipartite_subset_matching_dataset(**dataset_params)\n",
    "    elif dt == \"quadratic\":\n",
    "        X, y, gt = generate_convex_quadratic_dataset(**dataset_params)\n",
    "    else:\n",
    "        raise ValueError(\"dataset_type must be: mdvsp | assignment | quadratic\")\n",
    "\n",
    "    X = torch.as_tensor(X, dtype=torch.float32)\n",
    "    y = torch.as_tensor(y, dtype=torch.float32).view(-1)\n",
    "\n",
    "    # ---- print a quick dataset stats ----\n",
    "    if print_stats:\n",
    "        with torch.no_grad():\n",
    "            xmn = X.mean(0)\n",
    "            xsd = X.std(0, unbiased=False)\n",
    "            print(\n",
    "                f\"\\n--- Dataset stats ({dataset_type}) ---\\n\"\n",
    "                f\"  X: shape={tuple(X.shape)}  mean(mean)={xmn.mean():.3g}  std(mean)={xsd.mean():.3g}  \"\n",
    "                f\"min={float(X.min()):.3g}  max={float(X.max()):.3g}\\n\"\n",
    "                f\"  y: shape={tuple(y.shape)}  mean={float(y.mean()):.3g}  std={float(y.std(unbiased=False)):.3g}  \"\n",
    "                f\"min={float(y.min()):.3g}  max={float(y.max()):.3g}\\n\"\n",
    "            )\n",
    "\n",
    "    (Xtr, ytr), (Xva, yva), (Xte, yte) = _split_train_val_test(X, y, val_frac=val_frac, test_frac=test_frac, seed=seed)\n",
    "\n",
    "    scaler = _fit_standardizer(Xtr, ytr, eps=eps)\n",
    "    XtrN, ytrN = _apply_standardizer(Xtr, ytr, scaler)\n",
    "    XvaN, yvaN = _apply_standardizer(Xva, yva, scaler)\n",
    "    XteN, yteN = _apply_standardizer(Xte, yte, scaler)\n",
    "\n",
    "    train_loader = DataLoader(TensorDataset(XtrN, ytrN), batch_size=batch_sz, shuffle=True)\n",
    "    val_loader   = DataLoader(TensorDataset(XvaN, yvaN), batch_size=batch_sz, shuffle=False)\n",
    "\n",
    "    # subset for plotting\n",
    "    Nv = int(XvaN.shape[0])\n",
    "    if plot_points <= 0 or plot_points >= Nv:\n",
    "        plot_idx = torch.arange(Nv)\n",
    "    else:\n",
    "        g_plot = torch.Generator().manual_seed(seed + 12345)\n",
    "        plot_idx = torch.randperm(Nv, generator=g_plot)[:plot_points]\n",
    "\n",
    "    # ---- model ----\n",
    "    mt = str(model_type)\n",
    "    mp = dict(model_params)\n",
    "\n",
    "    if mt == \"DFN\":\n",
    "        model = DFN(**mp)\n",
    "        extra = f\"layers={mp.get('layer_sizes')} p_list={mp.get('p_list')} alpha={getattr(model,'alpha',None)} beta={getattr(model,'beta',None)}\"\n",
    "    else:\n",
    "        mp.pop(\"alpha\", None)\n",
    "        mp.pop(\"beta\", None)\n",
    "        if mt == \"MLP\":\n",
    "            model = MLP(**mp)\n",
    "            extra = f\"hidden={mp.get('hidden_dims')}\"\n",
    "        elif mt == \"MaxAffine\":\n",
    "            model = MaxAffine(**mp)\n",
    "            extra = f\"n_pieces={mp.get('n_pieces')}\"\n",
    "        elif mt == \"LSET\":\n",
    "            model = LSET(**mp)\n",
    "            extra = f\"n_pieces={mp.get('n_pieces')} T={mp.get('T')}\"\n",
    "        else:\n",
    "            raise ValueError(\"model_type must be: DFN | MLP | MaxAffine | LSET\")\n",
    "\n",
    "    model = model.to(device)\n",
    "    opt = torch.optim.Adam(model.parameters(), lr=lr, weight_decay=wd)\n",
    "\n",
    "    n_params = sum(p.numel() for p in model.parameters())\n",
    "    n_trainable = sum(p.numel() for p in model.parameters() if p.requires_grad)\n",
    "    print(\n",
    "        f\"\\n=== Run: {dataset_type} | {model_type} ===\\n\"\n",
    "        f\"  data: N={len(X)}  train/val/test={len(Xtr)}/{len(Xva)}/{len(Xte)}  dim={X.shape[1]}\\n\"\n",
    "        f\"  model: params={n_params:,} {extra}\\n\"\n",
    "        f\"  train: device={device}  epochs={epochs}  batch={batch_sz}  lr={lr:g}  wd={wd:g}  seed={seed}\\n\"\n",
    "    )\n",
    "\n",
    "    history = {\"train_mse_norm\": [], \"val_mse_norm\": []}\n",
    "    best_val, best_ep, best_state = float(\"inf\"), 0, None\n",
    "\n",
    "    live = display(None, display_id=True) if plot_every > 0 else None\n",
    "\n",
    "    for ep in range(1, epochs + 1):\n",
    "        model.train()\n",
    "        for xb, yb in train_loader:\n",
    "            xb, yb = xb.to(device), yb.to(device)\n",
    "            loss = F.mse_loss(model(xb).squeeze(-1), yb)\n",
    "            opt.zero_grad(set_to_none=True)\n",
    "            loss.backward()\n",
    "            opt.step()\n",
    "\n",
    "        tr_mse = _mse_norm(model, train_loader, device)\n",
    "        va_mse = _mse_norm(model, val_loader, device)\n",
    "        history[\"train_mse_norm\"].append(tr_mse)\n",
    "        history[\"val_mse_norm\"].append(va_mse)\n",
    "\n",
    "        if va_mse < best_val:\n",
    "            best_val, best_ep = va_mse, ep\n",
    "            best_state = {k: v.detach().cpu().clone() for k, v in model.state_dict().items()}\n",
    "\n",
    "        if plot_every > 0 and (ep == 1 or ep % plot_every == 0 or ep == epochs):\n",
    "            model.eval()\n",
    "            x_plot = XvaN[plot_idx].to(device)\n",
    "            y_true = yvaN[plot_idx].to(device)\n",
    "            y_pred = _predict_in_chunks(model, x_plot, chunk=plot_chunk)\n",
    "\n",
    "            fig, ax = plt.subplots(1, 2, figsize=(10, 4))\n",
    "            ax[0].plot(history[\"train_mse_norm\"], label=\"train\")\n",
    "            ax[0].plot(history[\"val_mse_norm\"], label=\"val\")\n",
    "            ax[0].set_yscale(\"log\")\n",
    "            ax[0].set_title(f\"Epoch {ep}/{epochs} | val MSE={va_mse:.3e} (norm)\")\n",
    "            ax[0].legend()\n",
    "\n",
    "            yt = y_true.detach().cpu().numpy()\n",
    "            yp = y_pred.detach().cpu().numpy()\n",
    "            ax[1].scatter(yt, yp, s=10)\n",
    "            lo = float(min(yt.min(), yp.min()))\n",
    "            hi = float(max(yt.max(), yp.max()))\n",
    "            ax[1].plot([lo, hi], [lo, hi])\n",
    "            ax[1].set_xlabel(\"y_true (norm)\")\n",
    "            ax[1].set_ylabel(\"y_pred (norm)\")\n",
    "            ax[1].set_title(f\"Val scatter (n={len(yt)})\")\n",
    "\n",
    "            plt.tight_layout()\n",
    "            live.update(fig)\n",
    "            plt.close(fig)\n",
    "\n",
    "    if best_state is not None:\n",
    "        model.load_state_dict(best_state)\n",
    "\n",
    "    if print_stats:\n",
    "        print(f\"[DONE] best val MSE (norm) = {best_val:.3e} @ epoch {best_ep}\\n\")\n",
    "\n",
    "    data = {\n",
    "        \"raw\":  {\"Xtr\": Xtr,  \"ytr\": ytr,  \"Xva\": Xva,  \"yva\": yva,  \"Xte\": Xte,  \"yte\": yte},\n",
    "        \"norm\": {\"Xtr\": XtrN, \"ytr\": ytrN, \"Xva\": XvaN, \"yva\": yvaN, \"Xte\": XteN, \"yte\": yteN},\n",
    "        \"scaler\": scaler,\n",
    "        \"best_val_mse_norm\": float(best_val),\n",
    "        \"best_epoch\": int(best_ep),\n",
    "        \"stats\": {\"n_params\": int(n_params), \"n_trainable\": int(n_trainable)},\n",
    "        \"true\": gt,\n",
    "        \"device\": device,\n",
    "    }\n",
    "\n",
    "    spec = {\n",
    "        \"model\": model_type,\n",
    "        \"extra\": extra,\n",
    "        \"n_params\": int(n_params),\n",
    "        \"n_trainable\": int(n_trainable),\n",
    "        \"model_params\": dict(model_params),\n",
    "        \"train_params\": {\n",
    "            \"epochs\": int(epochs),\n",
    "            \"batch_size\": int(batch_sz),\n",
    "            \"lr\": float(lr),\n",
    "            \"weight_decay\": float(wd),\n",
    "            \"seed\": int(seed),\n",
    "            \"device\": str(device),\n",
    "            \"val_frac\": float(val_frac),\n",
    "            \"test_frac\": float(test_frac),\n",
    "        },\n",
    "    }\n",
    "\n",
    "    return model, data, history, spec\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ace7f63f",
   "metadata": {},
   "source": [
    "## Optimization Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "38416530",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def eval_true_obj(gt, x):\n",
    "    if not isinstance(gt, dict):\n",
    "        return np.nan\n",
    "\n",
    "    x = np.asarray(x).reshape(-1)\n",
    "    t = gt.get(\"type\", None)\n",
    "\n",
    "    if t == \"quadratic\":\n",
    "        Q = np.asarray(gt[\"Q\"], float)\n",
    "        xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "        d = x.astype(float) - xs\n",
    "        return float(d @ Q @ d)\n",
    "\n",
    "    if t == \"assignment\":\n",
    "        C = np.asarray(gt[\"C\"], float)\n",
    "        idx = np.flatnonzero(x > 0.5)\n",
    "        if idx.size == 0:\n",
    "            return 0.0\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        return float(C[idx, :][r, c].sum())\n",
    "\n",
    "    if t == \"mdvsp\":\n",
    "        N, SS, TT = int(gt[\"N\"]), int(gt[\"SS\"]), int(gt[\"TT\"])\n",
    "        src = np.asarray(gt[\"src\"], np.int64)\n",
    "        dst = np.asarray(gt[\"dst\"], np.int64)\n",
    "        cost = np.asarray(gt[\"cost\"], float)\n",
    "        cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "        idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "        idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "        cap = cap0.copy()\n",
    "        cap[idxSS] = x.astype(float)\n",
    "        cap[idxTT] = x.astype(float)\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, float)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "        return float(lemon_mcf.solve_mcf(N, src, dst, cost, cap, supply)[\"total_cost\"])\n",
    "\n",
    "    return np.nan\n",
    "\n",
    "\n",
    "def local_search_l1_int(\n",
    "    f,\n",
    "    x0,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    delta: int,\n",
    "    sum_eq=None,\n",
    "    max_iters: int = 10_000,\n",
    "    print_every: int = 1,\n",
    "):\n",
    "    x  = np.asarray(x0,    int).ravel()\n",
    "    lo = np.asarray(x_min, int).ravel()\n",
    "    hi = np.asarray(x_max, int).ravel()\n",
    "    n = int(x.size)\n",
    "    delta = int(delta)\n",
    "\n",
    "    assert lo.size == n and hi.size == n\n",
    "    assert np.all(x >= lo) and np.all(x <= hi)\n",
    "    if sum_eq is not None:\n",
    "        sum_eq = int(sum_eq)\n",
    "        assert int(x.sum()) == sum_eq\n",
    "\n",
    "    def eval_batch(X):\n",
    "        y = np.asarray(f(X), float).reshape(-1)\n",
    "        return y\n",
    "\n",
    "    def ok(z):\n",
    "        if np.any(z < lo) or np.any(z > hi):\n",
    "            return False\n",
    "        if sum_eq is not None and int(z.sum()) != sum_eq:\n",
    "            return False\n",
    "        return True\n",
    "\n",
    "    # integer deltas with exact L1 = k\n",
    "    def deltas_exact(k):\n",
    "        d = np.zeros(n, int)\n",
    "\n",
    "        def rec(i, rem):\n",
    "            if i == n:\n",
    "                if rem == 0:\n",
    "                    yield d.copy()\n",
    "                return\n",
    "            for t in range(rem + 1):\n",
    "                if t == 0:\n",
    "                    d[i] = 0\n",
    "                    yield from rec(i + 1, rem)\n",
    "                else:\n",
    "                    d[i] = +t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "                    d[i] = -t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "            d[i] = 0\n",
    "\n",
    "        yield from rec(0, k)\n",
    "\n",
    "    t0 = time.perf_counter()\n",
    "    y = float(eval_batch(x[None, :])[0])\n",
    "    hist = [{\"iter\": 0, \"t\": 0.0, \"best_y\": y, \"x\": x.copy()}]\n",
    "    print(f\"iter=0  t=0.00s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    for it in range(1, int(max_iters) + 1):\n",
    "        cand = []\n",
    "        for k in range(1, delta + 1):\n",
    "            for dlt in deltas_exact(k):\n",
    "                z = x + dlt\n",
    "                if ok(z):\n",
    "                    cand.append(z)\n",
    "\n",
    "        if not cand:\n",
    "            print(f\"STOP: no feasible neighbors. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        Y = eval_batch(np.stack(cand, 0))\n",
    "        j = int(np.argmin(Y))\n",
    "        if float(Y[j]) >= y:\n",
    "            print(f\"STOP: local minimum. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        x, y = cand[j], float(Y[j])\n",
    "        hist.append({\"iter\": it, \"t\": time.perf_counter() - t0, \"best_y\": y, \"x\": x.copy()})\n",
    "        if it % max(1, int(print_every)) == 0:\n",
    "            print(f\"iter={it}  t={hist[-1]['t']:.2f}s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    return x, y, hist\n",
    "\n",
    "\n",
    "def _scaler_np(scaler):\n",
    "    xm = scaler[\"x_mean\"].detach().cpu().numpy().reshape(-1)\n",
    "    xs = scaler[\"x_std\"].detach().cpu().numpy().reshape(-1)\n",
    "    ym = float(scaler[\"y_mean\"].detach().cpu())\n",
    "    ys = float(scaler[\"y_std\"].detach().cpu())\n",
    "    return xm, xs, ym, ys\n",
    "\n",
    "\n",
    "def solve_dfn_ip_gurobi(dfn, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    x_mean, x_std, y_mean, y_std = _scaler_np(scaler)\n",
    "\n",
    "    cost = np.r_[dfn.cost_raw.detach().cpu().numpy(), dfn.cost_fixed.detach().cpu().numpy()]\n",
    "    cap  = np.r_[torch.round(F.softplus(dfn.cap_raw.detach())).cpu().numpy(), dfn.cap_fixed.detach().cpu().numpy()]\n",
    "\n",
    "    A = dfn.A.detach().cpu().numpy()\n",
    "    if isinstance(dfn.A, torch.nn.Parameter):\n",
    "        A = np.round(A)\n",
    "    b = np.round(dfn.b_raw.detach().cpu().numpy())\n",
    "\n",
    "    src = dfn.src.detach().cpu().numpy().astype(int)\n",
    "    dst = dfn.dst.detach().cpu().numpy().astype(int)\n",
    "    boundary = dfn.boundary.detach().cpu().numpy().astype(int)\n",
    "    fix = int(dfn.fix_node)\n",
    "    n = int(dfn.n)\n",
    "    m = int(src.size)\n",
    "    alpha = float(dfn.alpha)\n",
    "    beta  = float(dfn.beta)\n",
    "\n",
    "    out = [[] for _ in range(n)]\n",
    "    inn = [[] for _ in range(n)]\n",
    "    for e in range(m):\n",
    "        out[src[e]].append(e)\n",
    "        inn[dst[e]].append(e)\n",
    "\n",
    "    M = gp.Model(\"DFN_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    f = M.addVars(m, lb=0.0, ub=cap.tolist(), vtype=GRB.CONTINUOUS, name=\"f\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq)\n",
    "\n",
    "    xm_over_xs = x_mean / x_std\n",
    "    s = [0] * n\n",
    "    s_boundary = []\n",
    "    for r, v in enumerate(boundary):\n",
    "        const = float(b[r] - (A[r] * xm_over_xs).sum())\n",
    "        expr = const + gp.quicksum((A[r, j] / x_std[j]) * x[j] for j in range(d) if A[r, j] != 0)\n",
    "        s[v] = expr\n",
    "        s_boundary.append(expr)\n",
    "    s[fix] = -gp.quicksum(s_boundary)\n",
    "\n",
    "    for v in range(n):\n",
    "        M.addConstr(gp.quicksum(f[e] for e in out[v]) - gp.quicksum(f[e] for e in inn[v]) == s[v])\n",
    "\n",
    "    flow_cost = gp.quicksum(cost[e] * f[e] for e in range(m))\n",
    "    M.setObjective((alpha * flow_cost + beta) * y_std + y_mean, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No solution (Gurobi status {M.Status})\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None)}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_mlp_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    if not hasattr(base, \"net\"):\n",
    "        raise ValueError(\"Expected an MLP with attribute .net (nn.Sequential).\")\n",
    "    linears = [L for L in base.net if isinstance(L, torch.nn.Linear)]\n",
    "    if not linears:\n",
    "        raise ValueError(\"No Linear layers found in base.net\")\n",
    "\n",
    "    W = [L.weight.detach().cpu().numpy().astype(float) for L in linears]\n",
    "    b = [L.bias.detach().cpu().numpy().astype(float) for L in linears]\n",
    "\n",
    "    W[0] = W[0] / xs[None, :]\n",
    "    b[0] = b[0] - W[0] @ xm\n",
    "\n",
    "    W[-1] *= a_out\n",
    "    b[-1] = a_out * b[-1] + b_out\n",
    "\n",
    "    u = np.maximum(np.abs(x_min), np.abs(x_max))\n",
    "    preLU = []\n",
    "    for k in range(len(W) - 1):\n",
    "        U = np.abs(W[k]) @ u + np.abs(b[k])\n",
    "        preLU.append((-U, U))\n",
    "        u = np.maximum(0.0, U)\n",
    "\n",
    "    M = gp.Model(\"MLP_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    prev = [x[i] for i in range(d)]\n",
    "\n",
    "    for k in range(len(W) - 1):\n",
    "        Lk, Uk = preLU[k]\n",
    "        h = W[k].shape[0]\n",
    "\n",
    "        a = [M.addVar(lb=float(Lk[j]), ub=float(Uk[j]), name=f\"a{k}_{j}\") for j in range(h)]\n",
    "        z = [M.addVar(lb=0.0, ub=float(max(0.0, Uk[j])), name=f\"z{k}_{j}\") for j in range(h)]\n",
    "\n",
    "        for j in range(h):\n",
    "            M.addConstr(a[j] == b[k][j] + gp.quicksum(W[k][j, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "            Lj, Uj = float(Lk[j]), float(Uk[j])\n",
    "            if Uj <= 0.0:\n",
    "                M.addConstr(z[j] == 0.0)\n",
    "            elif Lj >= 0.0:\n",
    "                M.addConstr(z[j] == a[j])\n",
    "            else:\n",
    "                s = M.addVar(vtype=GRB.BINARY, name=f\"s{k}_{j}\")\n",
    "                M.addConstr(z[j] >= a[j])\n",
    "                M.addConstr(z[j] >= 0.0)\n",
    "                M.addConstr(z[j] <= Uj * s)\n",
    "                M.addConstr(z[j] <= a[j] - Lj * (1 - s))\n",
    "\n",
    "        prev = z\n",
    "\n",
    "    if W[-1].shape[0] != 1:\n",
    "        raise ValueError(f\"Expected scalar output, got out_dim={W[-1].shape[0]}\")\n",
    "\n",
    "    y_norm = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == b[-1][0] + gp.quicksum(W[-1][0, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "    M.setObjective(ys * y_norm + ym, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_maxaffine_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    W = base.W.detach().cpu().numpy().astype(float)\n",
    "    b = base.b.detach().cpu().numpy().astype(float)\n",
    "\n",
    "    Weff = W / xs[None, :]\n",
    "    beff = b - (Weff @ xm)\n",
    "\n",
    "    Weff *= a_out\n",
    "    beff  = a_out * beff + b_out\n",
    "\n",
    "    K = int(Weff.shape[0])\n",
    "\n",
    "    M = gp.Model(\"MaxAffine_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    t = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"t_norm\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(t >= beff[k] + gp.quicksum(Weff[k, j] * x[j] for j in range(d) if Weff[k, j] != 0.0))\n",
    "\n",
    "    M.setObjective(ys * t + ym, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_lset_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    A = model.A.detach().cpu().numpy().astype(float)\n",
    "    b = model.b.detach().cpu().numpy().astype(float)\n",
    "    T = float(model.T)\n",
    "    if T == 0.0:\n",
    "        raise ValueError(\"T must be nonzero\")\n",
    "\n",
    "    Aeff = A / xs[None, :]\n",
    "    beff = b - (Aeff @ xm)\n",
    "    K = int(Aeff.shape[0])\n",
    "\n",
    "    lin_lo = np.empty(K, dtype=float)\n",
    "    lin_hi = np.empty(K, dtype=float)\n",
    "    for k in range(K):\n",
    "        a = Aeff[k]\n",
    "        lo = beff[k]\n",
    "        hi = beff[k]\n",
    "        pos = a >= 0\n",
    "        lo += (a[pos] * x_min[pos]).sum() + (a[~pos] * x_max[~pos]).sum()\n",
    "        hi += (a[pos] * x_max[pos]).sum() + (a[~pos] * x_min[~pos]).sum()\n",
    "        lin_lo[k], lin_hi[k] = lo, hi\n",
    "\n",
    "    z_lo = lin_lo / T\n",
    "    z_hi = lin_hi / T\n",
    "    m_lo = float(np.max(z_lo))\n",
    "    m_hi = float(np.max(z_hi))\n",
    "\n",
    "    w_lo = float(np.min(z_lo - m_hi))  # <= 0\n",
    "    u_lo = float(np.exp(max(-700.0, w_lo)))\n",
    "    s_lo = max(1e-12, K * u_lo)\n",
    "    s_hi = float(K * 1.0)\n",
    "    v_lo = float(np.log(s_lo))\n",
    "    v_hi = float(np.log(s_hi))\n",
    "\n",
    "    yN_lo = float(T * (m_lo + v_lo))\n",
    "    yN_hi = float(T * (m_hi + v_hi))\n",
    "    y_lo  = float(ys * yN_lo + ym)\n",
    "    y_hi  = float(ys * yN_hi + ym)\n",
    "\n",
    "    M = gp.Model(\"lset_ip_stable_bounded\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    M.Params.FuncNonlinear = 1\n",
    "    M.Params.FeasibilityTol = 1e-9\n",
    "    M.Params.OptimalityTol  = 1e-9\n",
    "    M.Params.IntFeasTol     = 1e-9\n",
    "    M.Params.NumericFocus   = 3\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    z = M.addVars(K, lb=z_lo.tolist(), ub=z_hi.tolist(), vtype=GRB.CONTINUOUS, name=\"z\")\n",
    "    for k in range(K):\n",
    "        lin = beff[k] + gp.quicksum(Aeff[k, j] * x[j] for j in range(d) if Aeff[k, j] != 0.0)\n",
    "        M.addConstr(z[k] == lin / T, name=f\"zdef_{k}\")\n",
    "\n",
    "    m = M.addVar(lb=m_lo, ub=m_hi, vtype=GRB.CONTINUOUS, name=\"m\")\n",
    "    w = M.addVars(K, lb=w_lo, ub=0.0, vtype=GRB.CONTINUOUS, name=\"w\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(m >= z[k], name=f\"m_ge_z_{k}\")\n",
    "        M.addConstr(w[k] == z[k] - m, name=f\"wdef_{k}\")\n",
    "\n",
    "    u = M.addVars(K, lb=0.0, ub=1.0, vtype=GRB.CONTINUOUS, name=\"u\")\n",
    "    for k in range(K):\n",
    "        M.addGenConstrExp(w[k], u[k], name=f\"exp_{k}\")\n",
    "\n",
    "    s = M.addVar(lb=s_lo, ub=s_hi, vtype=GRB.CONTINUOUS, name=\"s\")\n",
    "    M.addConstr(s == gp.quicksum(u[k] for k in range(K)), name=\"sumexp_shifted\")\n",
    "\n",
    "    v = M.addVar(lb=v_lo, ub=v_hi, vtype=GRB.CONTINUOUS, name=\"v\")\n",
    "    M.addGenConstrLog(s, v, name=\"log_shifted\")\n",
    "\n",
    "    y_norm = M.addVar(lb=yN_lo, ub=yN_hi, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == T * (m + v), name=\"y_norm_def\")\n",
    "\n",
    "    y_raw = M.addVar(lb=y_lo, ub=y_hi, vtype=GRB.CONTINUOUS, name=\"y_raw\")\n",
    "    M.addConstr(y_raw == ys * y_norm + ym, name=\"y_raw_def\")\n",
    "\n",
    "    M.setObjective(y_raw, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution found. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], dtype=float)\n",
    "    y_star = float(y_raw.X)\n",
    "\n",
    "    info = {\n",
    "        \"status\": M.Status,\n",
    "        \"runtime\": M.Runtime,\n",
    "        \"gap\": getattr(M, \"MIPGap\", None),\n",
    "        \"sol_count\": M.SolCount,\n",
    "        \"obj_gurobi\": float(M.ObjVal),\n",
    "        \"obj_bound\": float(getattr(M, \"ObjBound\", float(\"nan\"))),\n",
    "    }\n",
    "    return x_star, y_star, info\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d4858979",
   "metadata": {},
   "source": [
    "## Test Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "569f2165",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def solve_ip(model_type, model, scaler, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    if model_type == \"DFN\":\n",
    "        return solve_dfn_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MLP\":\n",
    "        return solve_mlp_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MaxAffine\":\n",
    "        return solve_maxaffine_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"LSET\":\n",
    "        return solve_lset_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise ValueError(model_type)\n",
    "\n",
    "\n",
    "def suppress_stdout(fn, *args, silence: bool = False, **kwargs):\n",
    "    if not silence:\n",
    "        return fn(*args, **kwargs), \"\"\n",
    "    buf = io.StringIO()\n",
    "    with contextlib.redirect_stdout(buf):\n",
    "        out = fn(*args, **kwargs)\n",
    "    return out, buf.getvalue()\n",
    "\n",
    "\n",
    "def make_obj(model, scaler, device, chunk: int = 4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    @torch.no_grad()\n",
    "    def obj(Xraw):\n",
    "        Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "        if Xraw.dim() == 1:\n",
    "            Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "        outs = []\n",
    "        B = int(Xraw.shape[0])\n",
    "        for i in range(0, B, int(chunk)):\n",
    "            Xb = Xraw[i:i+chunk]\n",
    "            try:\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)  \n",
    "                y  = (yn * ys + ym).reshape(-1).detach().cpu() \n",
    "                if y.numel() != Xb.shape[0]:\n",
    "                    y = y.expand(Xb.shape[0]).contiguous()\n",
    "                outs.append(y)\n",
    "            except Exception as e:\n",
    "                obj._err_count += int(Xb.shape[0])\n",
    "                if obj._err_first is None:\n",
    "                    obj._err_first = repr(e)\n",
    "                outs.append(torch.full((int(Xb.shape[0]),), float(\"inf\"), device=\"cpu\"))\n",
    "\n",
    "        return torch.cat(outs, 0).numpy()\n",
    "\n",
    "    obj._err_count = 0\n",
    "    obj._err_first = None\n",
    "    return obj\n",
    "\n",
    "\n",
    "def safe_raw_mse(obj, Xraw, yraw):\n",
    "    yp = np.asarray(obj(Xraw), dtype=float).reshape(-1)\n",
    "    yt = yraw.detach().cpu().numpy().reshape(-1) if torch.is_tensor(yraw) else np.asarray(yraw, float).reshape(-1)\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "    mask = np.isfinite(yp)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions were non-finite (inf/nan)\"\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def safe_norm_mse(model, scaler, device, Xraw, yraw, *, chunk=4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "    if Xraw.dim() == 1:\n",
    "        Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "    yraw_t = yraw\n",
    "    if torch.is_tensor(yraw_t):\n",
    "        yraw_t = yraw_t.to(device=device, dtype=torch.float32)\n",
    "    else:\n",
    "        yraw_t = torch.as_tensor(yraw_t, dtype=torch.float32, device=device)\n",
    "    yraw_t = yraw_t.reshape(-1)\n",
    "\n",
    "    preds = []\n",
    "    B = int(Xraw.shape[0])\n",
    "    try:\n",
    "        with torch.no_grad():\n",
    "            for i in range(0, B, int(chunk)):\n",
    "                Xb = Xraw[i:i+chunk]\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)\n",
    "                if yn.numel() != Xb.shape[0]:\n",
    "                    yn = yn.expand(Xb.shape[0]).contiguous()\n",
    "                preds.append(yn.detach().cpu())\n",
    "    except Exception as e:\n",
    "        return np.nan, f\"model forward failed in safe_norm_mse: {repr(e)}\"\n",
    "\n",
    "    yp = torch.cat(preds, 0).numpy().reshape(-1)\n",
    "    yt = ((yraw_t - ym) / ys).detach().cpu().numpy().reshape(-1)\n",
    "\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "\n",
    "    mask = np.isfinite(yp) & np.isfinite(yt)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions or targets were non-finite (inf/nan)\"\n",
    "\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def t_to_best(hist, y_best):\n",
    "    for r in hist:\n",
    "        if abs(float(r.get(\"best_y\", np.inf)) - float(y_best)) < 1e-12:\n",
    "            return float(r.get(\"t\", float(\"nan\")))\n",
    "    return float(\"nan\")\n",
    "\n",
    "\n",
    "def check_ip_matches_obj(name, obj, x_ip, y_ip, *, strict: bool, tol: float):\n",
    "    y_obj = float(np.asarray(obj(np.asarray(x_ip)), dtype=float).reshape(-1)[0])\n",
    "    y_ip  = float(y_ip)\n",
    "    rel = abs(y_obj - y_ip) / (abs(y_obj) + 1e-12)\n",
    "    print(f\"[CHECK {name}] obj(x_ip)={y_obj:.6g}  ip_y={y_ip:.6g}  rel_err={rel:.3e}\")\n",
    "    if strict and rel > tol:\n",
    "        raise RuntimeError(f\"{name}: IP objective != obj() (rel_err={rel:.3e})\")\n",
    "    return rel\n",
    "\n",
    "\n",
    "def mean_se(x):\n",
    "    x = pd.to_numeric(pd.Series(x), errors=\"coerce\").to_numpy()\n",
    "    x = x[np.isfinite(x)]\n",
    "    n = int(x.shape[0])\n",
    "    if n == 0:\n",
    "        return np.nan, np.nan\n",
    "    m = float(x.mean())\n",
    "    se = float(x.std(ddof=1) / np.sqrt(n)) if n > 1 else 0.0\n",
    "    return m, se\n",
    "\n",
    "\n",
    "def fmt_mean_se(m, se):\n",
    "    if not np.isfinite(m):\n",
    "        return \"nan\"\n",
    "    if not np.isfinite(se):\n",
    "        return f\"{m:.6g}\"\n",
    "    return f\"{m:.6g} ± {se:.3g}\"\n",
    "\n",
    "\n",
    "def repr_solution(xs: pd.Series, seeds=None):\n",
    "    xs = xs.dropna().astype(str)\n",
    "    xs = xs[xs != \"None\"]\n",
    "    if xs.empty:\n",
    "        return None\n",
    "\n",
    "    # With seed information: show per-seed if solutions differ\n",
    "    if seeds is not None:\n",
    "        df = pd.DataFrame({\"seed\": pd.Series(seeds), \"x\": xs})\n",
    "        df = df.dropna()\n",
    "        df[\"x\"] = df[\"x\"].astype(str)\n",
    "        if df.empty:\n",
    "            return None\n",
    "        if df[\"x\"].nunique() == 1:\n",
    "            return df[\"x\"].iloc[0]\n",
    "        df = df.sort_values(\"seed\")\n",
    "        return \"\\n\".join([f\"seed={int(r.seed)}: {r.x}\" for r in df.itertuples(index=False)])\n",
    "\n",
    "    # No seed info: keep compact\n",
    "    if xs.nunique() == 1:\n",
    "        return xs.iloc[0]\n",
    "    vc = xs.value_counts()\n",
    "    top = vc.index[0]\n",
    "    n_unique = int(vc.shape[0])\n",
    "    return f\"{top} (+{n_unique-1} other)\"\n",
    "    \n",
    "\n",
    "# -----------------------------\n",
    "# Ground-truth optimum solvers\n",
    "# -----------------------------\n",
    "\n",
    "def solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    Q  = np.asarray(gt[\"Q\"], float)\n",
    "    xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "    n  = int(xs.shape[0])\n",
    "    assert Q.shape == (n, n)\n",
    "\n",
    "    m = gp.Model(\"gt_quadratic\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    x = m.addVars(n, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(n):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(n)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    expr = gp.quicksum(float(Q[i, j]) * (x[i] - float(xs[i])) * (x[j] - float(xs[j])) for i in range(n) for j in range(n))\n",
    "    m.setObjective(expr, GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT quadratic solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(n)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_assignment(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    C = np.asarray(gt[\"C\"], float)\n",
    "    n_rows, n_cols = C.shape\n",
    "    k = int(sum_eq)\n",
    "    if k > n_cols:\n",
    "        raise RuntimeError(f\"sum_eq={k} exceeds number of columns={n_cols} (infeasible)\")\n",
    "\n",
    "    m = gp.Model(\"gt_assignment\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    z = m.addVars(n_rows, vtype=GRB.BINARY, name=\"z\")  # select row i\n",
    "    y = m.addVars(n_rows, n_cols, vtype=GRB.BINARY, name=\"y\")  # assign row i to col j\n",
    "\n",
    "    m.addConstr(gp.quicksum(z[i] for i in range(n_rows)) == k, name=\"k_rows\")\n",
    "    for i in range(n_rows):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for j in range(n_cols)) == z[i], name=f\"assign_row_{i}\")\n",
    "    for j in range(n_cols):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for i in range(n_rows)) <= 1, name=f\"assign_col_{j}\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(C[i, j]) * y[i, j] for i in range(n_rows) for j in range(n_cols)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT assignment solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(z[i].X)) for i in range(n_rows)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    N  = int(gt[\"N\"])\n",
    "    SS = int(gt[\"SS\"])\n",
    "    TT = int(gt[\"TT\"])\n",
    "    src  = np.asarray(gt[\"src\"], np.int64)\n",
    "    dst  = np.asarray(gt[\"dst\"], np.int64)\n",
    "    cost = np.asarray(gt[\"cost\"], float)\n",
    "    cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "    idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "    idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "    E = int(src.shape[0])\n",
    "    k = int(idxSS.shape[0])\n",
    "    assert idxTT.shape[0] == k, \"Expect idxSS and idxTT to be same length\"\n",
    "\n",
    "    out_edges = [[] for _ in range(N)]\n",
    "    in_edges  = [[] for _ in range(N)]\n",
    "    for e in range(E):\n",
    "        out_edges[int(src[e])].append(e)\n",
    "        in_edges[int(dst[e])].append(e)\n",
    "\n",
    "    idxSS = idxSS.astype(int)\n",
    "    idxTT = idxTT.astype(int)\n",
    "    var_arc_to_i = {int(idxSS[i]): i for i in range(k)}\n",
    "    var_arc_to_i.update({int(idxTT[i]): i for i in range(k)})\n",
    "    var_arcs = sorted(var_arc_to_i.keys())\n",
    "\n",
    "    m = gp.Model(\"gt_mdvsp_true_opt\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    m.Params.NonConvex = 2\n",
    "\n",
    "    x = m.addVars(k, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(k):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(k)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    f = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, name=\"f\")\n",
    "\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            m.addConstr(f[e] <= x[i], name=f\"cap_var_{e}\")\n",
    "        else:\n",
    "            m.addConstr(f[e] <= float(cap0[e]), name=f\"cap_fix_{e}\")\n",
    "\n",
    "    F = m.addVar(vtype=GRB.CONTINUOUS, lb=0.0, name=\"F\")\n",
    "\n",
    "    # flow conservation: out - in = F at SS, = -F at TT, else 0\n",
    "    for v in range(N):\n",
    "        out_sum = gp.quicksum(f[e] for e in out_edges[v])\n",
    "        in_sum  = gp.quicksum(f[e] for e in in_edges[v])\n",
    "        rhs = F if v == SS else (-F if v == TT else 0.0)\n",
    "        m.addConstr(out_sum - in_sum == rhs, name=f\"flow_{v}\")\n",
    "\n",
    "    y = m.addVars(N, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"y\")\n",
    "    m.addConstr(y[SS] == 1.0, name=\"ySS\")\n",
    "    m.addConstr(y[TT] == 0.0, name=\"yTT\")\n",
    "\n",
    "    z = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"z\")\n",
    "    for e in range(E):\n",
    "        m.addConstr(z[e] >= y[int(src[e])] - y[int(dst[e])], name=f\"dual_{e}\")\n",
    "\n",
    "    dual_obj = gp.QuadExpr()\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            # bilinear term: x_i * z_e\n",
    "            dual_obj += x[i] * z[e]\n",
    "        else:\n",
    "            dual_obj += float(cap0[e]) * z[e]\n",
    "    m.addConstr(F == dual_obj, name=\"strong_duality\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(cost[e]) * f[e] for e in range(E)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT mdvsp true-opt failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(k)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status), \"F\": float(F.X)}\n",
    "    \n",
    "\n",
    "\n",
    "def solve_true_opt(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    \"\"\"Top-level dispatcher for ground-truth optimum.\"\"\"\n",
    "    if not isinstance(gt, dict):\n",
    "        raise RuntimeError(\"No ground-truth structure found (gt must be a dict).\")\n",
    "    t = gt.get(\"type\", None)\n",
    "    if t == \"quadratic\":\n",
    "        return solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"assignment\":\n",
    "        return solve_true_opt_assignment(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"mdvsp\":\n",
    "        return solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise RuntimeError(f\"Unknown gt type: {t}\")\n",
    "\n",
    "\n",
    "# -----------------------------\n",
    "# Benchmark runner\n",
    "# -----------------------------\n",
    "def run_benchmark(\n",
    "    *,\n",
    "    dataset_type: str,\n",
    "    dataset_params: dict,\n",
    "    runs: list[tuple[str, str, dict]],\n",
    "    train_base: dict,\n",
    "    lr_map: dict,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    n_seeds: int = 1,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    allow_plots_multi_seed: bool = True,\n",
    "    time_limit=None,\n",
    "):\n",
    "    seeds = list(range(int(n_seeds)))\n",
    "\n",
    "    learn_rows, opt_rows, spec_rows, fail_rows = [], [], [], []\n",
    "    gt_rows = []\n",
    "\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    for seed in seeds:\n",
    "        print(f\"\\n\\n===================== SEED {seed} =====================\")\n",
    "\n",
    "        for name, model_type, model_params_base in runs:\n",
    "            # ---- per-run params ----\n",
    "            dp = dict(dataset_params)\n",
    "            if vary_dataset_seed and \"seed\" in dp:\n",
    "                dp[\"seed\"] = int(seed)\n",
    "\n",
    "            mp = dict(model_params_base)\n",
    "            if vary_model_init_seed and \"seed\" in mp:\n",
    "                mp[\"seed\"] = int(seed)\n",
    "\n",
    "            tp = dict(train_base)\n",
    "            tp[\"seed\"] = int(seed)\n",
    "            tp[\"lr\"] = float(lr_map[model_type])\n",
    "\n",
    "            # avoid plot spam unless explicitly allowed\n",
    "            if (n_seeds > 1) and (tp.get(\"plot_every\", 0) not in (0, None)) and (not allow_plots_multi_seed):\n",
    "                tp[\"plot_every\"] = 0\n",
    "\n",
    "            # ---- TRAIN ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                out = generate_and_train_simple(dataset_type, dp, model_type, mp, tp)\n",
    "            except Exception as e:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=repr(e)))\n",
    "                learn_rows.append(dict(seed=seed, model=name, train_time=np.nan, best_epoch=np.nan,\n",
    "                                       best_val=np.nan, test=np.nan, train_err=repr(e)))\n",
    "                continue\n",
    "            train_time = time.perf_counter() - t0\n",
    "\n",
    "            if isinstance(out, tuple) and len(out) == 4:\n",
    "                model, data, hist, spec = out\n",
    "            elif isinstance(out, tuple) and len(out) == 3:\n",
    "                model, data, hist = out\n",
    "                n_params = sum(p.numel() for p in model.parameters())\n",
    "                spec = dict(n_params=int(n_params), extra=\"\", train_params=tp)\n",
    "            else:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=f\"Unexpected return: {type(out)}\"))\n",
    "                continue\n",
    "\n",
    "            # ---- model spec (seed 0 only) ----\n",
    "            if seed == seeds[0]:\n",
    "                spec_rows.append(dict(\n",
    "                    model=name,\n",
    "                    n_params=int(spec.get(\"n_params\", np.nan)),\n",
    "                    details=str(spec.get(\"extra\", \"\")),\n",
    "                    lr=float(spec.get(\"train_params\", {}).get(\"lr\", tp[\"lr\"])),\n",
    "                    batch_size=int(spec.get(\"train_params\", {}).get(\"batch_size\", tp[\"batch_size\"])),\n",
    "                    epochs=int(spec.get(\"train_params\", {}).get(\"epochs\", tp[\"epochs\"])),\n",
    "                ))\n",
    "\n",
    "            device = data[\"device\"]\n",
    "            scaler = data[\"scaler\"]\n",
    "            obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "            gt = data.get(\"true\", None)\n",
    "\n",
    "            # ---- compute GT optimum (once per seed) ----\n",
    "            if seed not in gt_cache_by_seed:\n",
    "                t_gt0 = time.perf_counter()\n",
    "                try:\n",
    "                    x_gt, y_gt, gt_info = solve_true_opt(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=False)\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    # evaluate with the existing helper to be extra sure\n",
    "                    y_gt_check = float(eval_true_obj(gt, x_gt))\n",
    "                    if np.isfinite(y_gt_check) and abs(y_gt_check - float(y_gt)) / (abs(float(y_gt_check)) + 1e-12) > 1e-6:\n",
    "                        fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT_CHECK\",\n",
    "                                              error=f\"gt solver mismatch: solver={y_gt:.6g} eval_true_obj={y_gt_check:.6g}\"))\n",
    "                    gt_cache_by_seed[seed] = dict(x=str(np.asarray(x_gt, int).tolist()),\n",
    "                                                  true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                                                  runtime=float(gt_time),\n",
    "                                                  err=None)\n",
    "                except Exception as e:\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "            # ---- learning metrics: best val (normalized) + test loss (normalized) ----\n",
    "            test_norm, err_te = safe_norm_mse(model, scaler, device, data['raw']['Xte'], data['raw']['yte'], chunk=int(tp.get('plot_chunk', 4096)))\n",
    "            if err_te:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"EVAL_TEST\", error=err_te))\n",
    "\n",
    "            if obj._err_count > 0:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                      error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "            val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "            best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "            best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "            learn_rows.append(dict(\n",
    "                seed=seed, model=name,\n",
    "                train_time=float(train_time),\n",
    "                best_epoch=best_ep,\n",
    "                best_val=float(best_val),\n",
    "                test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "                train_err=(err_te),\n",
    "            ))\n",
    "\n",
    "            # ---- LOCAL SEARCH ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "                (ls_out, _ls_log) = suppress_stdout(\n",
    "                    local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                    delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                    silence=silence_local_search\n",
    "                )\n",
    "                x_best, y_best, ls_hist = ls_out\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=str(np.asarray(x_best, int).tolist()),\n",
    "                    y=float(y_best),\n",
    "                    true_y=float(eval_true_obj(gt, x_best)),\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                    iters=int(len(ls_hist) - 1),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=np.nan, iters=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "            # ---- IP SOLVE (learned objective) ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                x_ip, y_ip, info = solve_ip(\n",
    "                    model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                    time_limit=time_limit, verbose=True\n",
    "                )\n",
    "                ip_time = time.perf_counter() - t0\n",
    "\n",
    "                rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "                if rel_err > ip_check_tol:\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                          error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=str(np.asarray(x_ip, int).tolist()),\n",
    "                    y=float(y_ip),\n",
    "                    true_y=float(eval_true_obj(gt, x_ip)),\n",
    "                    runtime=float(ip_time),\n",
    "                    status=(info.get(\"status\") if isinstance(info, dict) else None),\n",
    "                    gap=(info.get(\"gap\") if isinstance(info, dict) else None),\n",
    "                    ip_rel_err=float(rel_err),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ip_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ip_time),\n",
    "                    status=None, gap=None, ip_rel_err=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    spec_df  = pd.DataFrame(spec_rows).drop_duplicates(\"model\").sort_values(\"model\").reset_index(drop=True)\n",
    "    learn_df = pd.DataFrame(learn_rows).sort_values([\"model\", \"seed\"]).reset_index(drop=True)\n",
    "    opt_df   = pd.DataFrame(opt_rows).sort_values([\"model\", \"seed\", \"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # ---- per-seed gaps ----\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\", \"model\", \"ls_vs_ip_pct\", \"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        # IP true vs GT optimum true\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    for model in sorted(learn_df[\"model\"].unique()):\n",
    "        sub = learn_df[learn_df[\"model\"] == model]\n",
    "        m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "        learn_sum_rows.append(dict(model=model, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- OPTIMIZATION SUMMARY ----\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model]\n",
    "        row = {\"model\": model}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== MODEL SPECS (from seed 0 run) ===\")\n",
    "    if not spec_df.empty:\n",
    "        print(spec_df[[\"model\", \"n_params\", \"details\", \"lr\", \"batch_size\", \"epochs\"]].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not learn_summary_df.empty:\n",
    "        print(learn_summary_df.to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    return spec_df, learn_df, opt_df, fail_df, learn_summary_df, opt_summary_df, gt_df\n",
    "\n",
    "# ===========================\n",
    "# Separate Learning & Optimization helpers (no retraining needed to re-optimize)\n",
    "# ===========================\n",
    "\n",
    "def run_learning_benchmark(\n",
    "    *,\n",
    "    dataset_type: str,\n",
    "    dataset_params: dict,\n",
    "    runs: list[tuple[str, str, dict]],\n",
    "    train_base: dict,\n",
    "    lr_map: dict,\n",
    "    n_seeds: int = 1,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    allow_plots_multi_seed: bool = True,\n",
    "):\n",
    "    \"\"\"\n",
    "    Trains all models (for all seeds) and returns a bundle you can reuse for optimization in later cells.\n",
    "\n",
    "    Returns:\n",
    "        bundle: dict with trained models + metadata\n",
    "        spec_df, learn_df, fail_df, learn_summary_df: dataframes\n",
    "    \"\"\"\n",
    "    seeds = list(range(int(n_seeds)))\n",
    "\n",
    "    learn_rows, spec_rows, fail_rows = [], [], []\n",
    "    trained = {}  # (seed, model_name) -> dict\n",
    "\n",
    "    for seed in seeds:\n",
    "        print(f\"\\n\\n===================== LEARNING | SEED {seed} =====================\")\n",
    "\n",
    "        for name, model_type, model_params_base in runs:\n",
    "            # ---- per-run params ----\n",
    "            dp = dict(dataset_params)\n",
    "            if vary_dataset_seed and \"seed\" in dp:\n",
    "                dp[\"seed\"] = int(seed)\n",
    "\n",
    "            mp = dict(model_params_base)\n",
    "            if vary_model_init_seed and \"seed\" in mp:\n",
    "                mp[\"seed\"] = int(seed)\n",
    "\n",
    "            tp = dict(train_base)\n",
    "            tp[\"seed\"] = int(seed)\n",
    "            tp[\"lr\"] = float(lr_map[model_type])\n",
    "\n",
    "            # avoid plot spam unless explicitly allowed\n",
    "            if (n_seeds > 1) and (tp.get(\"plot_every\", 0) not in (0, None)) and (not allow_plots_multi_seed):\n",
    "                tp[\"plot_every\"] = 0\n",
    "\n",
    "            # ---- TRAIN ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                out = generate_and_train_simple(dataset_type, dp, model_type, mp, tp)\n",
    "            except Exception as e:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=repr(e)))\n",
    "                learn_rows.append(dict(seed=seed, model=name, train_time=np.nan, best_epoch=np.nan,\n",
    "                                       best_val=np.nan, test=np.nan, train_err=repr(e)))\n",
    "                continue\n",
    "            train_time = time.perf_counter() - t0\n",
    "\n",
    "            if isinstance(out, tuple) and len(out) == 4:\n",
    "                model, data, hist, spec = out\n",
    "            elif isinstance(out, tuple) and len(out) == 3:\n",
    "                model, data, hist = out\n",
    "                n_params = sum(p.numel() for p in model.parameters())\n",
    "                spec = dict(n_params=int(n_params), extra=\"\", train_params=tp)\n",
    "            else:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=f\"Unexpected return: {type(out)}\"))\n",
    "                continue\n",
    "\n",
    "            # ---- model spec (seed 0 only) ----\n",
    "            if seed == seeds[0]:\n",
    "                spec_rows.append(dict(\n",
    "                    model=name,\n",
    "                    n_params=int(spec.get(\"n_params\", np.nan)),\n",
    "                    details=str(spec.get(\"extra\", \"\")),\n",
    "                    lr=float(spec.get(\"train_params\", {}).get(\"lr\", tp[\"lr\"])),\n",
    "                    batch_size=int(spec.get(\"train_params\", {}).get(\"batch_size\", tp[\"batch_size\"])),\n",
    "                    epochs=int(spec.get(\"train_params\", {}).get(\"epochs\", tp[\"epochs\"])),\n",
    "                ))\n",
    "\n",
    "            # store trained artifacts for later optimization\n",
    "            device = data[\"device\"]\n",
    "            scaler = data[\"scaler\"]\n",
    "            gt = data.get(\"true\", None)\n",
    "            model.eval()\n",
    "\n",
    "            trained[(int(seed), str(name))] = dict(\n",
    "                seed=int(seed),\n",
    "                name=str(name),\n",
    "                model_type=str(model_type),\n",
    "                model=model,\n",
    "                scaler=scaler,\n",
    "                device=device,\n",
    "                gt=gt,\n",
    "                train_params=tp,\n",
    "                dataset_params=dp,\n",
    "                hist=hist,\n",
    "            )\n",
    "\n",
    "            # ---- learning metrics ----\n",
    "            test_norm, err_te = safe_norm_mse(\n",
    "                model, scaler, device, data['raw']['Xte'], data['raw']['yte'],\n",
    "                chunk=int(tp.get('plot_chunk', 4096))\n",
    "            )\n",
    "            if err_te:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"EVAL_TEST\", error=err_te))\n",
    "\n",
    "            val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "            best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "            best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "            learn_rows.append(dict(\n",
    "                seed=seed, model=name,\n",
    "                train_time=float(train_time),\n",
    "                best_epoch=best_ep,\n",
    "                best_val=float(best_val),\n",
    "                test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "                train_err=(err_te),\n",
    "            ))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    spec_df  = pd.DataFrame(spec_rows).drop_duplicates(\"model\").sort_values(\"model\").reset_index(drop=True)\n",
    "    learn_df = pd.DataFrame(learn_rows).sort_values([\"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    if not learn_df.empty and (\"model\" in learn_df.columns):\n",
    "        for model_name in sorted(learn_df[\"model\"].unique()):\n",
    "            sub = learn_df[learn_df[\"model\"] == model_name]\n",
    "            m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "            m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "            m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "            learn_sum_rows.append(dict(model=model_name, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== MODEL SPECS (from seed 0 run) ===\")\n",
    "    if not spec_df.empty:\n",
    "        print(spec_df[[\"model\", \"n_params\", \"details\", \"lr\", \"batch_size\", \"epochs\"]].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not learn_summary_df.empty:\n",
    "        print(learn_summary_df.to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    bundle = dict(\n",
    "        dataset_type=str(dataset_type),\n",
    "        dataset_params=dict(dataset_params),\n",
    "        runs=list(runs),\n",
    "        seeds=seeds,\n",
    "        trained=trained,\n",
    "    )\n",
    "    return bundle, spec_df, learn_df, fail_df, learn_summary_df\n",
    "\n",
    "\n",
    "def run_optimization_benchmark(\n",
    "    bundle: dict,\n",
    "    *,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    time_limit=None,\n",
    "):\n",
    "    \"\"\"\n",
    "    Runs optimization on an already-trained bundle (no retraining).\n",
    "\n",
    "    Returns:\n",
    "        opt_df, fail_df, opt_summary_df, gt_df, gap_seed_df\n",
    "    \"\"\"\n",
    "    if not isinstance(bundle, dict) or \"trained\" not in bundle:\n",
    "        raise ValueError(\"bundle must be the output of run_learning_benchmark(...).\")\n",
    "\n",
    "    trained = bundle[\"trained\"]\n",
    "    if not trained:\n",
    "        raise ValueError(\"No trained models found in bundle. Run learning first.\")\n",
    "\n",
    "    opt_rows, fail_rows = [], []\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    # compute GT optimum once per seed (if gt available)\n",
    "    # NOTE: Uses current xmin/xmax/sum_eq/time_limit, so changing constraints changes GT too.\n",
    "    for (seed, _name), info in trained.items():\n",
    "        seed = int(seed)\n",
    "        if seed in gt_cache_by_seed:\n",
    "            continue\n",
    "        gt = info.get(\"gt\", None)\n",
    "        if gt is None:\n",
    "            gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=np.nan, err=\"No gt object in trained data\")\n",
    "            continue\n",
    "        t_gt0 = time.perf_counter()\n",
    "        try:\n",
    "            x_gt, y_gt, gt_info = solve_true_opt(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=False)\n",
    "            gt_time = time.perf_counter() - t_gt0\n",
    "            y_gt_check = float(eval_true_obj(gt, x_gt))\n",
    "            if np.isfinite(y_gt_check) and abs(y_gt_check - float(y_gt)) / (abs(float(y_gt_check)) + 1e-12) > 1e-6:\n",
    "                fail_rows.append(dict(seed=seed, model=\"(GT)\", stage=\"GT_OPT_CHECK\",\n",
    "                                      error=f\"gt solver mismatch: solver={y_gt:.6g} eval_true_obj={y_gt_check:.6g}\"))\n",
    "            gt_cache_by_seed[seed] = dict(\n",
    "                x=str(np.asarray(x_gt, int).tolist()),\n",
    "                true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                runtime=float(gt_time),\n",
    "                err=None,\n",
    "            )\n",
    "        except Exception as e:\n",
    "            gt_time = time.perf_counter() - t_gt0\n",
    "            gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "            fail_rows.append(dict(seed=seed, model=\"(GT)\", stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "    # ---- per-model optimization (LS + IP) ----\n",
    "    # stable iteration order\n",
    "    keys = sorted(trained.keys(), key=lambda k: (int(k[0]), str(k[1])))\n",
    "    for (seed, name) in keys:\n",
    "        info = trained[(seed, name)]\n",
    "        model_type = info[\"model_type\"]\n",
    "        model = info[\"model\"]\n",
    "        scaler = info[\"scaler\"]\n",
    "        device = info[\"device\"]\n",
    "        gt = info.get(\"gt\", None)\n",
    "        tp = info.get(\"train_params\", {})\n",
    "\n",
    "        print(f\"\\n\\n===================== OPTIMIZATION | SEED {seed} | {name} =====================\")\n",
    "\n",
    "        obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "\n",
    "        if obj._err_count > 0:\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                  error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "        # ---- LOCAL SEARCH ----\n",
    "        t0 = time.perf_counter()\n",
    "        try:\n",
    "            obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "            (ls_out, _ls_log) = suppress_stdout(\n",
    "                local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                silence=silence_local_search\n",
    "            )\n",
    "            x_best, y_best, ls_hist = ls_out\n",
    "            ls_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"LS\",\n",
    "                x=str(np.asarray(x_best, int).tolist()),\n",
    "                y=float(y_best),\n",
    "                true_y=float(eval_true_obj(gt, x_best)) if gt is not None else np.nan,\n",
    "                runtime=float(ls_time),\n",
    "                t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                iters=int(len(ls_hist) - 1),\n",
    "                err=None,\n",
    "            ))\n",
    "        except Exception as e:\n",
    "            ls_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"LS\",\n",
    "                x=None, y=np.nan, true_y=np.nan,\n",
    "                runtime=float(ls_time),\n",
    "                t_best=np.nan, iters=np.nan,\n",
    "                err=repr(e),\n",
    "            ))\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "        # ---- IP SOLVE (learned objective) ----\n",
    "        t0 = time.perf_counter()\n",
    "        try:\n",
    "            x_ip, y_ip, info_ip = solve_ip(\n",
    "                model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                time_limit=time_limit, verbose=True\n",
    "            )\n",
    "            ip_time = time.perf_counter() - t0\n",
    "\n",
    "            rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "            if rel_err > ip_check_tol:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                      error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"IP\",\n",
    "                x=str(np.asarray(x_ip, int).tolist()),\n",
    "                y=float(y_ip),\n",
    "                true_y=float(eval_true_obj(gt, x_ip)) if gt is not None else np.nan,\n",
    "                runtime=float(ip_time),\n",
    "                status=(info_ip.get(\"status\") if isinstance(info_ip, dict) else None),\n",
    "                gap=(info_ip.get(\"gap\") if isinstance(info_ip, dict) else None),\n",
    "                ip_rel_err=float(rel_err),\n",
    "                err=None,\n",
    "            ))\n",
    "        except Exception as e:\n",
    "            ip_time = time.perf_counter() - t0\n",
    "            opt_rows.append(dict(\n",
    "                seed=seed, model=name, method=\"IP\",\n",
    "                x=None, y=np.nan, true_y=np.nan,\n",
    "                runtime=float(ip_time),\n",
    "                status=None, gap=None, ip_rel_err=np.nan,\n",
    "                err=repr(e),\n",
    "            ))\n",
    "            fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    opt_df = pd.DataFrame(opt_rows).sort_values([\"model\", \"seed\", \"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # ---- per-seed gaps ----\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\", \"model\", \"ls_vs_ip_pct\", \"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        # IP true vs GT optimum true\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    # ---- OPTIMIZATION SUMMARY ----\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model_name in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model_name]\n",
    "        row = {\"model\": model_name}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model_name] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    return opt_df, fail_df, opt_summary_df, gt_df, gap_seed_df\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e88e7a1f",
   "metadata": {},
   "source": [
    "## Tests"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "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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5KAufZvEiRMn1P1jjz2Gn376Sb0ugY0cqE+1rsCECRPUhZKtW7eWXuiR36RMdyfbVH4bEijKRR65OGBrTPBOXnf/vyM357/zNo+nW6D9/vvv6l76YNsZDAYV+LmzT5Mg0G7p0qXq5OSrr75SAbGcTMoJhPQVl/dxduutt6paB+eee666QCTbVX6Hsn/wpd+ir7/pynxWb+R3N2jQII8XOo4ePap+qxJQyz5QToynTZuGTz/9tHQeOVGWz7V48WI89dRTaj753Pfdd586qXMn+0ZZP6lXsXDhQrXvsLv88svViZ9s45tvvlnVgrjnnntU37wLLrig9ALhgw8+qH7L7uT/jbxnRb9dIgqeG2+8UQVJ7heBpZ+zXESUoFz2p/4+HkrDiATxcs4j+w7ZF8v+RM6jnAPufv36qdflIp/sP+R8S4LrrKwsNY9ckJZ1k32avMcXX3yhzuGkYUL6S9tJTQy5yCrnaPJYbrLvl32X9PW2n9PZX5PpQvanw4YNUwG3nDf873//U4G91FPxVJtGlivHawlCy7vIePDgQXWhQwJ9T2Qby8VkOa+Vfa4sUy7WO18c9ucxXN533759Ho9Lch4qy5eLy3Ih4aGHHlLnn87kGCzkHF22v5xLyYUHORc/fPhwmfeU342s/4IFCypcN6JyVapdnLyml3u66XQ6l3llmqTqHj161CVlqmPHjta2bduWTrviiitUiu7+/ftd/n7EiBGqf630XRJPPvmkes9ffvmlwvTybt26qWXZrVy5Uk3/4osvKvWtSv9T+TtJp3VP9/E0XXz++ecuKa6S2i3Pjxw5UmZeScdt3769y7TJkyer9femvPRySX+WbemJLMc5xV3SlW699dYy88l6y/vL53BOg5e+Qu4kjV36i/srvfz7779Xy5K+uO5pdpdcconXv5N5JI1Y+uOOGTOmyunlkvpt77993333laYsS38q6fPsadvLb6t58+al/w+kX7SkZ0l/KfdULm//d9zTt8v7f+Z+88YffeEOHjxobdCggerq4Jy6Lr9RSRnft2+fy/zSj12W6dy9QX6Psk2kb7j045a+ZQ899JDaX1x11VWl88nvy1OfcKmzIPuRBx54oML19fU37cn69evVctx/P95SAOX9nn/++TKvScqfvPbvv/+6TJcU9+HDh5c+nzp1qsf5brvtNtUVZPv27S77tDZt2lgN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D09cfmwFlLxsRIXYnpvcXEUKuIWxY8eiX79+NreJi4szCbjbtm2L5s2bY/ly670ajGdzMJ7RoTRN6VIbj9aJwOcHLmLtLxdx8dYD+Hi4YebjyeibFsNkaUREVKZUE3RzTDcRlbXYyHAcKohDPe1ZHAtJRztLG4lu03GmU1w5i74dWuLVHSvRsVECioSfwdWAoV8X+xmi1XDN083sS0IVULIgV7Gze3mtiEBMGTcG3727GZeVUAysZLvLcGSTniXaj4oqPDxcLva4ePGiDLhTUlKwYsUKaM0T/JWxu9l5WP3TeXy294J8Xq9qEN7s1xDxD28qERERlSXVBN1ERGVNJGUakPMSIjQ3MKxG8d25nU2T6qFoUr373/oMMRa7WQmy4P/ebDZi9szE+aazUK2YbY8kjEDqge+wKq89BhWzba3IYIRO2oBUdze2YpYx0cKdnp6OatWqyWzl167pppoTIiIK54ovS+NW78fW41flPa/RbRIwsUNNeLqX740AIiKquBh0ExH9DSuf7YDdv11Hb47Pt0tM54lA66dQzbf4ued7P9YJswu2okP9WLs+21EJCcm2TZs24bfffpNLdLRpngrF3u4Kpey5jjVx6tod/K9e9Ut0U4iIiMgRNEp5XREdOKb7xIkTcrqSwMC/lumYiIioLIkkYkFBQRX62lXax0DMBCASExIREZX3tUurpjHdR48elZlUiYiIqGJjwE1ERM5CNUE3ERERERERkbNh0E1ERERERETkIAy6iYiIiIiIiBxENUG3SKKWnJyMtLS08t4VIiIiIiIiInUF3UykRkRERERERM5GNUE3ERERERERkbNh0E1ERERERETkIAy6iYiIiIiIiByEQTcRERERERGRgzDoJiIiIiIiInIQd6iMoijy5+3bt8t7V4iIiOyiv2bpr2EVEa/fRESk1uu36oLurKws+TMmJqa8d4WIiKjE17CgoKAKedR4/SYiIrVevzWKym6rFxQU4NKlSwgICIBGoymVuxcigP/9998RGBgItWH5XBvPn2vj+XNtpXn+xKVYXLCjoqKg1VbMkV+lff2uqL93pY3Hi8eM3zPnw99L5zlm9l6/VdfSLQobHR1d6p8rTo6aL84sn2vj+XNtPH+urbTOX0Vt4Xb09bui/t6VNh4vHjN+z5wPfy+d45jZc/2umLfTiYiIiIiIiMoAg24iIiIiIiIiB2HQXQwvLy/MnDlT/lQjls+18fy5Np4/16b286dWPG88XvyOOR/+XvJ4qf07prpEakRERERERETOgi3dRERERERERA7CoJuIiIiIiIjIQRh0ExERERERETkIg24b3n33XVSvXh3e3t5ISUnBzp074Qp27NiBxx9/XE7SrtFo8Pnnn5u8Lobxz5o1S77u4+OD9PR0HDlyxGSb7OxsjBs3DuHh4fDz88MTTzyBCxcuoLy99tprSEtLQ0BAACpXrowePXogIyNDNeVbsmQJ6tevb5hDsHnz5vj2229VUTZr51N8RydOnKiaMop9F2UyXiIiIlRTPuHixYsYNGgQwsLC4Ovri4YNG+KXX35RRRnj4uKKnD+xjBkzxuXLRqbOnj2L4cOHy+u8OJcJCQkyyU5OTg4PlQ3z5s1DixYt5O9+cHAwj5VK6o7OWmelkteDyf56dVli0G3Fp59+KgOB6dOnY//+/XjkkUfQpUsXnD9/Hs7u7t27aNCgAd5++22Lr8+fPx+LFi2Sr//8888yIOjYsSOysrIM24iyr1+/HmvWrMGuXbtw584ddOvWDfn5+ShP33//vaz87tmzB5s3b0ZeXh46deoky6yG8kVHR+P111/H3r175dKuXTt0797dUKl35bKZE/u/fPly+cfQmBrKWKdOHVy+fNmwHDp0SDXlu3nzJlq2bAkPDw954Tp69CgWLlxoUvl25TKK/TU+d+LvjNC7d2+XLxuZOn78OAoKCrBs2TL5N/aNN97A0qVLMW3aNB4qG8RNCfH7MHr0aB4nFdUdnbXOSiWvB5P99eoyJbKXU1FNmjRRRo0aZbKuVq1aypQpU1zqcIlTvH79esPzgoICJSIiQnn99dcN6x48eKAEBQUpS5culc9v3bqleHh4KGvWrDFsc/HiRUWr1SobNmxQnMnVq1dlGb///ntVlk8ICQlR/v3vf6uqbFlZWUqNGjWUzZs3K23atFEmTJgg16uhjDNnzlQaNGhg8TU1lG/y5MlKq1atrL6uhjIaE9/NhIQEWS61lY2Kmj9/vlK9enUeGjusWLFCfvdJfXVHZ6mzUsnrwWR/vbqssaXbyl1c0VVS3DkyJp7/8MMPcGVnzpzBlStXTMom5qtr06aNoWyi7Lm5uSbbiG4/devWdbryZ2Zmyp+hoaGqK59oFRMtZeLupegOo6ayibu0Xbt2RYcOHUzWq6WMJ0+elPskuhj269cPp0+fVk35vvjiC6SmpsqWLtG1rVGjRnjvvfcMr6uhjMbXglWrVmHYsGGy26OaykbWryn66wlRSai57kjOy7weTPbXq8sag24Lrl+/Lk9MlSpVTNaL56LC5cr0+2+rbOKnp6cnQkJCrG7jDMRN0eeeew6tWrWSFVq1lE90Rfb395eV+VGjRsluqsnJyaoomyD+4O3bt0+OSzKnhjI2bdoUK1euxMaNG2UwKvZJjH/8888/VVE+cQNBjJGqUaOGLKP4jo4fP16WWVBDGfXE2MJbt25hyJAhqisbFXXq1CksXrxYfqeJSkrNdUdyTpbqwWR/vbqsMei2QbRsmH+5zddVpLI5W/nHjh2LgwcPYvXq1aoqX1JSEg4cOCDH64gxc4MHD5bjZtVQtt9//x0TJkyQrYciyYw1rlxGMX6vV69eqFevnmzJ//rrr+X6Dz/8UBXlE2NgGzdujFdffVW2cj/zzDMYOXKkDMSNuXIZ9d5//315PkVLtdrKpmaWkhmaL2Jsn7FLly6hc+fOsgfHiBEjUNH8lWNGFa/uSM7FVj2Y7K9XlxUG3RaIjLNubm5F7kxevXq1yB1MV6PPomyrbGIb0U1KJEyytk15E5mBRTfXbdu2ySQJaiqfaCVLTEyUXXhFa7BIMPLmm2+qomyi653YF5HR1d3dXS4iKchbb70lH+v30ZXLaE5krxYBuOhyroZzGBkZWeQOce3atQ2JgtRQRuHcuXPYsmWLSQCmlrJVhIrosWPHbC7GrUIi4G7btq3sbiiSO1ZEJT1mVLHqjuR8rNWDyf56dVlj0G3l5IigQJ+1Vk88F91EXZkYYyoqhcZlExVEEfjoyybKLjITG28jsvgePny43Msv7hiLysG6deuwdetWWR41lc9amcUURGooW/v27WU3H3HHUb+IP4IDBw6Uj+Pj412+jObEuRMVVhGsquEciszl5tOTnDhxArGxsfKxGsoorFixQo5ZF7kH9NRStooQ/NSqVcvmou9pI6a/E9O+id4b4pxrtRWzWlSSY0YVr+5IzqO4ejDZX68uc2Weus1FiMyzIgPt+++/rxw9elSZOHGi4ufnp5w9e1ZxdiIz9P79++UiTvGiRYvk43PnzsnXReZdkXF03bp1yqFDh5T+/fsrkZGRyu3btw2fIbJvRkdHK1u2bFH27duntGvXTmZkzsvLK8eSKcro0aPlvm/fvl25fPmyYbl3755hG1cu39SpU5UdO3YoZ86cUQ4ePKhMmzZNZj3etGmTy5fNGuPs5Woo4/PPPy+/n6dPn1b27NmjdOvWTQkICDD87XD18v3000+Ku7u7Mm/ePOXkyZPKxx9/rPj6+iqrVq0ybOPqZczPz1eqVasmM7Wbc/WykWKSVT4xMVGenwsXLphcU8g6UZcQdYrZs2cr/v7+hvqGqHtUdK5cd3TWOiuVvB5M9teryxKDbhveeecdJTY2VvH09FQaN27sMun4t23bJv9wmS+DBw+Wr4tpb8S0RmLqGy8vL6V169ay8mjs/v37ytixY5XQ0FDFx8dHBg7nz59XypulcolFTF2i58rlGzZsmOE7V6lSJaV9+/YmfxhcuWz2Bt2uXsa+ffvKIExUvKKiopSePXsqR44cUU35hC+//FKpW7eu3H8xHc7y5ctNXnf1Mm7cuFH+XcnIyCjymquXjQqJ64a1awpZJ+oSlo6ZqHuQ69YdnbXOSiWvB5P99eqypBH/lH37OhEREREREZH6VczBS0RERERERERlgEE3ERERERERkYMw6CYiIiIiIiJyEAbdRERERERERA7CoJuIiIiIiIjIQRh0ExERERERETkIg24iIiIiIiIiB2HQTUREREREROQgDLqJiIiIiCqInJwcJCYmYvfu3XBWaWlpWLduXXnvBlGpYdBNRERERORgs2bNQsOGDcv9OC9fvhyxsbFo2bIlnNXLL7+MKVOmoKCgoLx3hahUMOgmIiIiInISubm5Dv38xYsXY8SIESiLFvW/qmvXrsjMzMTGjRtLdZ+IyguDbiIiIiKiYqxcuRJhYWHIzs42Wd+rVy889dRTNt/7wQcfYPbs2fj111+h0WjkItYJ4vHSpUvRvXt3+Pn54ZVXXpGvBQcHm3zG559/Lrc19uWXXyIlJQXe3t6Ij4+X/0deXp7V/di3bx9+++03GdTqnT17Vn6u6M7dtm1b+Pr6okGDBvjxxx9N3rt27VrUqVMHXl5eiIuLw8KFC01eF+vEvg8ZMgRBQUEYOXKkoRxfffUVkpKS5Gc/+eSTuHv3Lj788EP5npCQEIwbNw75+fmGz3Jzc8Njjz2G1atX2zyuRK6CQTcRERERUTF69+4tA8MvvvjCsO769esyoBw6dKjN9/bt2xfPP/+8DFovX74sF7FOb+bMmTLoPnToEIYNG2bXuRCtwIMGDcL48eNx9OhRLFu2TAa58+bNs/qeHTt2oGbNmggMDCzy2vTp0/HCCy/gwIEDcpv+/fsbAvhffvkFffr0Qb9+/eQ+iq7yogu4/saB3oIFC1C3bl25vXhduHfvHt566y2sWbMGGzZswPbt29GzZ0988803cvnoo49kl/f//ve/Jp/VpEkT7Ny5065jQeTs3Mt7B4iIiIiInJ2Pjw8GDBiAFStWyABc+PjjjxEdHY309PRi3+vv7w93d3dEREQUeV18rr3Btp4IrsW458GDB8vnoqV77ty5ePHFF2UQb4lo1Y6KirL4mgi49S3gosVc3CAQreK1atXCokWL0L59e0MgLYJyEeiLIFu0bOu1a9dOfo7erl27ZHf5JUuWICEhQa4TLd0i0P7jjz/kMUlOTpYt7Nu2bTO5EVG1alWcP39ejuvWatlOSK6N32AiIiIiIjuILtObNm3CxYsX5XMRgIug07zbd0mlpqaW+D2iNXnOnDkycNUvYv9EK7poXbbk/v37siu6JfXr1zc8joyMlD+vXr0qfx47dqxI4jXx/OTJkybdwi2VQ3Qp1wfcQpUqVWS3crG/xuv0/5fxjQoRcJt35ydyRWzpJiIiIiKyQ6NGjeR4ZzG++9FHH5VdrcW46r9LjOU2Jlp2FUWxmWBNBKSiRVp01TZnLbAODw+X+2yJh4eH4bH+JoI+e7jYF/MbC+b7Z6kc5p+r/2xL68wzld+4cUMG7CL4JnJ1DLqJiIiIiOwkMn+/8cYbsrW7Q4cOiImJset9np6eJq3CtlSqVAlZWVky4Zg+kBVjrY01btwYGRkZcs7tktw0EF29LQXRtogu4KKruLEffvhBdjMXSc8c4fDhw7KMRGrA7uVERERERHYaOHCgDLjfe++9Eo3DFl2qz5w5I4NnkYDNVrfppk2bylbeadOmyXHVn3zySZGkZTNmzJAt7iKp2ZEjR2QX8E8//RQvvfSS1c8VY6dFIC+2LwmRBO67776TY8ZPnDghM4+//fbbJuO3S5tIotapUyeHfT5RWWLQTURERERkJ5H5W0wTJsYk9+jRw+7jJt7TuXNnGfiKlmxb02GFhoZi1apVMrt3vXr15LYiuDYmureLzOmbN29GWloamjVrJhOexcbGWv1cMeWZ6I4uEsCVhGhx/uyzz2QGcpGdXAT8Yjy5cRK10iRuaoiW9OKywhO5Co1iaUAGERERERFZ1LFjR9SuXVtOheVqxJhu0S1etKAHBATAGU2aNAmZmZlyKjEiNWBLNxERERGRHURyL9Hau3XrVowZM8Ylj5loOZ8/f76cPsxZVa5cWXZlJ1ILtnQTEREREdk5LvvmzZtyvmrz8cxiXutz585ZfN+yZcvkWHAiqpgYdBMRERER/U0i4Daf1st4Hmpn7cpNRI7HoJuIiIiIiIjIQTimm4iIiIiIiMhBGHQTEREREREROQiDbiIiIiIiIiIHYdBNRERERERE5CAMuomIiIiIiIgchEE3ERERERERkYMw6CYiIiIiIiJyEAbdRERERERERHCM/w8orMa0g/98EgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ===================== ASSIGNMENT DATASET =====================\n",
    "N_SEEDS = 10\n",
    "VARY_DATASET_SEED = False\n",
    "VARY_MODEL_INIT_SEED = True\n",
    "STRICT_IP_CHECK = False\n",
    "IP_CHECK_TOL = 1e-4\n",
    "SILENCE_LOCAL_SEARCH = False\n",
    "ALLOW_PLOTS_MULTI_SEED = True\n",
    "\n",
    "dataset_type = \"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",
    "runs = [\n",
    "    (\"DFN\",        \"DFN\", dfn_params),\n",
    "    (\"DFN_AfixI\",  \"DFN\", dfn_Afix_params),\n",
    "    (\"MLP\",        \"MLP\", mlp_params),\n",
    "    (\"MaxAffine\",  \"MaxAffine\", maff_params),\n",
    "    (\"LSET\",       \"LSET\", lset_params),\n",
    "]\n",
    "\n",
    "# ---- LEARNING (train once) ----\n",
    "learn_bundle, spec_df, learn_df, learn_fail_df, learn_summary_df = run_learning_benchmark(\n",
    "    dataset_type=dataset_type,\n",
    "    dataset_params=dataset_params,\n",
    "    runs=runs,\n",
    "    train_base=train_base,\n",
    "    lr_map=lr_map,\n",
    "    n_seeds=N_SEEDS,\n",
    "    vary_dataset_seed=VARY_DATASET_SEED,\n",
    "    vary_model_init_seed=VARY_MODEL_INIT_SEED,\n",
    "    allow_plots_multi_seed=ALLOW_PLOTS_MULTI_SEED,\n",
    ")\n",
    "\n",
    "# Convenient notebook-friendly displays:\n",
    "try:\n",
    "    display(spec_df)\n",
    "    display(learn_summary_df)\n",
    "except NameError:\n",
    "    pass\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4f13dfad",
   "metadata": {},
   "source": [
    "### Optimization (rerunnable)\n",
    "\n",
    "Run this cell as many times as you want with different constraints **without retraining**.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "92dbc0ee-bbe1-4fdd-8782-44ece09dda4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# 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",
    "# ---- OPTIMIZATION (rerunnable; does NOT retrain) ----\n",
    "# You can change xmin/xmax/x0/delta/sum_eq/time_limit below and rerun this cell.\n",
    "opt_df, opt_fail_df, opt_summary_df, gt_df, gap_seed_df = run_optimization_benchmark(\n",
    "    learn_bundle,\n",
    "    x0=x0, xmin=xmin, xmax=xmax,\n",
    "    delta=delta, sum_eq=sum_eq,\n",
    "    strict_ip_check=STRICT_IP_CHECK,\n",
    "    ip_check_tol=IP_CHECK_TOL,\n",
    "    silence_local_search=SILENCE_LOCAL_SEARCH,\n",
    "    time_limit=time_limit,\n",
    ")\n",
    "\n",
    "try:\n",
    "    display(opt_summary_df)\n",
    "except NameError:\n",
    "    pass\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:dfn]",
   "language": "python",
   "name": "conda-env-dfn-py"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
