{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "f5b14808",
   "metadata": {},
   "source": [
    "# DFN"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2efb5921",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "ce8269a4",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "\n",
    "import cppimport\n",
    "\n",
    "import sys, time, math, io, contextlib\n",
    "from pathlib import Path\n",
    "from typing import Optional, List, Tuple, Dict, Any\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linear_sum_assignment\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "from IPython.display import display\n",
    "\n",
    "import gurobipy as gp\n",
    "from gurobipy import GRB\n",
    "\n",
    "_TOL = 1e-9"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "417583e0",
   "metadata": {},
   "source": [
    "## LEMON"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "62767112",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "repo = Path().resolve().parent\n",
    "if str(repo) not in sys.path:\n",
    "    sys.path.insert(0, str(repo))\n",
    "\n",
    "lemon_mcf = cppimport.imp(\"lemon_mcf\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "15e6bd1b",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "88381d28",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'status': 1, 'flow': array([1., 2., 0.]), 'potential': array([-2.,  0., -1.]), 'reduced_cost': array([0., 0., 4.]), 'at_capacity': array([False,  True, False]), 'total_cost': 4.0}\n",
      "{'value': 4.0, 'flow': array([2., 2., 2., 2.])}\n"
     ]
    }
   ],
   "source": [
    "n = 3\n",
    "src    = np.array([0, 0, 1], dtype=np.int64)\n",
    "dst    = np.array([1, 2, 2], dtype=np.int64)\n",
    "cost   = np.array([2.0, 1.0, 3.0], dtype=np.float64)\n",
    "cap    = np.array([5.0, 2.0, 4.0], dtype=np.float64)\n",
    "supply = np.array([3.0, -1.0, -2.0], dtype=np.float64)\n",
    "\n",
    "out_min_cost_flow = lemon_mcf.solve_mcf(n, src, dst, cost, cap, supply)\n",
    "print(out_min_cost_flow)\n",
    "\n",
    "n = 4\n",
    "src = np.array([0,0,1,2], dtype=np.int64)\n",
    "dst = np.array([1,2,3,3], dtype=np.int64)\n",
    "cap = np.array([3.0,2.0,2.0,4.0], dtype=np.float64)\n",
    "out_max_flow = lemon_mcf.max_flow(n, src, dst, cap, 0, 3)\n",
    "print(out_max_flow)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "75235271",
   "metadata": {},
   "source": [
    "## Dataset generators"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "67763cc2",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def make_mdvsp_dataset(\n",
    "    K: int,\n",
    "    filename: str,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    max_trips=None,\n",
    "    max_succ=None,\n",
    "):\n",
    "    \"\"\"Multiple-Depot Vehicle Scheduling (MDVSP) dataset.\n",
    "\n",
    "    Returns:\n",
    "      X: (K, m) integer-ish capacities (float32)\n",
    "      y: (K,) min-cost-flow objective values (float32)\n",
    "      gt: dict containing the fixed network pieces (for later evaluation)\n",
    "    \"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "\n",
    "    with open(filename) as f:\n",
    "        m, n, l = map(int, f.readline().split())\n",
    "        f.readline()  # blank line\n",
    "        trips = np.loadtxt(f, max_rows=n, dtype=np.int64)[:max_trips]\n",
    "        D = np.loadtxt(f, max_rows=l, dtype=np.int64)\n",
    "\n",
    "    p, s, q, e = trips.T\n",
    "    ntr = len(trips)\n",
    "\n",
    "    # node ids\n",
    "    SS = 0\n",
    "    depS = 1 + np.arange(m)\n",
    "    depT = 1 + m + np.arange(m)\n",
    "    trS  = 1 + 2*m + np.arange(ntr)\n",
    "    trT  = 1 + 2*m + ntr + np.arange(ntr)\n",
    "    TT = 1 + 2*m + 2*ntr\n",
    "    N = TT + 1\n",
    "\n",
    "    src, dst, cost, cap = [], [], [], []\n",
    "\n",
    "    # depot boundary arcs (capacity is what we learn/provide per sample)\n",
    "    for d in range(m):\n",
    "        src += [SS, int(depT[d])]\n",
    "        dst += [int(depS[d]), TT]\n",
    "        cost += [0.0, 0.0]\n",
    "        cap  += [0.0, 0.0]\n",
    "    idxSS = np.arange(0, 2*m, 2)  # arcs SS->depS\n",
    "    idxTT = np.arange(1, 2*m, 2)  # arcs depT->TT\n",
    "\n",
    "    # trip arcs (always cap=1)\n",
    "    src += trS.tolist()\n",
    "    dst += trT.tolist()\n",
    "    cost += [0.0] * ntr\n",
    "    cap  += [1.0] * ntr\n",
    "\n",
    "    # depot-to-trip and trip-to-depot arcs\n",
    "    for d in range(m):\n",
    "        # depS -> trS\n",
    "        src += [int(depS[d])] * ntr\n",
    "        dst += trS.tolist()\n",
    "        cost += (5000 + 10 * D[d, p]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "        # trT -> depT\n",
    "        src += trT.tolist()\n",
    "        dst += [int(depT[d])] * ntr\n",
    "        cost += (5000 + 10 * D[q, d]).astype(float).tolist()\n",
    "        cap  += [1.0] * ntr\n",
    "\n",
    "    # feasible trip successor arcs (trT -> next trS)\n",
    "    order = np.argsort(s)\n",
    "    p2, s2 = p[order], s[order]\n",
    "    max_succ_eff = ntr if max_succ is None else int(max_succ)\n",
    "\n",
    "    for i in range(ntr):\n",
    "        travel = D[q[i], p2]                          # time from trip i end depot -> next trip start depot\n",
    "        feas = np.flatnonzero(s2 >= e[i] + travel)[:max_succ_eff]\n",
    "        j = order[feas]\n",
    "        if j.size:\n",
    "            src += [int(trT[i])] * j.size\n",
    "            dst += trS[j].tolist()\n",
    "            cost += (8 * travel[feas] + 2 * (s[j] - e[i])).astype(float).tolist()\n",
    "            cap  += [1.0] * j.size\n",
    "\n",
    "    src = np.asarray(src, dtype=np.int64)\n",
    "    dst = np.asarray(dst, dtype=np.int64)\n",
    "    cost = np.asarray(cost, dtype=np.float64)\n",
    "    cap0 = np.asarray(cap, dtype=np.float64)\n",
    "\n",
    "    # sample capacities X and compute y via max-flow + min-cost-flow\n",
    "    X = rng.integers(x_min, np.asarray(x_max) + 1, size=(K, m)).astype(np.float64)\n",
    "    y = np.empty(K, dtype=np.float64)\n",
    "\n",
    "    for k in range(K):\n",
    "        cap_k = cap0.copy()\n",
    "        cap_k[idxSS] = X[k]\n",
    "        cap_k[idxTT] = X[k]\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap_k, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, dtype=np.float64)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "\n",
    "        y[k] = lemon_mcf.solve_mcf(N, src, dst, cost, cap_k, supply)[\"total_cost\"]\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = dict(\n",
    "        type=\"mdvsp\", N=int(N), SS=int(SS), TT=int(TT),\n",
    "        src=src, dst=dst, cost=cost, cap0=cap0, idxSS=idxSS, idxTT=idxTT\n",
    "    )\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n",
    "\n",
    "\n",
    "def generate_bipartite_subset_matching_dataset(\n",
    "    K: int, num_nodes: int, c_min: int, c_max: int, noise_std: float = 0.0, seed: int = 0\n",
    "):\n",
    "    \"\"\"Assignment-style dataset: choose a subset of left nodes, match to right nodes with min cost.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    C = rng.integers(c_min, c_max + 1, size=(num_nodes, num_nodes)).astype(np.float32)\n",
    "\n",
    "    X = np.zeros((K, num_nodes), dtype=np.float32)\n",
    "    y = np.zeros((K,), dtype=np.float32)\n",
    "\n",
    "    for k in range(K):\n",
    "        mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        while not mask.any():\n",
    "            mask = rng.integers(0, 2, size=num_nodes, dtype=np.int8)\n",
    "        idx = np.flatnonzero(mask)\n",
    "        X[k, idx] = 1.0\n",
    "\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        y[k] = C[idx, :][r, c].sum()\n",
    "        if noise_std:\n",
    "            y[k] += noise_std * rng.normal()\n",
    "\n",
    "    gt = {\"type\": \"assignment\", \"C\": C.astype(np.float32)}\n",
    "    return X, y, gt\n",
    "\n",
    "\n",
    "def generate_convex_quadratic_dataset(\n",
    "    K: int,\n",
    "    dim: int,\n",
    "    eigen_min: float,\n",
    "    eigen_max: float,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    noise_std: float = 0.0,\n",
    "    seed: int = 0,\n",
    "    x_star_zero: bool = False,\n",
    "):\n",
    "    \"\"\"y = (x - x*)^T Q (x - x*) + noise, with Q symmetric PSD.\"\"\"\n",
    "    rng = np.random.default_rng(seed)\n",
    "    U, R = np.linalg.qr(rng.standard_normal((dim, dim)))\n",
    "    U *= np.sign(np.diag(R) + 1e-12)\n",
    "    Q = U @ np.diag(rng.uniform(eigen_min, eigen_max, dim)) @ U.T\n",
    "\n",
    "    x_star = np.zeros(dim, dtype=np.int64) if x_star_zero else rng.integers(x_min, x_max + 1, size=dim)\n",
    "\n",
    "    X = rng.integers(x_min, x_max + 1, size=(K, dim)).astype(np.float32)\n",
    "    d = X - x_star\n",
    "    y = np.einsum(\"bi,ij,bj->b\", d, Q, d) + noise_std * rng.normal(size=K)\n",
    "\n",
    "    gt = {\"type\": \"quadratic\", \"Q\": Q.astype(np.float32), \"x_star\": x_star.astype(np.int64)}\n",
    "    return X.astype(np.float32), y.astype(np.float32), gt\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1dfc62e4",
   "metadata": {},
   "source": [
    "### Quick sanity checks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "89201b4a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(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",
      " [489783.12 366182.16 447190.56 325780.94 376081.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-05.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",
    "import json, hashlib\n",
    "\n",
    "# ---- Caching / persistence helpers ----\n",
    "_CACHE_ROOT_DEFAULT = Path(\"saved_runs\")\n",
    "\n",
    "def _to_hashable(x):\n",
    "    \"\"\"Convert nested objects to a deterministic, hashable (and mostly JSON-friendly) form.\"\"\"\n",
    "    import numpy as _np\n",
    "    import torch as _torch\n",
    "    from pathlib import Path as _Path\n",
    "\n",
    "    if x is None or isinstance(x, (bool, int, str)):\n",
    "        return x\n",
    "    if isinstance(x, float):\n",
    "        return float(x)\n",
    "    if isinstance(x, _Path):\n",
    "        return str(x)\n",
    "    if isinstance(x, (list, tuple)):\n",
    "        return [_to_hashable(v) for v in x]\n",
    "    if isinstance(x, dict):\n",
    "        return {str(k): _to_hashable(x[k]) for k in sorted(x.keys(), key=lambda z: str(z))}\n",
    "    if isinstance(x, _np.ndarray):\n",
    "        h = hashlib.sha256(x.tobytes(order=\"C\")).hexdigest()\n",
    "        return {\"__ndarray__\": True, \"dtype\": str(x.dtype), \"shape\": list(x.shape), \"sha256\": h}\n",
    "    if isinstance(x, _torch.Tensor):\n",
    "        t = x.detach().cpu()\n",
    "        h = hashlib.sha256(t.numpy().tobytes(order=\"C\")).hexdigest()\n",
    "        return {\"__tensor__\": True, \"dtype\": str(t.dtype), \"shape\": list(t.shape), \"sha256\": h}\n",
    "    return {\"__repr__\": repr(x)}\n",
    "\n",
    "def _run_signature(dataset_type, dataset_params, model_type, model_params, train_sig):\n",
    "    return {\n",
    "        \"dataset_type\": str(dataset_type).lower(),\n",
    "        \"dataset_params\": _to_hashable(dict(dataset_params)),\n",
    "        \"model_type\": str(model_type),\n",
    "        \"model_params\": _to_hashable(dict(model_params)),\n",
    "        \"train_params\": _to_hashable(dict(train_sig)),\n",
    "    }\n",
    "\n",
    "def _run_id_from_signature(sig) -> str:\n",
    "    blob = json.dumps(sig, sort_keys=True, separators=(\",\", \":\"), ensure_ascii=True).encode(\"utf-8\")\n",
    "    return hashlib.sha256(blob).hexdigest()[:16]\n",
    "\n",
    "def _get_run_dir(cache_root: Path, dataset_type, model_type, run_id: str) -> Path:\n",
    "    return Path(cache_root) / str(dataset_type).lower() / str(model_type) / run_id\n",
    "\n",
    "def _cpuify(obj):\n",
    "    import torch as _torch\n",
    "    if isinstance(obj, _torch.Tensor):\n",
    "        return obj.detach().cpu()\n",
    "    if isinstance(obj, dict):\n",
    "        return {k: _cpuify(v) for k, v in obj.items()}\n",
    "    if isinstance(obj, (list, tuple)):\n",
    "        return [_cpuify(v) for v in obj]\n",
    "    return obj\n",
    "\n",
    "def _try_load_cached_run(run_dir: Path):\n",
    "    artifact_path = Path(run_dir) / \"artifact.pt\"\n",
    "    if not artifact_path.exists():\n",
    "        return None\n",
    "    try:\n",
    "        return torch.load(str(artifact_path), map_location=\"cpu\")\n",
    "    except Exception:\n",
    "        return None\n",
    "\n",
    "def _build_model_from_params(model_type, model_params):\n",
    "    mt = str(model_type)\n",
    "    mp = dict(model_params)\n",
    "    if mt == \"DFN\":\n",
    "        model = DFN(**mp)\n",
    "    else:\n",
    "        mp.pop(\"alpha\", None)\n",
    "        mp.pop(\"beta\", None)\n",
    "        if mt == \"MLP\":\n",
    "            model = MLP(**mp)\n",
    "        elif mt == \"MaxAffine\":\n",
    "            model = MaxAffine(**mp)\n",
    "        elif mt == \"LSET\":\n",
    "            model = LSET(**mp)\n",
    "        else:\n",
    "            raise ValueError(\"model_type must be: DFN | MLP | MaxAffine | LSET\")\n",
    "    return model\n",
    "\n",
    "def _save_run(run_dir: Path, signature: dict, model: nn.Module, data: dict, history: dict, spec: dict):\n",
    "    run_dir = Path(run_dir)\n",
    "    run_dir.mkdir(parents=True, exist_ok=True)\n",
    "    artifact_path = run_dir / \"artifact.pt\"\n",
    "    if artifact_path.exists():\n",
    "        return\n",
    "    state_dict = {k: v.detach().cpu() for k, v in model.state_dict().items()}\n",
    "    artifact = {\n",
    "        \"signature\": signature,\n",
    "        \"state_dict\": state_dict,\n",
    "        \"data\": _cpuify(data),\n",
    "        \"history\": history,\n",
    "        \"spec\": spec,\n",
    "    }\n",
    "    torch.save(artifact, str(artifact_path))\n",
    "    try:\n",
    "        with open(run_dir / \"signature.json\", \"w\", encoding=\"utf-8\") as f:\n",
    "            json.dump(signature, f, indent=2, sort_keys=True)\n",
    "    except Exception:\n",
    "        pass\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",
    "\n",
    "    # ---- cache check (skip training if identical run already saved) ----\n",
    "    train_sig = {\n",
    "        \"epochs\": int(epochs),\n",
    "        \"batch_size\": int(batch_sz),\n",
    "        \"lr\": float(lr),\n",
    "        \"weight_decay\": float(wd),\n",
    "        \"val_frac\": float(val_frac),\n",
    "        \"test_frac\": float(test_frac),\n",
    "        \"eps\": float(eps),\n",
    "        \"seed\": int(seed),\n",
    "    }\n",
    "    cache_root = Path(tp.get(\"cache_root\", _CACHE_ROOT_DEFAULT))\n",
    "    signature = _run_signature(dataset_type, dataset_params, model_type, model_params, train_sig)\n",
    "    run_id = _run_id_from_signature(signature)\n",
    "    run_dir = _get_run_dir(cache_root, dataset_type, model_type, run_id)\n",
    "\n",
    "    cached = _try_load_cached_run(run_dir)\n",
    "    if cached is not None and isinstance(cached, dict) and \"state_dict\" in cached and \"data\" in cached and \"history\" in cached and \"spec\" in cached:\n",
    "        model = _build_model_from_params(model_type, model_params)\n",
    "        model.load_state_dict(cached[\"state_dict\"])\n",
    "        model = model.to(device)\n",
    "        return model, cached[\"data\"], cached[\"history\"], cached[\"spec\"]\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",
    "\n",
    "    # ---- save artifacts for this run ----\n",
    "\n",
    "    _save_run(run_dir, signature, model, data, history, spec)\n",
    "\n",
    "\n",
    "    return model, data, history, spec\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ace7f63f",
   "metadata": {},
   "source": [
    "## Optimization Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "38416530",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def eval_true_obj(gt, x):\n",
    "    if not isinstance(gt, dict):\n",
    "        return np.nan\n",
    "\n",
    "    x = np.asarray(x).reshape(-1)\n",
    "    t = gt.get(\"type\", None)\n",
    "\n",
    "    if t == \"quadratic\":\n",
    "        Q = np.asarray(gt[\"Q\"], float)\n",
    "        xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "        d = x.astype(float) - xs\n",
    "        return float(d @ Q @ d)\n",
    "\n",
    "    if t == \"assignment\":\n",
    "        C = np.asarray(gt[\"C\"], float)\n",
    "        idx = np.flatnonzero(x > 0.5)\n",
    "        if idx.size == 0:\n",
    "            return 0.0\n",
    "        r, c = linear_sum_assignment(C[idx, :])\n",
    "        return float(C[idx, :][r, c].sum())\n",
    "\n",
    "    if t == \"mdvsp\":\n",
    "        N, SS, TT = int(gt[\"N\"]), int(gt[\"SS\"]), int(gt[\"TT\"])\n",
    "        src = np.asarray(gt[\"src\"], np.int64)\n",
    "        dst = np.asarray(gt[\"dst\"], np.int64)\n",
    "        cost = np.asarray(gt[\"cost\"], float)\n",
    "        cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "        idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "        idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "        cap = cap0.copy()\n",
    "        cap[idxSS] = x.astype(float)\n",
    "        cap[idxTT] = x.astype(float)\n",
    "\n",
    "        Fmax = lemon_mcf.max_flow(N, src, dst, cap, SS, TT)[\"value\"]\n",
    "        supply = np.zeros(N, float)\n",
    "        supply[SS] = Fmax\n",
    "        supply[TT] = -Fmax\n",
    "        return float(lemon_mcf.solve_mcf(N, src, dst, cost, cap, supply)[\"total_cost\"])\n",
    "\n",
    "    return np.nan\n",
    "\n",
    "\n",
    "def local_search_l1_int(\n",
    "    f,\n",
    "    x0,\n",
    "    x_min,\n",
    "    x_max,\n",
    "    delta: int,\n",
    "    sum_eq=None,\n",
    "    max_iters: int = 10_000,\n",
    "    print_every: int = 1,\n",
    "):\n",
    "    x  = np.asarray(x0,    int).ravel()\n",
    "    lo = np.asarray(x_min, int).ravel()\n",
    "    hi = np.asarray(x_max, int).ravel()\n",
    "    n = int(x.size)\n",
    "    delta = int(delta)\n",
    "\n",
    "    assert lo.size == n and hi.size == n\n",
    "    assert np.all(x >= lo) and np.all(x <= hi)\n",
    "    if sum_eq is not None:\n",
    "        sum_eq = int(sum_eq)\n",
    "        assert int(x.sum()) == sum_eq\n",
    "\n",
    "    def eval_batch(X):\n",
    "        y = np.asarray(f(X), float).reshape(-1)\n",
    "        return y\n",
    "\n",
    "    def ok(z):\n",
    "        if np.any(z < lo) or np.any(z > hi):\n",
    "            return False\n",
    "        if sum_eq is not None and int(z.sum()) != sum_eq:\n",
    "            return False\n",
    "        return True\n",
    "\n",
    "    # integer deltas with exact L1 = k\n",
    "    def deltas_exact(k):\n",
    "        d = np.zeros(n, int)\n",
    "\n",
    "        def rec(i, rem):\n",
    "            if i == n:\n",
    "                if rem == 0:\n",
    "                    yield d.copy()\n",
    "                return\n",
    "            for t in range(rem + 1):\n",
    "                if t == 0:\n",
    "                    d[i] = 0\n",
    "                    yield from rec(i + 1, rem)\n",
    "                else:\n",
    "                    d[i] = +t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "                    d[i] = -t\n",
    "                    yield from rec(i + 1, rem - t)\n",
    "            d[i] = 0\n",
    "\n",
    "        yield from rec(0, k)\n",
    "\n",
    "    t0 = time.perf_counter()\n",
    "    y = float(eval_batch(x[None, :])[0])\n",
    "    hist = [{\"iter\": 0, \"t\": 0.0, \"best_y\": y, \"x\": x.copy()}]\n",
    "    print(f\"iter=0  t=0.00s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    for it in range(1, int(max_iters) + 1):\n",
    "        cand = []\n",
    "        for k in range(1, delta + 1):\n",
    "            for dlt in deltas_exact(k):\n",
    "                z = x + dlt\n",
    "                if ok(z):\n",
    "                    cand.append(z)\n",
    "\n",
    "        if not cand:\n",
    "            print(f\"STOP: no feasible neighbors. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        Y = eval_batch(np.stack(cand, 0))\n",
    "        j = int(np.argmin(Y))\n",
    "        if float(Y[j]) >= y:\n",
    "            print(f\"STOP: local minimum. best_y={y:.6g} x={x.tolist()}\")\n",
    "            break\n",
    "\n",
    "        x, y = cand[j], float(Y[j])\n",
    "        hist.append({\"iter\": it, \"t\": time.perf_counter() - t0, \"best_y\": y, \"x\": x.copy()})\n",
    "        if it % max(1, int(print_every)) == 0:\n",
    "            print(f\"iter={it}  t={hist[-1]['t']:.2f}s  best_y={y:.6g}  x={x.tolist()}\")\n",
    "\n",
    "    return x, y, hist\n",
    "\n",
    "\n",
    "def _scaler_np(scaler):\n",
    "    xm = scaler[\"x_mean\"].detach().cpu().numpy().reshape(-1)\n",
    "    xs = scaler[\"x_std\"].detach().cpu().numpy().reshape(-1)\n",
    "    ym = float(scaler[\"y_mean\"].detach().cpu())\n",
    "    ys = float(scaler[\"y_std\"].detach().cpu())\n",
    "    return xm, xs, ym, ys\n",
    "\n",
    "\n",
    "def solve_dfn_ip_gurobi(dfn, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    x_mean, x_std, y_mean, y_std = _scaler_np(scaler)\n",
    "\n",
    "    cost = np.r_[dfn.cost_raw.detach().cpu().numpy(), dfn.cost_fixed.detach().cpu().numpy()]\n",
    "    cap  = np.r_[torch.round(F.softplus(dfn.cap_raw.detach())).cpu().numpy(), dfn.cap_fixed.detach().cpu().numpy()]\n",
    "\n",
    "    A = dfn.A.detach().cpu().numpy()\n",
    "    if isinstance(dfn.A, torch.nn.Parameter):\n",
    "        A = np.round(A)\n",
    "    b = np.round(dfn.b_raw.detach().cpu().numpy())\n",
    "\n",
    "    src = dfn.src.detach().cpu().numpy().astype(int)\n",
    "    dst = dfn.dst.detach().cpu().numpy().astype(int)\n",
    "    boundary = dfn.boundary.detach().cpu().numpy().astype(int)\n",
    "    fix = int(dfn.fix_node)\n",
    "    n = int(dfn.n)\n",
    "    m = int(src.size)\n",
    "    alpha = float(dfn.alpha)\n",
    "    beta  = float(dfn.beta)\n",
    "\n",
    "    out = [[] for _ in range(n)]\n",
    "    inn = [[] for _ in range(n)]\n",
    "    for e in range(m):\n",
    "        out[src[e]].append(e)\n",
    "        inn[dst[e]].append(e)\n",
    "\n",
    "    M = gp.Model(\"DFN_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    f = M.addVars(m, lb=0.0, ub=cap.tolist(), vtype=GRB.CONTINUOUS, name=\"f\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq)\n",
    "\n",
    "    xm_over_xs = x_mean / x_std\n",
    "    s = [0] * n\n",
    "    s_boundary = []\n",
    "    for r, v in enumerate(boundary):\n",
    "        const = float(b[r] - (A[r] * xm_over_xs).sum())\n",
    "        expr = const + gp.quicksum((A[r, j] / x_std[j]) * x[j] for j in range(d) if A[r, j] != 0)\n",
    "        s[v] = expr\n",
    "        s_boundary.append(expr)\n",
    "    s[fix] = -gp.quicksum(s_boundary)\n",
    "\n",
    "    for v in range(n):\n",
    "        M.addConstr(gp.quicksum(f[e] for e in out[v]) - gp.quicksum(f[e] for e in inn[v]) == s[v])\n",
    "\n",
    "    flow_cost = gp.quicksum(cost[e] * f[e] for e in range(m))\n",
    "    M.setObjective((alpha * flow_cost + beta) * y_std + y_mean, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No solution (Gurobi status {M.Status})\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None)}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_mlp_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    if not hasattr(base, \"net\"):\n",
    "        raise ValueError(\"Expected an MLP with attribute .net (nn.Sequential).\")\n",
    "    linears = [L for L in base.net if isinstance(L, torch.nn.Linear)]\n",
    "    if not linears:\n",
    "        raise ValueError(\"No Linear layers found in base.net\")\n",
    "\n",
    "    W = [L.weight.detach().cpu().numpy().astype(float) for L in linears]\n",
    "    b = [L.bias.detach().cpu().numpy().astype(float) for L in linears]\n",
    "\n",
    "    W[0] = W[0] / xs[None, :]\n",
    "    b[0] = b[0] - W[0] @ xm\n",
    "\n",
    "    W[-1] *= a_out\n",
    "    b[-1] = a_out * b[-1] + b_out\n",
    "\n",
    "    u = np.maximum(np.abs(x_min), np.abs(x_max))\n",
    "    preLU = []\n",
    "    for k in range(len(W) - 1):\n",
    "        U = np.abs(W[k]) @ u + np.abs(b[k])\n",
    "        preLU.append((-U, U))\n",
    "        u = np.maximum(0.0, U)\n",
    "\n",
    "    M = gp.Model(\"MLP_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    prev = [x[i] for i in range(d)]\n",
    "\n",
    "    for k in range(len(W) - 1):\n",
    "        Lk, Uk = preLU[k]\n",
    "        h = W[k].shape[0]\n",
    "\n",
    "        a = [M.addVar(lb=float(Lk[j]), ub=float(Uk[j]), name=f\"a{k}_{j}\") for j in range(h)]\n",
    "        z = [M.addVar(lb=0.0, ub=float(max(0.0, Uk[j])), name=f\"z{k}_{j}\") for j in range(h)]\n",
    "\n",
    "        for j in range(h):\n",
    "            M.addConstr(a[j] == b[k][j] + gp.quicksum(W[k][j, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "            Lj, Uj = float(Lk[j]), float(Uk[j])\n",
    "            if Uj <= 0.0:\n",
    "                M.addConstr(z[j] == 0.0)\n",
    "            elif Lj >= 0.0:\n",
    "                M.addConstr(z[j] == a[j])\n",
    "            else:\n",
    "                s = M.addVar(vtype=GRB.BINARY, name=f\"s{k}_{j}\")\n",
    "                M.addConstr(z[j] >= a[j])\n",
    "                M.addConstr(z[j] >= 0.0)\n",
    "                M.addConstr(z[j] <= Uj * s)\n",
    "                M.addConstr(z[j] <= a[j] - Lj * (1 - s))\n",
    "\n",
    "        prev = z\n",
    "\n",
    "    if W[-1].shape[0] != 1:\n",
    "        raise ValueError(f\"Expected scalar output, got out_dim={W[-1].shape[0]}\")\n",
    "\n",
    "    y_norm = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == b[-1][0] + gp.quicksum(W[-1][0, i] * prev[i] for i in range(len(prev))))\n",
    "\n",
    "    M.setObjective(ys * y_norm + ym, GRB.MINIMIZE)\n",
    "\n",
    "    M.optimize()\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_maxaffine_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    base, a_out, b_out = model, 1.0, 0.0\n",
    "    if hasattr(model, \"base\") and hasattr(model, \"a\") and hasattr(model, \"b\"):\n",
    "        base, a_out, b_out = model.base, float(model.a), float(model.b)\n",
    "\n",
    "    W = base.W.detach().cpu().numpy().astype(float)\n",
    "    b = base.b.detach().cpu().numpy().astype(float)\n",
    "\n",
    "    Weff = W / xs[None, :]\n",
    "    beff = b - (Weff @ xm)\n",
    "\n",
    "    Weff *= a_out\n",
    "    beff  = a_out * beff + b_out\n",
    "\n",
    "    K = int(Weff.shape[0])\n",
    "\n",
    "    M = gp.Model(\"MaxAffine_IP\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    t = M.addVar(lb=-GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"t_norm\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(t >= beff[k] + gp.quicksum(Weff[k, j] * x[j] for j in range(d) if Weff[k, j] != 0.0))\n",
    "\n",
    "    M.setObjective(ys * t + ym, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], float)\n",
    "    info = {\"status\": M.Status, \"runtime\": M.Runtime, \"gap\": getattr(M, \"MIPGap\", None), \"sol_count\": M.SolCount}\n",
    "    return x_star, float(M.ObjVal), info\n",
    "\n",
    "\n",
    "def solve_lset_ip_gurobi(model, scaler, x_min, x_max, sum_eq, *, integer_x=True, verbose=False, time_limit=None):\n",
    "    import gurobipy as gp\n",
    "    from gurobipy import GRB\n",
    "\n",
    "    x_min = np.asarray(x_min, float).ravel()\n",
    "    x_max = np.asarray(x_max, float).ravel()\n",
    "    d = int(x_min.size)\n",
    "    sum_eq = float(sum_eq)\n",
    "\n",
    "    xm, xs, ym, ys = _scaler_np(scaler)\n",
    "\n",
    "    A = model.A.detach().cpu().numpy().astype(float)\n",
    "    b = model.b.detach().cpu().numpy().astype(float)\n",
    "    T = float(model.T)\n",
    "    if T == 0.0:\n",
    "        raise ValueError(\"T must be nonzero\")\n",
    "\n",
    "    Aeff = A / xs[None, :]\n",
    "    beff = b - (Aeff @ xm)\n",
    "    K = int(Aeff.shape[0])\n",
    "\n",
    "    lin_lo = np.empty(K, dtype=float)\n",
    "    lin_hi = np.empty(K, dtype=float)\n",
    "    for k in range(K):\n",
    "        a = Aeff[k]\n",
    "        lo = beff[k]\n",
    "        hi = beff[k]\n",
    "        pos = a >= 0\n",
    "        lo += (a[pos] * x_min[pos]).sum() + (a[~pos] * x_max[~pos]).sum()\n",
    "        hi += (a[pos] * x_max[pos]).sum() + (a[~pos] * x_min[~pos]).sum()\n",
    "        lin_lo[k], lin_hi[k] = lo, hi\n",
    "\n",
    "    z_lo = lin_lo / T\n",
    "    z_hi = lin_hi / T\n",
    "    m_lo = float(np.max(z_lo))\n",
    "    m_hi = float(np.max(z_hi))\n",
    "\n",
    "    w_lo = float(np.min(z_lo - m_hi))  # <= 0\n",
    "    u_lo = float(np.exp(max(-700.0, w_lo)))\n",
    "    s_lo = max(1e-12, K * u_lo)\n",
    "    s_hi = float(K * 1.0)\n",
    "    v_lo = float(np.log(s_lo))\n",
    "    v_hi = float(np.log(s_hi))\n",
    "\n",
    "    yN_lo = float(T * (m_lo + v_lo))\n",
    "    yN_hi = float(T * (m_hi + v_hi))\n",
    "    y_lo  = float(ys * yN_lo + ym)\n",
    "    y_hi  = float(ys * yN_hi + ym)\n",
    "\n",
    "    M = gp.Model(\"lset_ip_stable_bounded\")\n",
    "    M.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        M.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    M.Params.FuncNonlinear = 1\n",
    "    M.Params.FeasibilityTol = 1e-9\n",
    "    M.Params.OptimalityTol  = 1e-9\n",
    "    M.Params.IntFeasTol     = 1e-9\n",
    "    M.Params.NumericFocus   = 3\n",
    "\n",
    "    xt = GRB.INTEGER if integer_x else GRB.CONTINUOUS\n",
    "    x = M.addVars(d, lb=x_min.tolist(), ub=x_max.tolist(), vtype=xt, name=\"x\")\n",
    "    M.addConstr(gp.quicksum(x[i] for i in range(d)) == sum_eq, name=\"sum_eq\")\n",
    "\n",
    "    z = M.addVars(K, lb=z_lo.tolist(), ub=z_hi.tolist(), vtype=GRB.CONTINUOUS, name=\"z\")\n",
    "    for k in range(K):\n",
    "        lin = beff[k] + gp.quicksum(Aeff[k, j] * x[j] for j in range(d) if Aeff[k, j] != 0.0)\n",
    "        M.addConstr(z[k] == lin / T, name=f\"zdef_{k}\")\n",
    "\n",
    "    m = M.addVar(lb=m_lo, ub=m_hi, vtype=GRB.CONTINUOUS, name=\"m\")\n",
    "    w = M.addVars(K, lb=w_lo, ub=0.0, vtype=GRB.CONTINUOUS, name=\"w\")\n",
    "    for k in range(K):\n",
    "        M.addConstr(m >= z[k], name=f\"m_ge_z_{k}\")\n",
    "        M.addConstr(w[k] == z[k] - m, name=f\"wdef_{k}\")\n",
    "\n",
    "    u = M.addVars(K, lb=0.0, ub=1.0, vtype=GRB.CONTINUOUS, name=\"u\")\n",
    "    for k in range(K):\n",
    "        M.addGenConstrExp(w[k], u[k], name=f\"exp_{k}\")\n",
    "\n",
    "    s = M.addVar(lb=s_lo, ub=s_hi, vtype=GRB.CONTINUOUS, name=\"s\")\n",
    "    M.addConstr(s == gp.quicksum(u[k] for k in range(K)), name=\"sumexp_shifted\")\n",
    "\n",
    "    v = M.addVar(lb=v_lo, ub=v_hi, vtype=GRB.CONTINUOUS, name=\"v\")\n",
    "    M.addGenConstrLog(s, v, name=\"log_shifted\")\n",
    "\n",
    "    y_norm = M.addVar(lb=yN_lo, ub=yN_hi, vtype=GRB.CONTINUOUS, name=\"y_norm\")\n",
    "    M.addConstr(y_norm == T * (m + v), name=\"y_norm_def\")\n",
    "\n",
    "    y_raw = M.addVar(lb=y_lo, ub=y_hi, vtype=GRB.CONTINUOUS, name=\"y_raw\")\n",
    "    M.addConstr(y_raw == ys * y_norm + ym, name=\"y_raw_def\")\n",
    "\n",
    "    M.setObjective(y_raw, GRB.MINIMIZE)\n",
    "    M.optimize()\n",
    "\n",
    "    if M.SolCount == 0:\n",
    "        raise RuntimeError(f\"No feasible solution found. Gurobi status {M.Status}\")\n",
    "\n",
    "    x_star = np.array([x[i].X for i in range(d)], dtype=float)\n",
    "    y_star = float(y_raw.X)\n",
    "\n",
    "    info = {\n",
    "        \"status\": M.Status,\n",
    "        \"runtime\": M.Runtime,\n",
    "        \"gap\": getattr(M, \"MIPGap\", None),\n",
    "        \"sol_count\": M.SolCount,\n",
    "        \"obj_gurobi\": float(M.ObjVal),\n",
    "        \"obj_bound\": float(getattr(M, \"ObjBound\", float(\"nan\"))),\n",
    "    }\n",
    "    return x_star, y_star, info\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d4858979",
   "metadata": {},
   "source": [
    "## Test Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "569f2165",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    }
   },
   "outputs": [],
   "source": [
    "def solve_ip(model_type, model, scaler, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    if model_type == \"DFN\":\n",
    "        return solve_dfn_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MLP\":\n",
    "        return solve_mlp_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"MaxAffine\":\n",
    "        return solve_maxaffine_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if model_type == \"LSET\":\n",
    "        return solve_lset_ip_gurobi(model, scaler, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise ValueError(model_type)\n",
    "\n",
    "\n",
    "def suppress_stdout(fn, *args, silence: bool = False, **kwargs):\n",
    "    if not silence:\n",
    "        return fn(*args, **kwargs), \"\"\n",
    "    buf = io.StringIO()\n",
    "    with contextlib.redirect_stdout(buf):\n",
    "        out = fn(*args, **kwargs)\n",
    "    return out, buf.getvalue()\n",
    "\n",
    "\n",
    "def make_obj(model, scaler, device, chunk: int = 4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    @torch.no_grad()\n",
    "    def obj(Xraw):\n",
    "        Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "        if Xraw.dim() == 1:\n",
    "            Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "        outs = []\n",
    "        B = int(Xraw.shape[0])\n",
    "        for i in range(0, B, int(chunk)):\n",
    "            Xb = Xraw[i:i+chunk]\n",
    "            try:\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)  \n",
    "                y  = (yn * ys + ym).reshape(-1).detach().cpu() \n",
    "                if y.numel() != Xb.shape[0]:\n",
    "                    y = y.expand(Xb.shape[0]).contiguous()\n",
    "                outs.append(y)\n",
    "            except Exception as e:\n",
    "                obj._err_count += int(Xb.shape[0])\n",
    "                if obj._err_first is None:\n",
    "                    obj._err_first = repr(e)\n",
    "                outs.append(torch.full((int(Xb.shape[0]),), float(\"inf\"), device=\"cpu\"))\n",
    "\n",
    "        return torch.cat(outs, 0).numpy()\n",
    "\n",
    "    obj._err_count = 0\n",
    "    obj._err_first = None\n",
    "    return obj\n",
    "\n",
    "\n",
    "def safe_raw_mse(obj, Xraw, yraw):\n",
    "    yp = np.asarray(obj(Xraw), dtype=float).reshape(-1)\n",
    "    yt = yraw.detach().cpu().numpy().reshape(-1) if torch.is_tensor(yraw) else np.asarray(yraw, float).reshape(-1)\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "    mask = np.isfinite(yp)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions were non-finite (inf/nan)\"\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def safe_norm_mse(model, scaler, device, Xraw, yraw, *, chunk=4096):\n",
    "    xm = scaler[\"x_mean\"].to(device)\n",
    "    xs = scaler[\"x_std\"].to(device)\n",
    "    ym = scaler[\"y_mean\"].to(device)\n",
    "    ys = scaler[\"y_std\"].to(device)\n",
    "\n",
    "    Xraw = torch.as_tensor(Xraw, dtype=torch.float32, device=device)\n",
    "    if Xraw.dim() == 1:\n",
    "        Xraw = Xraw.unsqueeze(0)\n",
    "\n",
    "    yraw_t = yraw\n",
    "    if torch.is_tensor(yraw_t):\n",
    "        yraw_t = yraw_t.to(device=device, dtype=torch.float32)\n",
    "    else:\n",
    "        yraw_t = torch.as_tensor(yraw_t, dtype=torch.float32, device=device)\n",
    "    yraw_t = yraw_t.reshape(-1)\n",
    "\n",
    "    preds = []\n",
    "    B = int(Xraw.shape[0])\n",
    "    try:\n",
    "        with torch.no_grad():\n",
    "            for i in range(0, B, int(chunk)):\n",
    "                Xb = Xraw[i:i+chunk]\n",
    "                Xn = (Xb - xm) / xs\n",
    "                yn = model(Xn)\n",
    "                yn = torch.as_tensor(yn).reshape(-1)\n",
    "                if yn.numel() != Xb.shape[0]:\n",
    "                    yn = yn.expand(Xb.shape[0]).contiguous()\n",
    "                preds.append(yn.detach().cpu())\n",
    "    except Exception as e:\n",
    "        return np.nan, f\"model forward failed in safe_norm_mse: {repr(e)}\"\n",
    "\n",
    "    yp = torch.cat(preds, 0).numpy().reshape(-1)\n",
    "    yt = ((yraw_t - ym) / ys).detach().cpu().numpy().reshape(-1)\n",
    "\n",
    "    if yp.shape[0] != yt.shape[0]:\n",
    "        return np.nan, f\"shape mismatch: pred {yp.shape} vs true {yt.shape}\"\n",
    "\n",
    "    mask = np.isfinite(yp) & np.isfinite(yt)\n",
    "    if mask.sum() == 0:\n",
    "        return np.nan, \"all predictions or targets were non-finite (inf/nan)\"\n",
    "\n",
    "    return float(np.mean((yp[mask] - yt[mask])**2)), None\n",
    "\n",
    "\n",
    "def t_to_best(hist, y_best):\n",
    "    for r in hist:\n",
    "        if abs(float(r.get(\"best_y\", np.inf)) - float(y_best)) < 1e-12:\n",
    "            return float(r.get(\"t\", float(\"nan\")))\n",
    "    return float(\"nan\")\n",
    "\n",
    "\n",
    "def check_ip_matches_obj(name, obj, x_ip, y_ip, *, strict: bool, tol: float):\n",
    "    y_obj = float(np.asarray(obj(np.asarray(x_ip)), dtype=float).reshape(-1)[0])\n",
    "    y_ip  = float(y_ip)\n",
    "    rel = abs(y_obj - y_ip) / (abs(y_obj) + 1e-12)\n",
    "    print(f\"[CHECK {name}] obj(x_ip)={y_obj:.6g}  ip_y={y_ip:.6g}  rel_err={rel:.3e}\")\n",
    "    if strict and rel > tol:\n",
    "        raise RuntimeError(f\"{name}: IP objective != obj() (rel_err={rel:.3e})\")\n",
    "    return rel\n",
    "\n",
    "\n",
    "def mean_se(x):\n",
    "    x = pd.to_numeric(pd.Series(x), errors=\"coerce\").to_numpy()\n",
    "    x = x[np.isfinite(x)]\n",
    "    n = int(x.shape[0])\n",
    "    if n == 0:\n",
    "        return np.nan, np.nan\n",
    "    m = float(x.mean())\n",
    "    se = float(x.std(ddof=1) / np.sqrt(n)) if n > 1 else 0.0\n",
    "    return m, se\n",
    "\n",
    "\n",
    "def fmt_mean_se(m, se):\n",
    "    if not np.isfinite(m):\n",
    "        return \"nan\"\n",
    "    if not np.isfinite(se):\n",
    "        return f\"{m:.6g}\"\n",
    "    return f\"{m:.6g} ± {se:.3g}\"\n",
    "\n",
    "\n",
    "def repr_solution(xs: pd.Series, seeds=None):\n",
    "    xs = xs.dropna().astype(str)\n",
    "    xs = xs[xs != \"None\"]\n",
    "    if xs.empty:\n",
    "        return None\n",
    "\n",
    "    # With seed information: show per-seed if solutions differ\n",
    "    if seeds is not None:\n",
    "        df = pd.DataFrame({\"seed\": pd.Series(seeds), \"x\": xs})\n",
    "        df = df.dropna()\n",
    "        df[\"x\"] = df[\"x\"].astype(str)\n",
    "        if df.empty:\n",
    "            return None\n",
    "        if df[\"x\"].nunique() == 1:\n",
    "            return df[\"x\"].iloc[0]\n",
    "        df = df.sort_values(\"seed\")\n",
    "        return \"\\n\".join([f\"seed={int(r.seed)}: {r.x}\" for r in df.itertuples(index=False)])\n",
    "\n",
    "    # No seed info: keep compact\n",
    "    if xs.nunique() == 1:\n",
    "        return xs.iloc[0]\n",
    "    vc = xs.value_counts()\n",
    "    top = vc.index[0]\n",
    "    n_unique = int(vc.shape[0])\n",
    "    return f\"{top} (+{n_unique-1} other)\"\n",
    "    \n",
    "\n",
    "# -----------------------------\n",
    "# Ground-truth optimum solvers\n",
    "# -----------------------------\n",
    "\n",
    "def solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    Q  = np.asarray(gt[\"Q\"], float)\n",
    "    xs = np.asarray(gt[\"x_star\"], float).reshape(-1)\n",
    "    n  = int(xs.shape[0])\n",
    "    assert Q.shape == (n, n)\n",
    "\n",
    "    m = gp.Model(\"gt_quadratic\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    x = m.addVars(n, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(n):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(n)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    expr = gp.quicksum(float(Q[i, j]) * (x[i] - float(xs[i])) * (x[j] - float(xs[j])) for i in range(n) for j in range(n))\n",
    "    m.setObjective(expr, GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT quadratic solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(n)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_assignment(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    C = np.asarray(gt[\"C\"], float)\n",
    "    n_rows, n_cols = C.shape\n",
    "    k = int(sum_eq)\n",
    "    if k > n_cols:\n",
    "        raise RuntimeError(f\"sum_eq={k} exceeds number of columns={n_cols} (infeasible)\")\n",
    "\n",
    "    m = gp.Model(\"gt_assignment\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    z = m.addVars(n_rows, vtype=GRB.BINARY, name=\"z\")  # select row i\n",
    "    y = m.addVars(n_rows, n_cols, vtype=GRB.BINARY, name=\"y\")  # assign row i to col j\n",
    "\n",
    "    m.addConstr(gp.quicksum(z[i] for i in range(n_rows)) == k, name=\"k_rows\")\n",
    "    for i in range(n_rows):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for j in range(n_cols)) == z[i], name=f\"assign_row_{i}\")\n",
    "    for j in range(n_cols):\n",
    "        m.addConstr(gp.quicksum(y[i, j] for i in range(n_rows)) <= 1, name=f\"assign_col_{j}\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(C[i, j]) * y[i, j] for i in range(n_rows) for j in range(n_cols)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT assignment solve failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(z[i].X)) for i in range(n_rows)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status)}\n",
    "\n",
    "\n",
    "def solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    N  = int(gt[\"N\"])\n",
    "    SS = int(gt[\"SS\"])\n",
    "    TT = int(gt[\"TT\"])\n",
    "    src  = np.asarray(gt[\"src\"], np.int64)\n",
    "    dst  = np.asarray(gt[\"dst\"], np.int64)\n",
    "    cost = np.asarray(gt[\"cost\"], float)\n",
    "    cap0 = np.asarray(gt[\"cap0\"], float)\n",
    "    idxSS = np.asarray(gt[\"idxSS\"], np.int64)\n",
    "    idxTT = np.asarray(gt[\"idxTT\"], np.int64)\n",
    "\n",
    "    E = int(src.shape[0])\n",
    "    k = int(idxSS.shape[0])\n",
    "    assert idxTT.shape[0] == k, \"Expect idxSS and idxTT to be same length\"\n",
    "\n",
    "    out_edges = [[] for _ in range(N)]\n",
    "    in_edges  = [[] for _ in range(N)]\n",
    "    for e in range(E):\n",
    "        out_edges[int(src[e])].append(e)\n",
    "        in_edges[int(dst[e])].append(e)\n",
    "\n",
    "    idxSS = idxSS.astype(int)\n",
    "    idxTT = idxTT.astype(int)\n",
    "    var_arc_to_i = {int(idxSS[i]): i for i in range(k)}\n",
    "    var_arc_to_i.update({int(idxTT[i]): i for i in range(k)})\n",
    "    var_arcs = sorted(var_arc_to_i.keys())\n",
    "\n",
    "    m = gp.Model(\"gt_mdvsp_true_opt\")\n",
    "    m.Params.OutputFlag = 1 if verbose else 0\n",
    "    if time_limit is not None:\n",
    "        m.Params.TimeLimit = float(time_limit)\n",
    "\n",
    "    m.Params.NonConvex = 2\n",
    "\n",
    "    x = m.addVars(k, vtype=GRB.INTEGER, name=\"x\")\n",
    "    for i in range(k):\n",
    "        x[i].LB = int(xmin[i])\n",
    "        x[i].UB = int(xmax[i])\n",
    "    m.addConstr(gp.quicksum(x[i] for i in range(k)) == int(sum_eq), name=\"sum_eq\")\n",
    "\n",
    "    f = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, name=\"f\")\n",
    "\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            m.addConstr(f[e] <= x[i], name=f\"cap_var_{e}\")\n",
    "        else:\n",
    "            m.addConstr(f[e] <= float(cap0[e]), name=f\"cap_fix_{e}\")\n",
    "\n",
    "    F = m.addVar(vtype=GRB.CONTINUOUS, lb=0.0, name=\"F\")\n",
    "\n",
    "    # flow conservation: out - in = F at SS, = -F at TT, else 0\n",
    "    for v in range(N):\n",
    "        out_sum = gp.quicksum(f[e] for e in out_edges[v])\n",
    "        in_sum  = gp.quicksum(f[e] for e in in_edges[v])\n",
    "        rhs = F if v == SS else (-F if v == TT else 0.0)\n",
    "        m.addConstr(out_sum - in_sum == rhs, name=f\"flow_{v}\")\n",
    "\n",
    "    y = m.addVars(N, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"y\")\n",
    "    m.addConstr(y[SS] == 1.0, name=\"ySS\")\n",
    "    m.addConstr(y[TT] == 0.0, name=\"yTT\")\n",
    "\n",
    "    z = m.addVars(E, vtype=GRB.CONTINUOUS, lb=0.0, ub=1.0, name=\"z\")\n",
    "    for e in range(E):\n",
    "        m.addConstr(z[e] >= y[int(src[e])] - y[int(dst[e])], name=f\"dual_{e}\")\n",
    "\n",
    "    dual_obj = gp.QuadExpr()\n",
    "    for e in range(E):\n",
    "        if e in var_arc_to_i:\n",
    "            i = var_arc_to_i[e]\n",
    "            # bilinear term: x_i * z_e\n",
    "            dual_obj += x[i] * z[e]\n",
    "        else:\n",
    "            dual_obj += float(cap0[e]) * z[e]\n",
    "    m.addConstr(F == dual_obj, name=\"strong_duality\")\n",
    "\n",
    "    m.setObjective(gp.quicksum(float(cost[e]) * f[e] for e in range(E)), GRB.MINIMIZE)\n",
    "    m.optimize()\n",
    "\n",
    "    if m.Status not in (GRB.OPTIMAL, GRB.TIME_LIMIT, GRB.SUBOPTIMAL, GRB.INTERRUPTED):\n",
    "        raise RuntimeError(f\"GT mdvsp true-opt failed, status={m.Status}\")\n",
    "\n",
    "    xsol = np.array([int(round(x[i].X)) for i in range(k)], dtype=int)\n",
    "    return xsol, float(m.ObjVal), {\"status\": int(m.Status), \"F\": float(F.X)}\n",
    "    \n",
    "\n",
    "\n",
    "def solve_true_opt(gt, xmin, xmax, sum_eq, *, time_limit=None, verbose=False):\n",
    "    \"\"\"Top-level dispatcher for ground-truth optimum.\"\"\"\n",
    "    if not isinstance(gt, dict):\n",
    "        raise RuntimeError(\"No ground-truth structure found (gt must be a dict).\")\n",
    "    t = gt.get(\"type\", None)\n",
    "    if t == \"quadratic\":\n",
    "        return solve_true_opt_quadratic(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"assignment\":\n",
    "        return solve_true_opt_assignment(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    if t == \"mdvsp\":\n",
    "        return solve_true_opt_mdvsp(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=verbose)\n",
    "    raise RuntimeError(f\"Unknown gt type: {t}\")\n",
    "\n",
    "\n",
    "# -----------------------------\n",
    "# Benchmark runner\n",
    "# -----------------------------\n",
    "def run_benchmark(\n",
    "    *,\n",
    "    dataset_type: str,\n",
    "    dataset_params: dict,\n",
    "    runs: list[tuple[str, str, dict]],\n",
    "    train_base: dict,\n",
    "    lr_map: dict,\n",
    "    x0: np.ndarray,\n",
    "    xmin: np.ndarray,\n",
    "    xmax: np.ndarray,\n",
    "    delta: int,\n",
    "    sum_eq: int,\n",
    "    n_seeds: int = 1,\n",
    "    vary_dataset_seed: bool = False,\n",
    "    vary_model_init_seed: bool = True,\n",
    "    strict_ip_check: bool = False,\n",
    "    ip_check_tol: float = 1e-4,\n",
    "    silence_local_search: bool = False,\n",
    "    allow_plots_multi_seed: bool = True,\n",
    "    time_limit=None,\n",
    "):\n",
    "    seeds = list(range(int(n_seeds)))\n",
    "\n",
    "    learn_rows, opt_rows, spec_rows, fail_rows = [], [], [], []\n",
    "    gt_rows = []\n",
    "\n",
    "    gt_cache_by_seed = {}\n",
    "\n",
    "    for seed in seeds:\n",
    "        print(f\"\\n\\n===================== SEED {seed} =====================\")\n",
    "\n",
    "        for name, model_type, model_params_base in runs:\n",
    "            # ---- per-run params ----\n",
    "            dp = dict(dataset_params)\n",
    "            if vary_dataset_seed and \"seed\" in dp:\n",
    "                dp[\"seed\"] = int(seed)\n",
    "\n",
    "            mp = dict(model_params_base)\n",
    "            if vary_model_init_seed and \"seed\" in mp:\n",
    "                mp[\"seed\"] = int(seed)\n",
    "\n",
    "            tp = dict(train_base)\n",
    "            tp[\"seed\"] = int(seed)\n",
    "            tp[\"lr\"] = float(lr_map[model_type])\n",
    "\n",
    "            # avoid plot spam unless explicitly allowed\n",
    "            if (n_seeds > 1) and (tp.get(\"plot_every\", 0) not in (0, None)) and (not allow_plots_multi_seed):\n",
    "                tp[\"plot_every\"] = 0\n",
    "\n",
    "            # ---- TRAIN ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                out = generate_and_train_simple(dataset_type, dp, model_type, mp, tp)\n",
    "            except Exception as e:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=repr(e)))\n",
    "                learn_rows.append(dict(seed=seed, model=name, train_time=np.nan, best_epoch=np.nan,\n",
    "                                       best_val=np.nan, test=np.nan, train_err=repr(e)))\n",
    "                continue\n",
    "            train_time = time.perf_counter() - t0\n",
    "\n",
    "            if isinstance(out, tuple) and len(out) == 4:\n",
    "                model, data, hist, spec = out\n",
    "            elif isinstance(out, tuple) and len(out) == 3:\n",
    "                model, data, hist = out\n",
    "                n_params = sum(p.numel() for p in model.parameters())\n",
    "                spec = dict(n_params=int(n_params), extra=\"\", train_params=tp)\n",
    "            else:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"TRAIN\", error=f\"Unexpected return: {type(out)}\"))\n",
    "                continue\n",
    "\n",
    "            # ---- model spec (seed 0 only) ----\n",
    "            if seed == seeds[0]:\n",
    "                spec_rows.append(dict(\n",
    "                    model=name,\n",
    "                    n_params=int(spec.get(\"n_params\", np.nan)),\n",
    "                    details=str(spec.get(\"extra\", \"\")),\n",
    "                    lr=float(spec.get(\"train_params\", {}).get(\"lr\", tp[\"lr\"])),\n",
    "                    batch_size=int(spec.get(\"train_params\", {}).get(\"batch_size\", tp[\"batch_size\"])),\n",
    "                    epochs=int(spec.get(\"train_params\", {}).get(\"epochs\", tp[\"epochs\"])),\n",
    "                ))\n",
    "\n",
    "            device = data[\"device\"]\n",
    "            scaler = data[\"scaler\"]\n",
    "            obj = make_obj(model, scaler, device, chunk=int(tp.get(\"plot_chunk\", 4096)))\n",
    "            gt = data.get(\"true\", None)\n",
    "\n",
    "            # ---- compute GT optimum (once per seed) ----\n",
    "            if seed not in gt_cache_by_seed:\n",
    "                t_gt0 = time.perf_counter()\n",
    "                try:\n",
    "                    x_gt, y_gt, gt_info = solve_true_opt(gt, xmin, xmax, sum_eq, time_limit=time_limit, verbose=False)\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    # evaluate with the existing helper to be extra sure\n",
    "                    y_gt_check = float(eval_true_obj(gt, x_gt))\n",
    "                    if np.isfinite(y_gt_check) and abs(y_gt_check - float(y_gt)) / (abs(float(y_gt_check)) + 1e-12) > 1e-6:\n",
    "                        fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT_CHECK\",\n",
    "                                              error=f\"gt solver mismatch: solver={y_gt:.6g} eval_true_obj={y_gt_check:.6g}\"))\n",
    "                    gt_cache_by_seed[seed] = dict(x=str(np.asarray(x_gt, int).tolist()),\n",
    "                                                  true_y=float(y_gt_check if np.isfinite(y_gt_check) else y_gt),\n",
    "                                                  runtime=float(gt_time),\n",
    "                                                  err=None)\n",
    "                except Exception as e:\n",
    "                    gt_time = time.perf_counter() - t_gt0\n",
    "                    gt_cache_by_seed[seed] = dict(x=None, true_y=np.nan, runtime=float(gt_time), err=repr(e))\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"GT_OPT\", error=repr(e)))\n",
    "\n",
    "            # ---- learning metrics: best val (normalized) + test loss (normalized) ----\n",
    "            test_norm, err_te = safe_norm_mse(model, scaler, device, data['raw']['Xte'], data['raw']['yte'], chunk=int(tp.get('plot_chunk', 4096)))\n",
    "            if err_te:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"EVAL_TEST\", error=err_te))\n",
    "\n",
    "            if obj._err_count > 0:\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"OBJ\",\n",
    "                                      error=f\"obj() had {obj._err_count} forward failures; first={obj._err_first}\"))\n",
    "\n",
    "            val_curve = np.asarray(hist.get(\"val_mse_norm\", []), dtype=float)\n",
    "            best_ep = int(np.argmin(val_curve) + 1) if val_curve.size else np.nan\n",
    "            best_val = float(np.min(val_curve)) if val_curve.size else np.nan\n",
    "\n",
    "            learn_rows.append(dict(\n",
    "                seed=seed, model=name,\n",
    "                train_time=float(train_time),\n",
    "                best_epoch=best_ep,\n",
    "                best_val=float(best_val),\n",
    "                test=float(test_norm) if np.isfinite(test_norm) else np.nan,\n",
    "                train_err=(err_te),\n",
    "            ))\n",
    "\n",
    "            # ---- LOCAL SEARCH ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                obj_ls = lambda x: float(np.asarray(obj(np.asarray(x))).reshape(-1)[0])\n",
    "                (ls_out, _ls_log) = suppress_stdout(\n",
    "                    local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                    delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                    silence=silence_local_search\n",
    "                )\n",
    "                x_best, y_best, ls_hist = ls_out\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=str(np.asarray(x_best, int).tolist()),\n",
    "                    y=float(y_best),\n",
    "                    true_y=float(eval_true_obj(gt, x_best)),\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                    iters=int(len(ls_hist) - 1),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=np.nan, iters=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "            # ---- IP SOLVE (learned objective) ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                x_ip, y_ip, info = solve_ip(\n",
    "                    model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                    time_limit=time_limit, verbose=True\n",
    "                )\n",
    "                ip_time = time.perf_counter() - t0\n",
    "\n",
    "                rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "                if rel_err > ip_check_tol:\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                          error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=str(np.asarray(x_ip, int).tolist()),\n",
    "                    y=float(y_ip),\n",
    "                    true_y=float(eval_true_obj(gt, x_ip)),\n",
    "                    runtime=float(ip_time),\n",
    "                    status=(info.get(\"status\") if isinstance(info, dict) else None),\n",
    "                    gap=(info.get(\"gap\") if isinstance(info, dict) else None),\n",
    "                    ip_rel_err=float(rel_err),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ip_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ip_time),\n",
    "                    status=None, gap=None, ip_rel_err=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    spec_df  = pd.DataFrame(spec_rows).drop_duplicates(\"model\").sort_values(\"model\").reset_index(drop=True)\n",
    "    learn_df = pd.DataFrame(learn_rows).sort_values([\"model\", \"seed\"]).reset_index(drop=True)\n",
    "    opt_df   = pd.DataFrame(opt_rows).sort_values([\"model\", \"seed\", \"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # ---- per-seed gaps ----\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\", \"model\", \"ls_vs_ip_pct\", \"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        # IP true vs GT optimum true\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    for model in sorted(learn_df[\"model\"].unique()):\n",
    "        sub = learn_df[learn_df[\"model\"] == model]\n",
    "        m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "        learn_sum_rows.append(dict(model=model, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- OPTIMIZATION SUMMARY ----\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model]\n",
    "        row = {\"model\": model}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== MODEL SPECS (from seed 0 run) ===\")\n",
    "    if not spec_df.empty:\n",
    "        print(spec_df[[\"model\", \"n_params\", \"details\", \"lr\", \"batch_size\", \"epochs\"]].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not learn_summary_df.empty:\n",
    "        print(learn_summary_df.to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    return spec_df, learn_df, opt_df, fail_df, learn_summary_df, opt_summary_df, gt_df\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e88e7a1f",
   "metadata": {},
   "source": [
    "## Tests"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3ac79bcb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[ASSIGNMENT] inferred in_dim=10 from Xtmp shape=(1000, 10)\n",
      "\n",
      "\n",
      "===================== SEED 0 =====================\n",
      "\n",
      "--- Dataset stats (assignment) ---\n",
      "  X: shape=(1000, 10)  mean(mean)=0.503  std(mean)=0.5  min=0  max=1\n",
      "  y: shape=(1000,)  mean=760  std=384  min=6  max=1.92e+03\n",
      "\n",
      "\n",
      "=== Run: assignment | DFN ===\n",
      "  data: N=1000  train/val/test=700/150/150  dim=10\n",
      "  model: params=16,725 layers=[16, 128, 16] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA90AAAGGCAYAAABmGOKbAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjgsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvwVt1zgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAxFlJREFUeJzs3Qd4U2XbB/B/VhcdUEah7A1lTwVlichwAYrgBMWBoqK+LvRzvipOxAHu/ToQBRcooiBb2SAblFX2bGnpSnK+6z4h7clskiZNm/x/15UmOTk5OT1JTs59nvu5H52iKAqIiIiIiIiIKOj0wV8kERERERERETHoJiIiIiIiIgohtnQTERERERERhQiDbiIiIiIiIqIQYdBNREREREREFCIMuomIiIiIiIhChEE3ERERERERUYgw6CYiIiIiIiIKEQbdRERERERERCHCoNtHH3/8MXQ6ncfLH3/8gXDavXu3uh4vv/xywMv4v//7P1xyySWoW7euuqwxY8Z4nPfff//F8OHDUbVqVSQmJmLAgAFYs2aN23m/+uordOzYEXFxcUhPT8c999yDnJwcl/lkmjwm88i88hx5ric//PADjEYjjh49qt6fMmWKuk6NGzdW179v374en3vkyBH1/6tRowYSEhLQo0cP/P77727n/e2339THZT6ZX54nz3dWVFSEp556Co0aNUJsbCxatWqFN954A76S5z355JMI12dbPkO+zOfp864oCpo1a+Z22x8/fhwTJ05ERkYGqlSpgpSUFHX7XH/99diwYUO5fs9uvvlmtG3bVv3sxsfHo0WLFnjggQdw7Ngxn5exZ88e3HTTTepnVd5r+c4MGzbM7bzff/89+vTpg+TkZPV/b9OmDd59912HeQoLC/H444+rn92YmBg0bNhQ3V55eXkIBV8+0/v27VP/pyZNmhS/Z506dcKbb74Js9ns82s9/fTT6vtutVpRGT322GPo3LlzpV1/ospO9kOyrz516pTHea699lqYTCYcPnzY5+XKb0o4fnNL89xzz+G7775zmb5582Z1fUv7rQ6FTz/9FDVr1sTp06fL7TUzMzPVY0L5/ZTfa3m/5BjBWXZ2Np599ln1uKN27drqMWm7du3wwgsvID8/32X+nTt3qsceDRo0UD9XTZs2xX333acep2jJPEOHDg3p/0jRhUG3nz766CMsX77c5SIH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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[DONE] best val MSE (norm) = 1.042e-03 @ epoch 989\n",
      "\n",
      "Set parameter Username\n",
      "Set parameter LicenseID to value 2685751\n",
      "Academic license - for non-commercial use only - expires 2026-07-09\n",
      "iter=0  t=0.00s  best_y=549.96  x=[1, 1, 1, 1, 0, 0, 0, 0, 0, 0]\n",
      "STOP: local minimum. best_y=549.96 x=[1, 1, 1, 1, 0, 0, 0, 0, 0, 0]\n",
      "Set parameter OutputFlag to value 1\n",
      "Set parameter TimeLimit to value 300\n",
      "Gurobi Optimizer version 12.0.3 build v12.0.3rc0 (mac64[arm] - Darwin 25.2.0 25C56)\n",
      "\n",
      "CPU model: Apple M4 Max\n",
      "Thread count: 14 physical cores, 14 logical processors, using up to 14 threads\n",
      "\n",
      "Non-default parameters:\n",
      "TimeLimit  300\n",
      "\n",
      "Optimize a model with 161 rows, 8714 columns and 17635 nonzeros\n",
      "Model fingerprint: 0xd41c1284\n",
      "Variable types: 8704 continuous, 10 integer (0 binary)\n",
      "Coefficient statistics:\n",
      "  Matrix range     [1e+00, 3e+01]\n",
      "  Objective range  [3e-03, 2e+06]\n",
      "  Bounds range     [1e+00, 1e+06]\n",
      "  RHS range        [4e-02, 5e+01]\n",
      "Presolve removed 0 rows and 1035 columns\n",
      "Presolve time: 0.01s\n",
      "Presolved: 161 rows, 7679 columns, 15560 nonzeros\n",
      "Variable types: 7669 continuous, 10 integer (10 binary)\n",
      "\n",
      "Root relaxation: objective 5.232863e+01, 117 iterations, 0.00 seconds (0.00 work units)\n",
      "\n",
      "    Nodes    |    Current Node    |     Objective Bounds      |     Work\n",
      " Expl Unexpl |  Obj  Depth IntInf | Incumbent    BestBd   Gap | It/Node Time\n",
      "\n",
      "     0     0   52.32863    0    2          -   52.32863      -     -    0s\n",
      "     0     0   52.32863    0    2          -   52.32863      -     -    0s\n",
      "     0     2   52.32863    0    2          -   52.32863      -     -    0s\n",
      "*    1     0               1      54.4684617   54.46846  0.00%   2.0    0s\n",
      "\n",
      "Explored 3 nodes (142 simplex iterations) in 0.02 seconds (0.02 work units)\n",
      "Thread count was 14 (of 14 available processors)\n",
      "\n",
      "Solution count 1: 54.4685 \n",
      "\n",
      "Optimal solution found (tolerance 1.00e-04)\n",
      "Best objective 5.446846171877e+01, best bound 5.446846171877e+01, gap 0.0000%\n",
      "[CHECK DFN] obj(x_ip)=54.4685  ip_y=54.4685  rel_err=8.104e-07\n",
      "\n",
      "--- Dataset stats (assignment) ---\n",
      "  X: shape=(1000, 10)  mean(mean)=0.503  std(mean)=0.5  min=0  max=1\n",
      "  y: shape=(1000,)  mean=760  std=384  min=6  max=1.92e+03\n",
      "\n",
      "\n",
      "=== Run: assignment | DFN ===\n",
      "  data: N=1000  train/val/test=700/150/150  dim=10\n",
      "  model: params=17,610 layers=[5, 400, 6] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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W/gx1/+r8I0KDbiLTucHnIiRbgAAAMAHHTp0MOuddX331VdfbaZa67rqC6VruQvTkV3LskpNsKYBqY5I61TtokoKrOPi4kyZvalatar7a9dNBFf2cC1L0RsLRcvnrR5F39f13t4eK5qp/PfffzcBuwbfgNMRdAMAAAA+0szfM2fONKPdV155pSQlJfn0c+Hh4R6jwqWpW7euHDt2zCQccwWyuta6sI4dO8qOHTvMntvnctNAp3p7C6JLo1PAdap4YevWrTPTzDXpWSBs3brV1BEIBkwvBwAAAHw0fPhwE3C/+OKL57QOW6dU79q1ywTPmoCttGnT3bp1M6O848ePN+uq33rrrWJJyx577DEz4q5JzbZt22amgL/zzjvyt7/9rcT31bXTGsjr68+FJoH7xz/+YdaM//DDDybz+PPPP++xftvfNIla3759A/b+QHki6AYAAAB8pJm/dZswXZM8cOBAn8+b/ky/fv1M4Ksj2aVthxUbGytvvPGGye598cUXm9dqcF2YTm/XzOnLly+XLl26SPfu3U3Cs+Tk5BLfV7c80+nomgDuXOiI87vvvmsykGt2cg34dT154SRq/qQ3NXQkvays8IBThFjeFmQAAAAA8Oqqq66Siy66yGyF5TS6plunxesIenR0tNjRgw8+KJmZmWYrMSAYMNINAAAA+ECTe+lo7xdffCFjxoxx5DnTkfNp06aZ7cPsql69emYqOxAsGOkGAAAAfFyXfeTIEbNfddH1zLqv9Z49e7z+3Lx588xacACVE0E3AAAAcIE04C66rVfhfajtOpUbQOARdAMAAAAAECCs6QYAAAAAIEAIugEAAAAACBCCbgAAAAAAAoSgGwAAAACAACHoBgAAAAAgQAi6AQAAAAAIEIJuAAAAAAAChKAbAAAAAAAJjP8HMNvwn3J0pzUAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ===================== ASSIGNMENT DATASET =====================\n",
    "N_SEEDS = 1\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=1000,\n",
    "    num_nodes=10,\n",
    "    c_min=1,\n",
    "    c_max=1000,\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=1,\n",
    "    plot_points=128,\n",
    "    plot_chunk=128,\n",
    ")\n",
    "\n",
    "# ---- DFN (learnable A) ----\n",
    "dfn_params = dict(\n",
    "    input_dim=in_dim, layer_sizes=[16, 128, 16], p_list=[1, 1],\n",
    "    seed=0, alpha=5e-3, beta=-2.0\n",
    ")\n",
    "\n",
    "# ---- DFN (fixed A = I) ----\n",
    "L1 = 5\n",
    "LK = 6\n",
    "dfn_Afix_params = dict(\n",
    "    input_dim=in_dim,\n",
    "    layer_sizes=[L1, 400, 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=1600)\n",
    "lset_params = dict(in_dim=in_dim, n_pieces=1600, T=0.05)\n",
    "\n",
    "lr_map = dict(DFN=1e-1, MLP=1e-3, MaxAffine=1e-3, LSET=1e-3)\n",
    "time_limit = 300\n",
    "\n",
    "# feasible integer box + feasible x0 / sum_eq\n",
    "xmin  = np.full(in_dim, 0, dtype=int)\n",
    "xmax  = np.full(in_dim, 1, dtype=int)\n",
    "x0    = np.array([1, 1, 1, 1] + [0] * (in_dim - 4), dtype=int)\n",
    "delta = 2\n",
    "sum_eq = int(x0.sum())\n",
    "\n",
    "runs = [\n",
    "    (\"DFN\",        \"DFN\", dfn_params),\n",
    "    (\"DFN_AfixI\",  \"DFN\", dfn_Afix_params),\n",
    "    (\"MLP\",        \"MLP\", mlp_params),\n",
    "    (\"MaxAffine\",  \"MaxAffine\", maff_params),\n",
    "    (\"LSET\",       \"LSET\", lset_params),\n",
    "]\n",
    "\n",
    "_ = run_benchmark(\n",
    "    dataset_type=dataset_type,\n",
    "    dataset_params=dataset_params,\n",
    "    runs=runs,\n",
    "    train_base=train_base,\n",
    "    lr_map=lr_map,\n",
    "    x0=x0, xmin=xmin, xmax=xmax,\n",
    "    delta=delta, sum_eq=sum_eq,\n",
    "    n_seeds=N_SEEDS,\n",
    "    vary_dataset_seed=VARY_DATASET_SEED,\n",
    "    vary_model_init_seed=VARY_MODEL_INIT_SEED,\n",
    "    strict_ip_check=STRICT_IP_CHECK,\n",
    "    ip_check_tol=IP_CHECK_TOL,\n",
    "    silence_local_search=SILENCE_LOCAL_SEARCH,\n",
    "    allow_plots_multi_seed=ALLOW_PLOTS_MULTI_SEED,\n",
    "    time_limit=time_limit,\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "92dbc0ee-bbe1-4fdd-8782-44ece09dda4b",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:dfn]",
   "language": "python",
   "name": "conda-env-dfn-py"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
