{
 "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": {},
   "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",
    "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 = True,\n",
    "):\n",
    "    \"\"\"y = (x - x*)^T Q (x - x*) + noise, with Q symmetric *indefinite* (nonconvex).\"\"\"\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",
    "\n",
    "    # --- CHANGE STARTS HERE: make Q indefinite (mix + and - eigenvalues) ---\n",
    "    # If eigen_min/eigen_max are given as positive magnitudes, we use them as |lambda|.\n",
    "    lo = float(min(eigen_min, eigen_max))\n",
    "    hi = float(max(eigen_min, eigen_max))\n",
    "\n",
    "    mags = rng.uniform(max(0.0, lo), max(0.0, hi), dim)  # magnitudes\n",
    "    signs = np.ones(dim, dtype=np.float64)\n",
    "    if dim >= 2:\n",
    "        signs[: dim // 2] = -1.0\n",
    "        rng.shuffle(signs)\n",
    "    else:\n",
    "        signs[0] = -1.0  # 1D: doesn't matter much, but makes it \"nonconvex\" in the trivial sense\n",
    "\n",
    "    eigs = mags * signs\n",
    "\n",
    "    # Safety: ensure at least one + and one - eigenvalue when dim>=2\n",
    "    if dim >= 2 and (np.all(eigs >= 0) or np.all(eigs <= 0)):\n",
    "        eigs[0] = -abs(eigs[0]) if eigs[0] != 0 else -(hi if hi > 0 else 1.0)\n",
    "        eigs[1] =  abs(eigs[1]) if eigs[1] != 0 else  (hi if hi > 0 else 1.0)\n",
    "\n",
    "    Q = U @ np.diag(eigs) @ U.T\n",
    "    # --- CHANGE ENDS HERE ---\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",
    "\n",
    "\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": [
      "(10, 8) (10,) \n",
      " [[284. 307. 453. 571.  20.  86. 494. 570.]\n",
      " [149. 187. 522. 254. 164. 497. 154. 245.]\n",
      " [386. 330.  51.  16. 520. 452. 503. 323.]\n",
      " [491. 198. 272. 473.  74. 182.  74. 272.]\n",
      " [587.  80. 230. 242. 543. 122. 301. 157.]] \n",
      " [2.8892958e+07 2.2397388e+07 2.6657474e+07 2.1041728e+07 2.3339302e+07] \n",
      " 3.6817302e+06 \n",
      "\n",
      "(10, 10) (10,) \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",
      " [ 665.  226.  993. 1351.  851.] \n",
      " 296.35287 \n",
      "\n",
      "(10, 10) (10,) \n",
      " [[  0.   7.   3.  -9.   3.  -3.  -6.  -1.   8.  10.]\n",
      " [ -8.   1.   5.  -5.  -5.  -5.  -6.   8.  -6.  -6.]\n",
      " [ -8.  -8.   6.  -4.   6.   2.   8.   1.   6.   7.]\n",
      " [ -9.   1.  -1.  -4.  -1.  -2.   0.   7.   7.   3.]\n",
      " [  4.  10.   3.  -3.  -9.   1.  -6.   2. -10.   7.]] \n",
      " [-4108.1816  -703.3988  -760.6904 -1040.9944   153.1423] \n",
      " 2693.4705 \n",
      "\n"
     ]
    }
   ],
   "source": [
    "# NOTE: MDVSP requires that `filename` exists on disk.\n",
    "X, y, _ = make_mdvsp_dataset(\n",
    "    K=10, filename=\"RN-8-3000-05.dat\", x_min=0, x_max=600, noise_std=0.0, seed=1, max_trips=5000, max_succ=50\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\", y.std(), \"\\n\")\n",
    "\n",
    "X, y, _ = generate_bipartite_subset_matching_dataset(\n",
    "    K=10, num_nodes=10, c_min=1, c_max=1000, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\", y.std(), \"\\n\")\n",
    "\n",
    "X, y, _ = generate_convex_quadratic_dataset(\n",
    "    K=10, dim=10, eigen_min=1.0, eigen_max=20.0, x_min=-10, x_max=10, noise_std=0.0, seed=0\n",
    ")\n",
    "print(X.shape, y.shape, \"\\n\", X[:5], \"\\n\", y[:5], \"\\n\", y.std(), \"\\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 _set_full_determinism(seed: int):\n",
    "    import os, random\n",
    "    import numpy as np\n",
    "    import torch\n",
    "    seed = int(seed)\n",
    "\n",
    "    # NOTE: For strict CUDA determinism, this env var should be set before any CUDA work.\n",
    "    os.environ.setdefault(\"CUBLAS_WORKSPACE_CONFIG\", \":4096:8\")\n",
    "    os.environ[\"PYTHONHASHSEED\"] = str(seed)\n",
    "\n",
    "    random.seed(seed)\n",
    "    np.random.seed(seed)\n",
    "    torch.manual_seed(seed)\n",
    "    if torch.cuda.is_available():\n",
    "        torch.cuda.manual_seed(seed)\n",
    "        torch.cuda.manual_seed_all(seed)\n",
    "\n",
    "    # Determinism flags (will error if a non-deterministic op is used)\n",
    "    try:\n",
    "        torch.use_deterministic_algorithms(True)\n",
    "    except Exception:\n",
    "        pass\n",
    "\n",
    "    try:\n",
    "        torch.backends.cudnn.deterministic = True\n",
    "        torch.backends.cudnn.benchmark = False\n",
    "    except Exception:\n",
    "        pass\n",
    "\n",
    "    # Avoid TF32 variance on Ampere+ GPUs\n",
    "    try:\n",
    "        torch.backends.cuda.matmul.allow_tf32 = False\n",
    "        torch.backends.cudnn.allow_tf32 = False\n",
    "    except Exception:\n",
    "        pass\n",
    "\n",
    "def _seed_worker(worker_id: int):\n",
    "    import random\n",
    "    import numpy as np\n",
    "    import torch\n",
    "    worker_seed = torch.initial_seed() % 2**32\n",
    "    np.random.seed(worker_seed)\n",
    "    random.seed(worker_seed)\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",
    "    # ---- determinism (seed controls data, model init, dataloader order, etc.) ----\n",
    "    _set_full_determinism(seed)\n",
    "    # force dataset generation to use the same seed as training\n",
    "    dataset_params = dict(dataset_params)\n",
    "    dataset_params[\"seed\"] = int(seed)\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",
    "        \"deterministic\": True,\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",
    "    g_dl = torch.Generator()\n",
    "    g_dl.manual_seed(seed)\n",
    "    train_loader = DataLoader(\n",
    "        TensorDataset(XtrN, ytrN),\n",
    "        batch_size=batch_sz,\n",
    "        shuffle=True,\n",
    "        generator=g_dl,\n",
    "        worker_init_fn=_seed_worker,\n",
    "    )\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": {},
   "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": {},
   "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 = obj\n",
    "                (ls_out, _ls_log) = suppress_stdout(\n",
    "                    local_search_l1_int, obj_ls, x0, xmin, xmax,\n",
    "                    delta=delta, sum_eq=sum_eq, print_every=0,\n",
    "                    silence=silence_local_search\n",
    "                )\n",
    "                x_best, y_best, ls_hist = ls_out\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=str(np.asarray(x_best, int).tolist()),\n",
    "                    y=float(y_best),\n",
    "                    true_y=float(eval_true_obj(gt, x_best)),\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=float(t_to_best(ls_hist, y_best)),\n",
    "                    iters=int(len(ls_hist) - 1),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ls_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"LS\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ls_time),\n",
    "                    t_best=np.nan, iters=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"LS\", error=repr(e)))\n",
    "\n",
    "            # ---- IP SOLVE (learned objective) ----\n",
    "            t0 = time.perf_counter()\n",
    "            try:\n",
    "                x_ip, y_ip, info = solve_ip(\n",
    "                    model_type, model, scaler, xmin, xmax, sum_eq,\n",
    "                    time_limit=time_limit, verbose=True\n",
    "                )\n",
    "                ip_time = time.perf_counter() - t0\n",
    "\n",
    "                rel_err = check_ip_matches_obj(name, obj, x_ip, y_ip, strict=strict_ip_check, tol=ip_check_tol)\n",
    "                if rel_err > ip_check_tol:\n",
    "                    fail_rows.append(dict(seed=seed, model=name, stage=\"IP_CHECK\",\n",
    "                                          error=f\"rel_err={rel_err:.3e} (tol={ip_check_tol})\"))\n",
    "\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=str(np.asarray(x_ip, int).tolist()),\n",
    "                    y=float(y_ip),\n",
    "                    true_y=float(eval_true_obj(gt, x_ip)),\n",
    "                    runtime=float(ip_time),\n",
    "                    status=(info.get(\"status\") if isinstance(info, dict) else None),\n",
    "                    gap=(info.get(\"gap\") if isinstance(info, dict) else None),\n",
    "                    ip_rel_err=float(rel_err),\n",
    "                    err=None,\n",
    "                ))\n",
    "            except Exception as e:\n",
    "                ip_time = time.perf_counter() - t0\n",
    "                opt_rows.append(dict(\n",
    "                    seed=seed, model=name, method=\"IP\",\n",
    "                    x=None, y=np.nan, true_y=np.nan,\n",
    "                    runtime=float(ip_time),\n",
    "                    status=None, gap=None, ip_rel_err=np.nan,\n",
    "                    err=repr(e),\n",
    "                ))\n",
    "                fail_rows.append(dict(seed=seed, model=name, stage=\"IP\", error=repr(e)))\n",
    "\n",
    "    # ---- build DFs ----\n",
    "    spec_df  = pd.DataFrame(spec_rows).drop_duplicates(\"model\").sort_values(\"model\").reset_index(drop=True)\n",
    "    learn_df = pd.DataFrame(learn_rows).sort_values([\"model\", \"seed\"]).reset_index(drop=True)\n",
    "    opt_df   = pd.DataFrame(opt_rows).sort_values([\"model\", \"seed\", \"method\"]).reset_index(drop=True)\n",
    "\n",
    "    fail_df = pd.DataFrame(fail_rows)\n",
    "    if fail_df.empty:\n",
    "        fail_df = pd.DataFrame(columns=[\"seed\", \"model\", \"stage\", \"error\"])\n",
    "    else:\n",
    "        for c in [\"stage\", \"model\", \"seed\"]:\n",
    "            if c not in fail_df.columns:\n",
    "                fail_df[c] = np.nan\n",
    "        fail_df = fail_df.sort_values([\"stage\", \"model\", \"seed\"]).reset_index(drop=True)\n",
    "\n",
    "    gt_df = pd.DataFrame([dict(seed=s, **d) for s, d in sorted(gt_cache_by_seed.items())]).sort_values(\"seed\")\n",
    "\n",
    "    # ---- per-seed gaps ----\n",
    "    gap_seed_df = pd.DataFrame(columns=[\"seed\", \"model\", \"ls_vs_ip_pct\", \"ip_true_vs_gt_pct\"])\n",
    "    if not opt_df.empty:\n",
    "        pivot_y = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"y\", aggfunc=\"first\")\n",
    "        pivot_true = opt_df.pivot_table(index=[\"seed\", \"model\"], columns=\"method\", values=\"true_y\", aggfunc=\"first\")\n",
    "\n",
    "        if (\"LS\" in pivot_y.columns) and (\"IP\" in pivot_y.columns):\n",
    "            ls_vs_ip = 100.0 * (pivot_y[\"LS\"] - pivot_y[\"IP\"]) / (np.abs(pivot_y[\"IP\"]) + 1e-12)\n",
    "        else:\n",
    "            ls_vs_ip = pd.Series(index=pivot_y.index, dtype=float)\n",
    "\n",
    "        # IP true vs GT optimum true\n",
    "        gt_true = gt_df.set_index(\"seed\")[\"true_y\"] if not gt_df.empty else pd.Series(dtype=float)\n",
    "        ip_true_vs_gt = []\n",
    "        for (seed, model), row in pivot_true.iterrows():\n",
    "            ipt = float(row.get(\"IP\", np.nan))\n",
    "            gtt = float(gt_true.get(seed, np.nan))\n",
    "            if np.isfinite(ipt) and np.isfinite(gtt):\n",
    "                ip_true_vs_gt.append(((seed, model), 100.0 * (ipt - gtt) / (abs(gtt) + 1e-12)))\n",
    "            else:\n",
    "                ip_true_vs_gt.append(((seed, model), np.nan))\n",
    "        ip_true_vs_gt = pd.Series({k: v for k, v in ip_true_vs_gt})\n",
    "\n",
    "        gap_seed_df = pd.DataFrame({\n",
    "            \"seed\": [k[0] for k in ip_true_vs_gt.index],\n",
    "            \"model\": [k[1] for k in ip_true_vs_gt.index],\n",
    "            \"ls_vs_ip_pct\": [float(ls_vs_ip.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "            \"ip_true_vs_gt_pct\": [float(ip_true_vs_gt.get(k, np.nan)) for k in ip_true_vs_gt.index],\n",
    "        })\n",
    "\n",
    "    # ---- LEARNING SUMMARY (mean ± SE over seeds) ----\n",
    "    learn_sum_rows = []\n",
    "    for model in sorted(learn_df[\"model\"].unique()):\n",
    "        sub = learn_df[learn_df[\"model\"] == model]\n",
    "        m, se = mean_se(sub[\"train_time\"]); train_time_s = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"best_val\"]);   best_val_s   = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(sub[\"test\"]);       test_s       = fmt_mean_se(m, se)\n",
    "        learn_sum_rows.append(dict(model=model, train_time=train_time_s, best_val=best_val_s, test=test_s))\n",
    "    learn_summary_df = pd.DataFrame(learn_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- OPTIMIZATION SUMMARY ----\n",
    "    opt_sum_rows = []\n",
    "    gt_x_repr = (repr_solution(gt_df.get(\"x\", pd.Series(dtype=str)), gt_df.get(\"seed\", None))\n",
    "                if not gt_df.empty else None)\n",
    "    m, se = mean_se(gt_df.get(\"true_y\", pd.Series(dtype=float))); gt_true_s = fmt_mean_se(m, se)\n",
    "    m, se = mean_se(gt_df.get(\"runtime\", pd.Series(dtype=float))); gt_time_s = fmt_mean_se(m, se)\n",
    "\n",
    "    for model in sorted(opt_df[\"model\"].unique()):\n",
    "        sub = opt_df[opt_df[\"model\"] == model]\n",
    "        row = {\"model\": model}\n",
    "\n",
    "        ls = sub[sub[\"method\"] == \"LS\"]\n",
    "        ip = sub[sub[\"method\"] == \"IP\"]\n",
    "\n",
    "        row[\"LS_x\"] = repr_solution(ls[\"x\"], ls.get(\"seed\", None))\n",
    "        m, se = mean_se(ls[\"y\"]);       row[\"LS_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"true_y\"]);  row[\"LS_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ls[\"runtime\"]); row[\"LS_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"IP_x\"] = repr_solution(ip[\"x\"], ip.get(\"seed\", None))\n",
    "        m, se = mean_se(ip[\"y\"]);       row[\"IP_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"true_y\"]);  row[\"IP_true_y\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(ip[\"runtime\"]); row[\"IP_time\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        row[\"GT_x\"] = gt_x_repr\n",
    "        row[\"GT_true_y\"] = gt_true_s\n",
    "        row[\"GT_time\"] = gt_time_s\n",
    "\n",
    "        gsub = gap_seed_df[gap_seed_df[\"model\"] == model] if not gap_seed_df.empty else pd.DataFrame()\n",
    "        m, se = mean_se(gsub.get(\"ls_vs_ip_pct\", pd.Series(dtype=float))); row[\"LS_vs_IP_%\"] = fmt_mean_se(m, se)\n",
    "        m, se = mean_se(gsub.get(\"ip_true_vs_gt_pct\", pd.Series(dtype=float))); row[\"IP_true_vs_GT_%\"] = fmt_mean_se(m, se)\n",
    "\n",
    "        opt_sum_rows.append(row)\n",
    "\n",
    "    opt_summary_df = pd.DataFrame(opt_sum_rows).sort_values(\"model\").reset_index(drop=True)\n",
    "\n",
    "    # ---- Print tables ----\n",
    "    print(\"\\n=== MODEL SPECS (from seed 0 run) ===\")\n",
    "    if not spec_df.empty:\n",
    "        print(spec_df[[\"model\", \"n_params\", \"details\", \"lr\", \"batch_size\", \"epochs\"]].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== LEARNING SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not learn_summary_df.empty:\n",
    "        print(learn_summary_df.to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== OPTIMIZATION SUMMARY (mean ± SE over seeds) ===\")\n",
    "    if not opt_summary_df.empty:\n",
    "        cols = [\n",
    "            \"model\",\n",
    "            \"LS_x\", \"LS_y\", \"LS_true_y\", \"LS_time\",\n",
    "            \"IP_x\", \"IP_y\", \"IP_true_y\", \"IP_time\",\n",
    "            \"GT_x\", \"GT_true_y\", \"GT_time\",\n",
    "            \"LS_vs_IP_%\", \"IP_true_vs_GT_%\",\n",
    "        ]\n",
    "        print(opt_summary_df[cols].to_string(index=False))\n",
    "    else:\n",
    "        print(\"None\")\n",
    "\n",
    "    print(\"\\n=== FAILURES / WARNINGS (if any) ===\")\n",
    "    if fail_df.shape[0] == 0:\n",
    "        print(\"None\")\n",
    "    else:\n",
    "        print(fail_df.to_string(index=False))\n",
    "\n",
    "    return spec_df, learn_df, opt_df, fail_df, learn_summary_df, opt_summary_df, gt_df\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e88e7a1f",
   "metadata": {},
   "source": [
    "## Tests"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8edf3f97",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "===================== SEED 0 =====================\n",
      "\n",
      "--- Dataset stats (quadratic) ---\n",
      "  X: shape=(5000, 32)  mean(mean)=-0.0274  std(mean)=58  min=-100  max=100\n",
      "  y: shape=(5000,)  mean=3.01e+04  std=2.37e+05  min=-8.87e+05  max=9.32e+05\n",
      "\n",
      "\n",
      "=== Run: quadratic | MLP ===\n",
      "  data: N=5000  train/val/test=3500/750/750  dim=32\n",
      "  model: params=20,865 hidden=[128, 128]\n",
      "  train: device=cpu  epochs=500  batch=8  lr=0.001  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA90AAAGGCAYAAABmGOKbAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjgsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvwVt1zgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAw8FJREFUeJzs3Qd4FNXXBvCTTgIh1NCr9N6VphRFwIKoFMXeERXEgg2xg+VPUbH72QsWsCuiIEWk9957bwkE0vM9753M5u5ktiW7KZv39zwL2ZLd2dnN7px7zj03JCsrK0uIiIiIiIiIyO9C/X+XRERERERERMSgm4iIiIiIiCiAmOkmIiIiIiIiChAG3UREREREREQBwqCbiIiIiIiIKEAYdBMREREREREFCINuIiIiIiIiogBh0E1EREREREQUIAy6iYiIiIiIiAKEQXcAfPzxxxISEuLy9M8//0hh2rVrl9qO1157LV+/b3f6+uuvc91+x44dcvXVV0u5cuWkTJkycskll8iKFSts7xu/36ZNGylVqpRUr15dRo0aJWfOnLG9bUZGhsTHx8ukSZPU+VtuucV2m5o0aWL7+2+88Ya6LioqSurVqyfPPvuspKWl5brdkSNH1H1XqlRJYmJipHPnzvL333/7tK8K4zXHNtetW9er22EbY2Njbff17t27JTQ0VN3mmWeecbpu48aNcuONN0r9+vXVa4Z91K5dO7nvvvskMTEx12O4OuWXr+9Jq9OnT8ujjz4qffr0kcqVK9s+Vx3eJxMnTpSWLVtKdHS0em936dJFFi5caLv/brvtNvV+xnutRo0aMnDgQAkEb97T06dPl+uuu04aNGigth3vkWHDhsnWrVt9eqzevXvLPffcI8XVhRdeqD5fiKjow2cmPq9OnTrl8jb4HIuIiJDDhw97fb+ePusLy0svvSQ//PBDrss3bNigthffeQXt008/Vd+P+L4sKPv27VOf0xdddJH6nsXrhWNsKxxvvPjii9KjRw+pWrWqOtbE9/PLL78sycnJuW6/bds2dexSu3Zt9b4677zzZPTo0XL8+HGn2+E2V111VUCfI5Uc4YW9AcHso48+sg34mjVrJsHg/vvvl+uvv97psoYNGzqdP3r0qHTv3l3Kly8v//d//6cCs/Hjx6sPxqVLl0rjxo0dt/3iiy/khhtukDvuuEMF0lu2bJExY8aoL5k///wz1+PPmzdP3T8CehM+PGfPnu10O1xmhQ/nsWPHymOPPaYCLWzLU089Jfv375f33nvPcbuUlBQVXOCLfsqUKSrInzp1qvTt21f++usv9UUQDHCgkp6eLtOmTZPbb7891/sYAbkeRMPKlSula9eu0rRpU3n66adV8Hbs2DFZvXq1CnQffvhhKVu2rNvXpjDek3bwRYvXvXXr1uoL9oMPPnB5Wwz24ABwwYIFKlBHsJ2UlCTLly9X/+vWrVun3usYlMAgV82aNeXgwYMyc+ZM8Tdv39M4CMFByZNPPqm2a+/eveoAD4MlixYtkubNm3t8rB9//FH+/fdfdRBWXD3//PNqAHD48OFOn0NEVPTgewlB6Jdffin33ntvrusTEhJkxowZcvnll0uVKlWkuMNn8rXXXpsr4MPxEAZT8b3izaC6v5w9e1aeeOIJdUyG44GCguAYx4ZIxvTv31+++uor29vt2bNHJk+erIJkBM8IuufPn68GKGbNmqVO5gA/jhsvuOACdXyC7wEE3jieGTdunMyZM0d9lyPRAPh9HMfj2KVXr14F9rwpSGWR33300UdZ2LVLly4tknt3586davteffXVgP/+I488khUREZG1a9cux2UJCQlZlSpVyho8eLDjsvT09Kxq1apl9enTx+n3v/jiC/VYv/32W677vvfee7M6dOjgOH/zzTdnlS5d2uM2HTt2LKtUqVJZd911l9PlL774YlZISEjW+vXrHZdNnTpVPf7ChQsdl6WlpWU1a9Ysq1OnTl7vqzlz5mQVNOyPOnXqeHU77LehQ4dmdenSxem6zMxMdR933nmneh7jxo1zXHfTTTep30tMTLS9X/yu9TGK6nsa22pu79GjR3M9V92kSZOyQkNDs/777z+P99mmTRt1Sk5OzgokX97Thw8fzvX7+/fvV3+nt99+u1ePh/c+3i+FISkpyW/31aJFC/XeJqKiDccI1atXz2rfvr3t9W+//bb63P755599ul93n/WFCd+X+N60+vbbbwNyTOHpc/Wtt95S3zEnT57MKkgZGRmOn3FMjeeOY2yrM2fOqJMVjgnwO/Pnz3dc9v7776vL/vrrL6fbvvTSS+ryFStWOF1++eWXZ11yySV+ekZUkrG8vJBh5A2luO+++640atRIlYUiE25XEous2YABA1TWGBljjPx98sknuW6HrOxDDz2k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dR96fNqLrr3LXm7evGlD0b107NjR9zw9Pd1+XrlyxX4eP368ROE1fX3ixImAYeFev4cOKXcTbtW4cWMbVq7H67/M/bv8L1Rowh08nB+oiOjpBgAAACLQpUsXm++s87v79etnQ611XnVZ6Vxuf9qz6zhO2AJrmpBqj7QO1Q4WKrF++OGH7Zi91KhRw/fcvYjgVg/XYwm+sBB8fF6/R/B+3X17LQuuVP7HH39Ywq7JN1DRkXQDAAAAEdLK30uWLLHe7r59+8qjjz4a0XYpKSkBvcLhNGzYUK5fv24Fx9xEVuda+8vKypJff/3V7rkdzUUDHertlUSHo0PAdai4v/3799swcy16Fg9Hjx613xGoDBheDgAAAERo1KhRlnB/8sknUc3D1iHVhYWFljxrAbZww6Z79OhhvbyzZs2yedUbN24sUbTs3XfftR53LWp27NgxGwL+2Wefydtvvx1yvzp3WhN5XT8aWgTuxx9/tDnjv/32m1UeX758ecD87fKmRdSef/75uO0f+C+RdAMAAAAR0srfepswnZP84osvRhw33aZ///6W+GpPdrjbYaWlpcn69eutuneHDh1sXU2u/enwdq2cvn37dunWrZs89dRTVvAsIyMj5H71lmc6HF0LwEVDe5w///xzq0Cu1ck14df55P5F1MqTXtTQnvTSqsIDFUUVx2tCBgAAAABPzz33nLRt29ZuhVXR6JxuHRavPeh169aVRPTmm29KcXGx3UoMqAzo6QYAAAAioMW9tLd3x44dMmHChAoZM+05X7hwod0+LFE1atTIhrIDlQU93QAAAECE87KvXbtm96sOns+s97UuKiry3G7VqlU2FxxAciLpBgAAAMpIE+7g23r534c6UYdyA4g/km4AAAAAAOKEOd0AAAAAAMQJSTcAAAAAAHFC0g0AAAAAQJyQdAMAAAAAECck3QAAAAAAxAlJNwAAAAAAcULSDQAAAABAnJB0AwAAAAAg8fE/joD7ARnIclwAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[DONE] best val MSE (norm) = 4.818e-02 @ epoch 127\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",
      "Set parameter OutputFlag to value 1\n",
      "Set parameter TimeLimit to value 500\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  500\n",
      "\n",
      "Optimize a model with 1282 rows, 801 columns and 22945 nonzeros\n",
      "Model fingerprint: 0x47b54e65\n",
      "Variable types: 513 continuous, 288 integer (256 binary)\n",
      "Coefficient statistics:\n",
      "  Matrix range     [2e-06, 2e+02]\n",
      "  Objective range  [2e+05, 2e+05]\n",
      "  Bounds range     [1e+00, 2e+02]\n",
      "  RHS range        [6e-03, 2e+02]\n",
      "Presolve removed 257 rows and 1 columns\n",
      "Presolve time: 0.04s\n",
      "Presolved: 1025 rows, 800 columns, 22560 nonzeros\n",
      "Variable types: 512 continuous, 288 integer (256 binary)\n",
      "\n",
      "Interrupt request received\n",
      "\n",
      "Root relaxation: objective -2.137068e+07, 401 iterations, 0.01 seconds (0.02 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 -2.137e+07    0  145          - -2.137e+07      -     -    0s\n",
      "     0     0 -1.670e+07    0  169          - -1.670e+07      -     -    0s\n",
      "     0     0 -1.593e+07    0  186          - -1.593e+07      -     -    0s\n",
      "     0     0 -1.590e+07    0  186          - -1.590e+07      -     -    0s\n",
      "     0     0 -1.590e+07    0  183          - -1.590e+07      -     -    0s\n",
      "     0     0 -1.590e+07    0  186          - -1.590e+07      -     -    0s\n",
      "     0     0 -1.540e+07    0  200          - -1.540e+07      -     -    0s\n",
      "     0     0 -1.532e+07    0  210          - -1.532e+07      -     -    0s\n",
      "     0     0 -1.531e+07    0  207          - -1.531e+07      -     -    0s\n",
      "     0     0 -1.531e+07    0  210          - -1.531e+07      -     -    0s\n",
      "     0     0 -1.519e+07    0  217          - -1.519e+07      -     -    0s\n",
      "     0     0 -1.519e+07    0  216          - -1.519e+07      -     -    1s\n",
      "     0     0 -1.519e+07    0  216          - -1.519e+07      -     -    1s\n",
      "H    0     0                    -127411.1482 -1.519e+07      -     -    1s\n",
      "     0     0 -1.503e+07    0  214 -127411.15 -1.503e+07      -     -    1s\n",
      "     0     0 -1.459e+07    0  214 -127411.15 -1.459e+07      -     -    2s\n",
      "     0     2 -1.459e+07    0  214 -127411.15 -1.459e+07      -     -    2s\n",
      "  1795  1984 -1905290.4   28  123 -127411.15 -1.371e+07      -   151    5s\n",
      "H 3774  3559                    -128151.8217 -1.266e+07  9779%   100    7s\n",
      "  3782  3565 -1054637.1   62  214 -128151.82 -1.266e+07  9779%   100   10s\n",
      "H 3835  3439                    -140341.6561 -1.266e+07  8921%   104   13s\n",
      "  4875  4127 -2088152.0   39  138 -140341.66 -1.266e+07  8921%   121   15s\n",
      " 15033  9957 -1.060e+07   24  187 -140341.66 -1.111e+07  7814%  79.2   20s\n",
      "H17757 12306                    -140548.0665 -1.111e+07  7803%  73.7   20s\n",
      "H17758 12306                    -141148.2749 -1.111e+07  7769%  73.7   20s\n",
      " 23088 17977 -3899441.7   73   88 -141148.27 -1.077e+07  7530%  72.1   25s\n",
      " 34845 27542 -1976892.4   65  106 -141148.27 -1.075e+07  7519%  65.1   30s\n",
      " 39271 30676 -364836.85   94  214 -141148.27 -9959925.6  6956%  63.6   58s\n",
      " 39275 30679 -4491000.7   62  213 -141148.27 -9959925.6  6956%  63.6   60s\n",
      " 39281 30683 -3356324.9   74  217 -141148.27 -9959925.6  6956%  63.6   66s\n",
      " 39284 30685 -2386579.2   95  218 -141148.27 -9959925.6  6956%  63.6   71s\n",
      "H39285 29150                    -141267.0929 -9959925.6  6950%  63.6   74s\n",
      " 39287 29152 -2747118.1   76  218 -141267.09 -9959925.6  6950%  63.6   75s\n",
      " 39290 29154 -1693230.6   71  220 -141267.09 -9959925.6  6950%  63.6   80s\n",
      " 39294 29156 -4904481.2   45  219 -141267.09 -9959925.6  6950%  63.6   85s\n",
      " 39298 29159 -608774.37  114  220 -141267.09 -9959925.6  6950%  63.6   91s\n",
      " 39301 29161 -2631269.3   76  220 -141267.09 -9959925.6  6950%  63.6   96s\n",
      " 39304 29163 -5045135.3   57  218 -141267.09 -9959925.6  6950%  63.6  101s\n",
      "H39305 27705                    -141359.0675 -9959925.6  6946%  63.6  104s\n",
      " 39307 27706 -4431371.0   63  220 -141359.07 -9959925.6  6946%  63.6  106s\n",
      "H39308 26321                    -144387.7954 -9959925.6  6798%  63.6  108s\n",
      "H39309 25005                    -145116.3181 -9959925.6  6763%  63.6  109s\n",
      "H39309 23754                    -147544.8686 -9959925.6  6650%  63.6  109s\n",
      " 39311 23756 -4491000.7   62  221 -147544.87 -9959925.6  6650%  63.6  111s\n",
      " 39315 23758 -1135263.7   80  221 -147544.87 -9959925.6  6650%  63.6  115s\n",
      " 39319 23761 -6593776.4   49  221 -147544.87 -9959925.6  6650%  63.6  120s\n",
      " 39323 23764 -1326162.2  181  220 -147544.87 -9959925.6  6650%  63.6  126s\n",
      " 39326 23766 -4180782.1   65  221 -147544.87 -9959925.6  6650%  63.6  130s\n",
      "H39329 22578                    -150055.3747 -9959925.6  6538%  63.6  135s\n",
      " 39334 22581 -1166555.6  186  218 -150055.37 -9959925.6  6538%  63.5  140s\n",
      " 39338 22584 -956758.16  115  218 -150055.37 -9959925.6  6538%  63.5  145s\n",
      " 39342 22586 -830559.82   95  217 -150055.37 -9959925.6  6538%  63.5  151s\n",
      " 39345 22588 -675335.80  147  219 -150055.37 -9959925.6  6538%  63.5  155s\n",
      " 39350 22592 -1140810.1  165  219 -150055.37 -9959925.6  6538%  63.5  161s\n",
      "H39350 21461                    -153105.2952 -9959925.6  6405%  63.5  161s\n",
      " 39354 21463 -1279482.4  123  218 -153105.30 -9959925.6  6405%  63.5  166s\n",
      " 39358 21466 -1767288.9  121  220 -153105.30 -9959925.6  6405%  63.5  171s\n",
      "H39359 20392                    -153557.3871 -9959925.6  6386%  63.5  173s\n",
      " 39362 20394 -7574067.2   19  221 -153557.39 -9959925.6  6386%  63.5  176s\n",
      " 39365 20396 -1516693.8  118  219 -153557.39 -9959925.6  6386%  63.5  180s\n",
      " 39370 20399 -4421271.8   45  218 -153557.39 -9959925.6  6386%  63.5  186s\n",
      " 39373 20401 -388189.25   81  219 -153557.39 -9959925.6  6386%  63.5  190s\n",
      " 39378 20404 -9521765.6   29  217 -153557.39 -9959925.6  6386%  63.5  196s\n",
      " 39382 20407 -8281267.1   30  220 -153557.39 -9959925.6  6386%  63.5  200s\n",
      "H39382 19385                    -153759.3235 -9959925.6  6378%  63.5  200s\n",
      "H39385 18416                    -159640.5952 -9959925.6  6139%  63.5  205s\n",
      "H39385 17493                    -163321.9781 -9959925.6  5998%  63.5  205s\n",
      " 39390 17496 -1693230.6   71  218 -163321.98 -9959925.6  5998%  63.5  211s\n",
      " 39394 17499 -4904481.2   45  219 -163321.98 -9959925.6  5998%  63.4  216s\n",
      " 39398 17502 -608774.37  114  218 -163321.98 -9959925.6  5998%  63.4  220s\n",
      " 39402 17504 -1254002.2   73  220 -163321.98 -9959925.6  5998%  63.4  225s\n",
      " 39406 17507 -626813.31  127  217 -163321.98 -9959925.6  5998%  63.4  230s\n",
      " 39410 17510 -2061995.2  103  218 -163321.98 -9959925.6  5998%  63.4  235s\n",
      " 39414 17512 -8281267.1   30  217 -163321.98 -9959925.6  5998%  63.4  240s\n",
      " 39418 17515 -3041269.1   67  219 -163321.98 -9959925.6  5998%  63.4  245s\n",
      "H39419 16638                    -166031.2813 -9959925.6  5899%  63.4  247s\n",
      " 39423 16640 -1326162.2  181  220 -166031.28 -9959925.6  5899%  63.4  251s\n",
      " 39427 16643 -1964734.8   88  220 -166031.28 -9959925.6  5899%  63.4  255s\n",
      " 39431 16646 -539113.90  145  221 -166031.28 -9959925.6  5899%  63.4  260s\n",
      "H39433 15812                    -178694.6611 -9959925.6  5474%  63.4  264s\n",
      " 39435 15813 -1811233.7   96  221 -178694.66 -9959925.6  5474%  63.4  265s\n",
      " 39438 15815 -956758.16  115  220 -178694.66 -9959925.6  5474%  63.4  270s\n",
      " 39442 15818 -830559.82   95  219 -178694.66 -9959925.6  5474%  63.4  275s\n",
      " 39446 15821 -922832.27  161  219 -178694.66 -9959925.6  5474%  63.4  280s\n",
      " 39451 15824 -5228560.7   54  221 -178694.66 -9959925.6  5474%  63.4  286s\n",
      " 39454 15826 -1279482.4  123  218 -178694.66 -9959925.6  5474%  63.4  290s\n",
      " 39458 15829 -1767288.9  121  220 -178694.66 -9959925.6  5474%  63.3  295s\n",
      " 39463 15832 -4471267.9   55  219 -178694.66 -9959925.6  5474%  63.3  301s\n",
      " 39466 15834 -4006446.0   52  219 -178694.66 -9959925.6  5474%  63.3  305s\n",
      " 39471 15837 -364836.85   94  221 -178694.66 -9959925.6  5474%  63.3  310s\n",
      " 39552 15915 -6969613.5   31  188 -178694.66 -9959925.6  5474%  64.3  315s\n",
      "H40559 15760                    -179289.1822 -9959925.6  5455%  67.1  317s\n",
      "H40564 15003                    -191996.2821 -9959925.6  5088%  67.1  317s\n",
      "H40565 14285                    -195034.8370 -9959925.6  5007%  67.1  317s\n",
      "H41649 14234                    -196058.3441 -9959925.6  4980%  67.7  318s\n",
      "H42875 14353                    -214411.6587 -9959925.6  4545%  67.4  319s\n",
      "H42921 13722                    -218119.9659 -9959925.6  4466%  67.4  319s\n",
      "H44016 13740                    -290894.6151 -9959925.6  3324%  67.0  319s\n",
      " 44144 14032 -2345943.3  115   54 -290894.62 -9959925.6  3324%  66.9  320s\n",
      "*45014 12136             222    -1579628.883 -9959925.6   531%  66.5  320s\n",
      "H48219 13009                    -1581429.825 -9959925.6   530%  67.7  324s\n",
      " 49460 13662 -7318474.7   45  149 -1581429.8 -9959925.6   530%  68.0  325s\n",
      " 58518 18801 -5653072.4   62  141 -1581429.8 -9959925.6   530%  68.7  330s\n",
      " 67585 22916 -9103103.9   40  162 -1581429.8 -9959925.6   530%  69.0  335s\n",
      " 76786 30410 -5382488.6   49  144 -1581429.8 -9959925.6   530%  70.6  340s\n",
      "H83293 35306                    -1583278.113 -9959925.6   529%  70.8  343s\n",
      "H84260 36157                    -1586048.974 -9959925.6   528%  71.0  344s\n",
      "H85055 36150                    -1586337.962 -9959925.6   528%  71.0  344s\n",
      " 85291 38276 -6559895.0   49  150 -1586338.0 -9959925.6   528%  71.0  345s\n",
      "*86823 38163             524    -1600246.618 -9959925.6   522%  70.5  345s\n",
      "*86825 38163             525    -1600357.619 -9959925.6   522%  70.5  345s\n",
      "*86826 38161             525    -1600584.185 -9959925.6   522%  70.5  345s\n",
      "H89887 39957                    -1601947.923 -9959925.6   522%  70.2  346s\n",
      "H89968 39954                    -1602507.128 -9959925.6   522%  70.2  346s\n",
      "H91756 42302                    -1602793.969 -9959925.6   521%  70.2  348s\n",
      "H93160 43001                    -1602904.805 -9959925.6   521%  70.1  349s\n",
      " 94856 44714 -4141844.5   66  116 -1602904.8 -9959925.6   521%  70.5  350s\n",
      "H94939 44703                    -1603397.263 -9959925.6   521%  70.5  350s\n",
      "H95236 44700                    -1603885.096 -9959925.6   521%  70.6  350s\n",
      "H95390 44695                    -1604113.000 -9959925.6   521%  70.6  350s\n",
      "H95623 44675                    -1605774.899 -9959925.6   520%  70.7  350s\n",
      "H97747 46799                    -1606912.705 -9959925.6   520%  70.7  352s\n",
      "H97840 46798                    -1607098.159 -9959925.6   520%  70.7  352s\n",
      "H98159 46792                    -1607554.729 -9959925.6   520%  70.8  352s\n",
      "H98267 46733                    -1611617.353 -9959925.6   518%  70.8  352s\n",
      "H100489 49164                    -1611864.456 -9959925.6   518%  70.7  354s\n",
      "H101104 49162                    -1611945.705 -9959925.6   518%  70.7  354s\n",
      "H101215 49157                    -1612192.510 -9959925.6   518%  70.7  354s\n",
      "H101297 49152                    -1612512.870 -9959925.6   518%  70.7  354s\n",
      " 101650 50075 -5975603.0   56  145 -1612512.9 -9954048.7   517%  70.7  355s\n",
      "H102412 50074                    -1612567.782 -9953596.3   517%  70.6  355s\n",
      "H104140 51714                    -1619066.186 -9932061.6   513%  70.8  357s\n",
      "H106967 54624                    -1619321.567 -9916729.2   512%  71.1  359s\n",
      "H107753 54611                    -1620047.596 -9910574.0   512%  71.1  359s\n",
      "H108144 54598                    -1620743.433 -9909508.2   511%  71.1  359s\n",
      " 109713 56426     cutoff   64      -1620743.4 -9893392.3   510%  71.4  360s\n",
      "H109714 56390                    -1622887.346 -9893392.3   510%  71.4  360s\n",
      "H109763 56387                    -1623096.131 -9893392.3   510%  71.4  360s\n",
      "H110912 57563                    -1624533.127 -9884891.0   508%  71.7  361s\n",
      "H111145 57558                    -1624878.574 -9884891.0   508%  71.7  361s\n",
      "H111343 57515                    -1627097.898 -9884891.0   508%  71.7  361s\n",
      "H114458 60276                    -1628442.166 -9864303.3   506%  71.7  363s\n",
      "H114571 60163                    -1630486.349 -9864303.3   505%  71.8  363s\n",
      "H115688 61123                    -1630521.956 -9864303.3   505%  71.9  364s\n",
      "H115726 60983                    -1635096.727 -9864303.3   503%  71.9  364s\n",
      "H116670 60926                    -1636875.519 -9864303.3   503%  71.9  364s\n",
      " 116722 62059 -8149247.1   43  168 -1636875.5 -9864303.3   503%  71.9  365s\n",
      "H118171 63512                    -1637568.936 -9825756.0   500%  71.9  366s\n",
      "H118510 63402                    -1641301.112 -9825756.0   499%  71.9  366s\n",
      "H119374 63376                    -1642420.809 -9815928.2   498%  71.9  366s\n",
      "H121925 65725                    -1645112.347 -9799444.7   496%  71.7  368s\n",
      "H121929 65676                    -1646589.269 -9799444.7   495%  71.7  368s\n",
      "H122030 65589                    -1649327.955 -9799444.7   494%  71.7  368s\n",
      "H122329 65556                    -1651036.061 -9799444.7   494%  71.6  368s\n",
      "H123074 66098                    -1651939.200 -9798305.6   493%  71.6  369s\n",
      "H124728 67829                    -1652008.399 -9770432.3   491%  71.8  370s\n",
      "H124969 67822                    -1652426.599 -9768307.1   491%  71.8  370s\n",
      "H125404 67786                    -1653786.026 -9764042.1   490%  71.9  370s\n",
      "H125587 67755                    -1654690.067 -9764001.3   490%  72.1  370s\n",
      "H127221 69478                    -1655223.220 -9746464.9   489%  72.3  372s\n",
      "H130224 71897                    -1655527.699 -9723854.7   487%  72.3  374s\n",
      " 130874 73198 -3669927.3   74  116 -1655527.7 -9723854.7   487%  72.3  375s\n",
      "H132429 74512                    -1656686.367 -9710858.6   486%  72.4  376s\n",
      "H132782 74489                    -1657209.785 -9708982.5   486%  72.4  376s\n",
      "H133429 74484                    -1657319.600 -9706793.3   486%  72.4  376s\n",
      "H135656 76865                    -1658337.432 -9686155.1   484%  72.3  378s\n",
      "H135761 76823                    -1659235.533 -9686155.1   484%  72.4  378s\n",
      "H136021 76802                    -1659972.019 -9686155.1   484%  72.4  378s\n",
      "H136175 76726                    -1664283.586 -9686155.1   482%  72.5  378s\n",
      " 138680 79062 -6196642.3   57  132 -1664283.6 -9686155.1   482%  72.7  381s\n",
      "H139038 79007                    -1669042.026 -9686155.1   480%  72.7  381s\n",
      "H139039 79001                    -1669453.303 -9686155.1   480%  72.7  381s\n",
      "*140796 79538             240    -1742471.501 -9685944.9   456%  72.6  382s\n",
      " 146328 83883 -3957370.8   72  114 -1742471.5 -9651587.2   454%  72.6  385s\n",
      " 155171 91182 -2668741.9   80  109 -1742471.5 -9598391.7   451%  72.7  390s\n",
      " 162606 97677 -6074588.4   47  142 -1742471.5 -9545761.0   448%  73.1  395s\n",
      " 171379 103838 -5142144.3   35  155 -1742471.5 -9494726.0   445%  73.6  400s\n",
      " 178814 110126 -8071241.8   44  165 -1742471.5 -9451260.8   442%  73.5  405s\n",
      "H181885 109043                    -1820002.611 -9436557.1   418%  73.4  407s\n",
      "H183551 111187                    -1820373.270 -9430083.6   418%  73.4  408s\n",
      "H183712 111187                    -1820397.448 -9430083.6   418%  73.4  408s\n",
      "H184093 111163                    -1821098.879 -9428574.5   418%  73.4  408s\n",
      "H186424 113484                    -1821305.691 -9413358.3   417%  73.4  410s\n",
      "H186469 113416                    -1823940.809 -9413358.3   416%  73.4  410s\n",
      "H186758 113377                    -1825309.641 -9413358.3   416%  73.4  410s\n",
      "H187018 113361                    -1825743.311 -9413358.3   416%  73.4  410s\n",
      "H187225 113354                    -1825899.268 -9413358.3   416%  73.4  410s\n",
      "H188771 115361                    -1829600.358 -9412957.0   414%  73.6  412s\n",
      "H190088 116570                    -1829678.721 -9400897.4   414%  73.6  413s\n",
      "H190218 116554                    -1830296.827 -9400897.4   414%  73.6  413s\n",
      "H191812 117899                    -1832541.264 -9394180.7   413%  73.7  414s\n",
      " 193289 119006 -8767020.0   45  161 -1832541.3 -9387961.4   412%  73.7  415s\n",
      "H193370 118996                    -1832840.902 -9387961.4   412%  73.7  415s\n",
      "H193469 118986                    -1833143.319 -9387961.4   412%  73.6  415s\n",
      "H193663 118982                    -1833280.249 -9387770.8   412%  73.6  415s\n",
      "H194464 118968                    -1833624.132 -9383843.8   412%  73.6  415s\n",
      "H196786 121483                    -1833838.961 -9376858.1   411%  73.6  418s\n",
      "H197123 121479                    -1833993.346 -9376354.3   411%  73.6  418s\n",
      "H197539 121476                    -1834235.079 -9375582.0   411%  73.6  418s\n",
      "H199462 123406                    -1834430.516 -9367663.9   411%  73.6  420s\n",
      "H199615 123370                    -1835547.911 -9367663.9   410%  73.6  420s\n",
      "H202425 125447                    -1835829.118 -9357237.1   410%  73.6  421s\n",
      "H204566 127992                    -1835901.123 -9347419.7   409%  73.8  423s\n",
      "H206331 129152                    -1836058.166 -9347419.7   409%  73.9  424s\n",
      "H206602 129143                    -1836397.371 -9347419.7   409%  73.9  424s\n",
      "H206929 129136                    -1836665.704 -9347419.7   409%  73.9  424s\n",
      " 207484 130381 -2035956.1  104   78 -1836665.7 -9347419.7   409%  74.0  425s\n",
      "H209340 131105                    -1836927.103 -9337734.9   408%  74.0  427s\n",
      "H209719 131102                    -1836978.984 -9337734.9   408%  74.0  427s\n",
      "H209967 131077                    -1837400.963 -9337734.9   408%  74.0  427s\n",
      " 212765 134418     cutoff   94      -1837401.0 -9322584.2   407%  73.9  430s\n",
      "H215972 136594                    -1837496.117 -9307701.8   407%  74.0  432s\n",
      " 219731 140062 -5851296.2   49  159 -1837496.1 -9289357.6   406%  74.3  435s\n",
      " 226801 145689 -7861050.5   44  150 -1837496.1 -9267365.3   404%  74.2  440s\n",
      "H229799 146827                    -1837692.255 -9255389.7   404%  74.2  442s\n",
      "H229800 146808                    -1838489.191 -9255389.7   403%  74.2  442s\n",
      "H229801 146793                    -1838776.683 -9255389.7   403%  74.2  442s\n",
      "H229814 146778                    -1839135.167 -9255389.7   403%  74.2  442s\n",
      "H229845 146768                    -1839391.171 -9255389.7   403%  74.2  442s\n",
      " 232972 149430 -8100864.2   41  182 -1839391.2 -9246316.4   403%  74.1  445s\n",
      "H233006 149229                    -1845276.532 -9246316.4   401%  74.1  445s\n",
      " 239250 155249 -6059499.1   49  147 -1845276.5 -9227283.6   400%  73.9  450s\n",
      " 247381 162308 -8185467.3   41  165 -1845276.5 -9201508.4   399%  74.0  455s\n",
      " 254559 167776 -6355769.1   47  152 -1845276.5 -9184638.4   398%  74.1  460s\n",
      " 263516 175351 -7367613.5   40  148 -1845276.5 -9152002.7   396%  74.6  465s\n",
      " 268922 179547 -8585786.5   38  177 -1845276.5 -9140930.1   395%  74.7  470s\n",
      " 277719 186314 -7752944.1   44  161 -1845276.5 -9113242.1   394%  74.7  475s\n",
      "*282666 188962             232    -1884841.034 -9100706.3   383%  74.6  479s\n",
      "H284115 189742                    -1886846.721 -9097819.8   382%  74.6  480s\n",
      "H284115 189431                    -1894179.030 -9097819.8   380%  74.6  480s\n",
      "H285216 190306                    -1897256.452 -9091597.6   379%  74.6  481s\n",
      "H285394 190208                    -1898782.194 -9091580.2   379%  74.6  481s\n",
      "H285529 190034                    -1901876.609 -9091269.4   378%  74.6  481s\n",
      "H286661 191491                    -1902238.619 -9088637.5   378%  74.6  482s\n",
      "H288834 192495                    -1902263.902 -9085578.1   378%  74.5  483s\n",
      "H289398 192481                    -1902585.254 -9085331.0   378%  74.5  483s\n",
      "H290726 194318                    -1903544.833 -9079133.5   377%  74.6  485s\n",
      "H291510 194315                    -1903596.665 -9078780.0   377%  74.6  485s\n",
      "H291595 194285                    -1904173.186 -9078780.0   377%  74.6  485s\n",
      "H294057 196701                    -1904234.652 -9071312.8   376%  74.5  487s\n",
      "H294533 196591                    -1907099.922 -9071303.3   376%  74.5  487s\n",
      "H295644 197131                    -1908308.402 -9067271.5   375%  74.6  488s\n",
      "H297365 199239                    -1908735.996 -9059613.3   375%  74.6  490s\n",
      "H298753 200349                    -1909305.754 -9056275.7   374%  74.5  491s\n",
      "H298991 200333                    -1909962.430 -9056275.7   374%  74.5  491s\n",
      "H299112 200331                    -1910015.879 -9056275.7   374%  74.5  491s\n",
      "H299568 200316                    -1910331.546 -9053696.8   374%  74.5  491s\n",
      "H301853 201696                    -1911977.773 -9044379.2   373%  74.5  494s\n",
      " 301875 203044 -4363678.9   63  126 -1911977.8 -9044379.2   373%  74.5  495s\n",
      "H303596 204566                    -1913114.393 -9040313.6   373%  74.5  496s\n",
      "H304899 204545                    -1913544.428 -9037818.1   372%  74.5  496s\n",
      "H307269 206805                    -1913844.243 -9034818.7   372%  74.5  498s\n",
      "H307463 206781                    -1914253.458 -9034818.7   372%  74.4  498s\n",
      "H307463 206775                    -1914439.537 -9034818.7   372%  74.4  498s\n",
      " 309985 208885 -3361306.9   73  108 -1914439.5 -9028118.7   372%  74.4  500s\n",
      "H310331 208791                    -1916546.433 -9028118.7   371%  74.4  500s\n",
      "\n",
      "Cutting planes:\n",
      "  Gomory: 67\n",
      "  Implied bound: 15\n",
      "  MIR: 1909\n",
      "  Flow cover: 916\n",
      "  RLT: 1701\n",
      "  Relax-and-lift: 22\n",
      "\n",
      "Explored 310638 nodes (23126772 simplex iterations) in 500.02 seconds (1102.39 work units)\n",
      "Thread count was 14 (of 14 available processors)\n",
      "\n",
      "Solution count 10: -1.91655e+06 -1.91444e+06 -1.91425e+06 ... -1.90996e+06\n",
      "\n",
      "Time limit reached\n",
      "Best objective -1.916546432930e+06, best bound -9.028118714315e+06, gap 371.0618%\n",
      "[CHECK MLP] obj(x_ip)=-1.91655e+06  ip_y=-1.91655e+06  rel_err=1.002e-07\n",
      "\n",
      "--- Dataset stats (quadratic) ---\n",
      "  X: shape=(5000, 32)  mean(mean)=-0.0274  std(mean)=58  min=-100  max=100\n",
      "  y: shape=(5000,)  mean=3.01e+04  std=2.37e+05  min=-8.87e+05  max=9.32e+05\n",
      "\n",
      "\n",
      "=== Run: quadratic | DFN ===\n",
      "  data: N=5000  train/val/test=3500/750/750  dim=32\n",
      "  model: params=12,323 layers=[64, 160, 64] p_list=[0.1, 0.1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=500  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "None"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ===================== Quadratic DATASET =====================\n",
    "N_SEEDS = 1\n",
    "\n",
    "VARY_DATASET_SEED = False\n",
    "VARY_MODEL_INIT_SEED = True\n",
    "STRICT_IP_CHECK = False\n",
    "IP_CHECK_TOL = 1e-4\n",
    "\n",
    "SILENCE_LOCAL_SEARCH = False\n",
    "ALLOW_PLOTS_MULTI_SEED = True\n",
    "\n",
    "dataset_type = \"quadratic\"\n",
    "dataset_params = dict(\n",
    "    K=5000, dim=32, eigen_min=-1, eigen_max=15.0,\n",
    "    x_min=-100, x_max=100, noise_std=0.1, seed=0\n",
    ")\n",
    "in_dim = int(dataset_params[\"dim\"])\n",
    "\n",
    "train_base = dict(\n",
    "    epochs=500,\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=5,\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=[64, 160, 64], p_list=[0.1, 0.1],\n",
    "    seed=0, alpha=5e-3, beta=-2.0\n",
    ")\n",
    "\n",
    "# ---- DFN (fixed A = I) ----\n",
    "dfnA_layer_sizes = [5, 400, 6]\n",
    "# assert dfnA_layer_sizes[0] + dfnA_layer_sizes[-1] == in_dim + 1, \"Need |L1|+|LK| = dim+1 for A_fixed=I\"\n",
    "\n",
    "dfn_Afix_params = dict(\n",
    "    input_dim=in_dim,\n",
    "    layer_sizes=dfnA_layer_sizes,\n",
    "    p_list=[1, 1],\n",
    "    seed=0,\n",
    "    alpha=5e-3,\n",
    "    beta=0.0,\n",
    "    A_fixed=np.eye(in_dim, dtype=np.float32),  # shape (dim, dim)\n",
    ")\n",
    "\n",
    "# ---- other models ----\n",
    "mlp_params  = dict(in_dim=in_dim, hidden_dims=[128, 128], out_dim=1)\n",
    "maff_params = dict(in_dim=in_dim, n_pieces=1600)\n",
    "lset_params = dict(in_dim=in_dim, n_pieces=1600, T=0.05)\n",
    "\n",
    "lr_map = dict(DFN=1e-1, MLP=1e-3, MaxAffine=1e-3, LSET=1e-3)\n",
    "\n",
    "time_limit = 500\n",
    "x0    = np.array([50,-20,1,-10,7,30,0,2,-30,-11], dtype=int)\n",
    "xmin  = np.full(in_dim, -100, dtype=int)\n",
    "xmax  = np.full(in_dim,  100, dtype=int)\n",
    "delta = 4\n",
    "sum_eq = int(x0.sum())\n",
    "\n",
    "runs = [\n",
    "    (\"MLP\",        \"MLP\", mlp_params),\n",
    "    (\"DFN\",        \"DFN\", dfn_params),\n",
    "    # (\"DFN_AfixI\",  \"DFN\", dfn_Afix_params),\n",
    "    # (\"MaxAffine\",  \"MaxAffine\", maff_params),\n",
    "    # (\"LSET\",       \"LSET\", lset_params),\n",
    "]\n",
    "\n",
    "_ = run_benchmark(\n",
    "    dataset_type=dataset_type,\n",
    "    dataset_params=dataset_params,\n",
    "    runs=runs,\n",
    "    train_base=train_base,\n",
    "    lr_map=lr_map,\n",
    "    x0=x0, xmin=xmin, xmax=xmax,\n",
    "    delta=delta, sum_eq=sum_eq,\n",
    "    n_seeds=N_SEEDS,\n",
    "    vary_dataset_seed=VARY_DATASET_SEED,\n",
    "    vary_model_init_seed=VARY_MODEL_INIT_SEED,\n",
    "    strict_ip_check=STRICT_IP_CHECK,\n",
    "    ip_check_tol=IP_CHECK_TOL,\n",
    "    silence_local_search=SILENCE_LOCAL_SEARCH,\n",
    "    allow_plots_multi_seed=ALLOW_PLOTS_MULTI_SEED,\n",
    "    time_limit=time_limit,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c230cfb2-cd2f-416e-a270-25b42d01fed6",
   "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
}
