{
 "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": 22,
   "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",
      " [[-5. -9.  0.  7.  3. -9.  3. -3. -6. -1.]\n",
      " [ 8. 10. -8.  1.  5. -5. -5. -5. -6.  8.]\n",
      " [-6. -6. -8. -8.  6. -4.  6.  2.  8.  1.]\n",
      " [ 6.  7. -9.  1. -1. -4. -1. -2.  0.  7.]\n",
      " [ 7.  3.  4. 10.  3. -3. -9.  1. -6.  2.]] \n",
      " [16987.797  9943.988  8211.554  5990.925 11936.206] \n",
      " 2931.8052 \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=(2000, 10)  mean(mean)=-2.39  std(mean)=579  min=-1e+03  max=1e+03\n",
      "  y: shape=(2000,)  mean=4.77e+06  std=1.35e+07  min=-5.19e+07  max=5.46e+07\n",
      "\n",
      "\n",
      "=== Run: quadratic | MLP ===\n",
      "  data: N=2000  train/val/test=1400/300/300  dim=10\n",
      "  model: params=18,049 hidden=[128, 128]\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.001  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA94AAAGGCAYAAACNL1mYAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjgsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvwVt1zgAAAAlwSFlzAAAPYQAAD2EBqD+naQAA1k5JREFUeJzsnQd4FGX3xW8SUqih9957x4LSLDQrIIK9oaKoKPbP7qeC5UNQwa6ofwsWsCOCVEV67733lkACqft/zmwmeXd2ZnZ2s5tsNuf3PAvZ2dnZaTs75733nhvlcrlcQgghhBBCCCGEkJAQHZrFEkIIIYQQQgghhMKbEEIIIYQQQggJMYx4E0IIIYQQQgghIYTCmxBCCCGEEEIICSEU3oQQQgghhBBCSAih8CaEEEIIIYQQQkIIhTchhBBCCCGEEBJCKLwJIYQQQgghhJAQQuFNCCGEEEIIIYSEEApvh0yaNEmioqIsH3PmzJHCZOfOndp6vPHGGwEv4+mnn5bLL79catWqpS3r1ltvtZx3+/btMnDgQClfvryUKVNGLr30Ulm+fLnpvN988420b99eEhISpGbNmvLggw/K6dOnvebDNLyGeTAv3oP3WvHzzz9LiRIl5MiRI9rzcePGaevUoEEDbf179uxp+d7Dhw9r21e5cmUpVaqUnH/++fLXX3+Zzjtz5kztdcyH+fE+vN9IRkaGvPDCC1K/fn2Jj4+X5s2by9tvvy1Owfuef/55KaxzG+eQk/mszneXyyWNGzc23ffHjh2TJ598Ulq2bCmlS5eWxMREbf/cdNNNsnr16gL/nh09elRGjhyZe6yqVasm/fr1k+PHjzt6P44r1h/vxfmG447jbzxv8L3A+Yz5qlatKhdddJH8/vvvXsvD/jLb3r59+0oocHJO79mzRwYMGCANGzbMPWYdOnSQd955RzIzMx1/1osvvqgd9+zsbCmKPPPMM9KxY8ciu/6ERAK4FpUsWVJOnjxpOc8NN9wgsbGxcujQIcfLxXW2MH53ffHKK6/Ijz/+6DV9/fr12vr6+r0OBZ9//rlUqVJFTp06VWCfuXfvXu2+sEePHtr9Jo4X7hOMJCcny8svv6z9llavXl27L23Tpo28+uqrcvbsWa/5t27dqt1/1K1bVzuvGjVqJKNGjdLuVVQwz9VXXx3SbSTFixKFvQJFjU8//VS74TaCG8uizptvvilt27aVK6+8Uj755BPL+SB0u3XrJhUqVNDmg0gePXq0dsFbsmSJNGvWLHfeL7/8Um688UYZNmyYtvzNmzfL448/rv14/Pnnnx7LhWjG+8eMGSNNmzaVr776Sq677jrthvf666/3Wo8ffvhBunfvrv0QgPfee08TCBA3v/zyi+X6p6WlycUXX6z9gI8fP14TRBMmTNBEDgQJLvA6c+fO1QTZZZddJj/99JMmTrD+eP/SpUs1QaVz7733yhdffCH//e9/pUuXLjJ9+nRN3OFH6j//+Y9ECmXLlpWPP/7YS1xjX23btk173Tigct5552n/P/roo9KuXTs5c+aMdi5MmTJFVq5cqZ13BfU9279/v3b+YtAGoqpJkyaaEJ89e7akp6f7fD9+3PG+J554Qnr37q2dsxi02rdvn3zwwQe58+EHvFWrVtq5jxsBiHqcoziXcJ7ge6ECgYvviwpuNIKN03M6JSVFypUrp20rbk6wbzBocP/992vH7KOPPnK0r1977TXtRik6umiO8z7yyCPaYMNnn30mt912W2GvDiHFkjvuuEMTorgvwG+tkaSkJJk6daoWPMBAalEHwvuaa67xEn24d8JAL35/MXBcUKSmpmr3MfitMP7GhxIIZPwuIhDTv39/+frrr03n2717txZ8gVCGgIbwnj9/vjZIMWPGDO0B0a7fw+KeBL9vuF/D79uKFSvkueee0+4Dli1blvt7hffjXmTWrFnavSUh+cZFHPHpp5+6sLuWLFkSlntsx44d2vq9/vrrAS8jKysr9+/SpUu7brnlFtP5Hn30UVdsbKxr586dudOSkpJclStXdl177bW50zIzM101atRw9e7d2+P9X375pbauv//+e+603377TZv21Vdfecx76aWXumrWrKktSyU9Pd1Vvnx51zvvvGO6/q1atXL16NHDdP0nTJigfdaCBQtyp2VkZLhatmzpOuecczzm7dKlizYdr+v8888/2vsnTpyYO23t2rWuqKgo1yuvvOLx/jvvvNNVsmRJ17Fjx1y+qFevnuu5555zFda5jXPIyXzDhg3TtgnHXOXGG290nX/++V77/pNPPtHeN2vWLNPlqsetIL5nV111latWrVqu48eP+/3eo0ePuhISElx33XWXx/SXX35ZO/7r1q2zfT/OW3x2t27dPKZjf2G/FQROz2kr8B0vUaKE6+zZsz7nfeyxx7TtVY9xQZGSkhK0Zd13332upk2burKzs4O2TEKIc3APgHuBTp06mb7+7rvvatewX375xa/divcUxu+uL6zuwb777jttnWfPnl2g10v8NuC378SJE66CRP3twH0Bth33CUZOnz6tPYzgnhjvmT9/fu60Dz/8UJs2c+ZMj3lx/4bpy5cv95h++eWXa/eihASDohmCCHMwqnbffffJ+++/r0VuEUFCpM4sbXrt2rVy1VVXadFjPb0akRUjiM4+/PDDWlRMT1vF6N/GjRu95h07dqyW/ooRP6STLly40NF6O41IYVQZI3/16tXLnYaRQ0SsEWnW01DxuQcOHPCKEg0ePFhbNyxHXSam4TUVvBdRs0WLFnlMR1o4RriRfhbI+iMqj32jg+gnIpCLFy/WIpcA/yOaiRFUvK7TtWtX7biq64+RePyGG7cVzxHd/eOPPyQYYEQX5xdGgY1gJDouLk6L3gKM8OLcql27tnZuIQ387rvvzn09UJCFANSRZxwLZCDcfvvtXvPrqVs1atQwXV5BRkKRnocShTvvvFP7zvkLjiPS1syOM46/WWqgCtIgEcVWzyd/OXjwoHYccVxxvPVUdyfp3/6c01YgwwTHLCYmxnY+RMiRGYFsFfUYq2UxTq5VOF56WjwiLUjf//fffz3mQVQCy0S5C6JEOLZIHQSICiEK9uuvv2qp8kgrbNGihfYcIBqP58iWOeecc7SovxHsL2RoIBpCCCl4cL255ZZbtGjkmjVrvF5HlhR+Y5DNg4gmouK478K1RS/zQQQ0UN59910tWwvLw3UIUVBjJhuur3fddZfUqVNHuzajzAjXIz31Hb8duI/DfR5KdypWrKhd25B5pIJrGTKOcC+olx0hwo1rlX6P1KtXr9zX1NRrZO0hewn3ZLhmXnDBBV5ldHbXS7vtv+KKK7yysPT7XWRx4TqKz8R+0q+v+cXp/QGu33gYwTVdL51Sf4cBjoGKvm24XzJe/7FfkdFHSH6h8PaTrKws7QZXfWCa2c3iW2+9pdU3fv/995pIhWDB3zqbNm3SbnjXrVunzYu0W/xQoN4S6Zk6SFW+8MILNSGPG3yIW6Ss4kYZwlYFKdMQXBBoSM/BxRsCHcIoGEBE4uJjTA0GmIbXUf+tDyro01Vw0cOPlv66Pi8u2kZBor9XnRdA5OEHCz9s/oJlWa0/wPGwW399mnH9IUiQUuxk/QMFgwP4QTfWOOEc/L//+z/thxE1uwDHCfsIP5hI63/22We1AQycS8Z6ZH/ADzp+rNVyBIhw/EAOGTLEa359gOPmm2/WhKmxhirQ75nZPGYPtTYXN14QyDhv8H3ETRR+ZHFTYxRzZujHEbVjKrjhw343O874fKwHBpCQygYBh5svIzheuBHDdwA3QU899ZT2fTKKbtxIoIwBx3PatGlaCiZKPTCY4HT9nZzTOthfWP8TJ07I5MmTtXMP6+9r8ADnGo41bhDNcHKtQlopBo9wzuEcg5DHeuB4/f33317LxOAfBpi+++477Rqps2rVKs1jAINTuM7ihgvz4nggZR5pnVgHfDZEunG/d+rUSTtXfvvtN9ttJoSEDgzsQugZS+GQfo1BcwhzCHTdqwPfb3xnIcoRtMB1IxCfEARNIORRhobBSfyOPfTQQ9o1SxXdKDHD60h1xrUZ1zZca3DN0svcsG4oX8EycE3D7zGuRaif1sFvEQYIcT3E33hMnDhRKw/CtUq/fuqvYTrAPQDKn3C9hGj/9ttvtd+UPn36mHrYWF0vzeqsMdhhdS3HPkY5Du53cW+Gz0RQRL8XVH9HnDyCCVLEAcq+dJC+j/Ry/I7hfg9lcPPmzdPKHHEPhXtRFZw3WH8zfxZC/CYocfNigJ4Ca/aIiYnxmBfTkIp78OBBjzSp5s2buxo3bpw7bejQoa74+HjX7t27Pd7fr18/V6lSpVwnT57Unr/44ovaMmfMmOEz1bxNmzYeadmLFy/Wpn/99ddBSXPat2+ftrzRo0d7vYY0cTWFG+m3eH7gwAGveZF+jtRNnSZNmrj69OnjNd/+/fu1Zagp3Ng+pLX/73//s1x/u1RzpMnffffdXtOx3mq6u54S/++//3rNi1TjuLi43OdIQ2rWrJnp52E+Y2pyflLNBw4c6Kpdu7ZHChbS9u3S7JAii9TiXbt2afP99NNPAaeaI90LaW74Gyn2evryrbfearnvcQ5jP+jfmQYNGriGDx/uWrVqVcDfM3yG1bzqQz2Pcd5iWrly5bSU8z/++MP1ww8/uNq2baul0RnXxwhKB/CdNQPns7GsAuC81tcFnztlyhSveZ566iktlQ/p+Ci7QGoz0rm7d+/ucZxx3pYpU0Y7jipvvPGGtnxfqe7+nNPGfYYH0umxrk549dVXtfeo10F/rlXYbqSWYj51H5w6dcpVtWpVV9euXXOn4XuD9z777LOm3ytcj/fu3Zs7beXKldr8KIVRUyx//PFHbfrPP//stZwLLrjAde655zradkJIaMB1H7//KNvRefjhh7Xv7ebNm03fg+sMfv8uvvhi14ABA/xONcf1GKVtdtx+++3avcX69esdb4u+XnfccYerQ4cO+Uo1x3WsYsWKriuuuMJjOq6d7dq18yijs7temjF58mRt/oULF3q9hunVqlVzJScn507DNT86OtrjPlG/Z3DysLoXsUs1NwO/57j2G4+5fm+J0jj1cwcPHmxZQoWSqSFDhjj6XELsoLman2BU0jgaphs2qCDVRzX4wCgsooFICcXoIdJEMRKH+ZCWpIKIN0ZLMZIJwy/8jej2JZdc4nP9MPKppoDqka1du3ZJMDHbZqvXrOZ1Op/xNZhDIV0ao7VFcf3zC7IeELVD6hNGtwFG9BFtR5qdDkyzEBXFaDSirWrkd8OGDZqJXqBg5B9RWUQecL4iffl///uf5fww6EIKHkaMEQldsGCBNsKOaCO+U3r6uj/fM2SAOHFX1TMAgL4P8P3DyLz+XUFUHiP/yDRB1MAOf48zHNBRKoLsFCwb1wFEI9Rtfumllzzeg0gHUqQRGUEaol5SgfQ9RB0QsVcjAzjumBffDWTNIBvAfU/kBtkIasqe03Ma4Pji2oNIDa5Zr7/+uhYZ9uXYj3MOy1P3vz/XKmQEYRlwtFXXHZHnQYMGaccfhj9IbdTBdDOQ2oluDTr6uYVIhvp+fbrZ9RLpqjjPCSGFBzJ8kD2FrEJ833EdxHUVhpkwytTB7wvMLhENR6RZx8y00xfIMkJEF9fsoUOHaunbxusa7tNwbTb+bhlBdBmRcGThqBFzY3qzv+A3FddoRP2NUWPcR+K3DZ+npmNbXS+N4DqsXwPNwHarhmu498W86nUUWUNOr5+BZDIaQUkTspdwf200AkUGAjKp8PuBTCfMg2wvGK3hvgj3TMaMLmyPXoZISH6g8PYTXFQ7d+7scz5jyrE6DemXuPHH/2Z1r/pFR0/JRb0S0mKcUKlSJY/nukOxMXUyUFALhJtps3RhPb0LaUbqumBeo8so5tXn0+d1skyAdH1cxAN19HT6Wer6m81rXH84PRvBDx1qXdV58wtEFs4biG0Ib/yI4CYEDuq6kIHAxGv4wYToRWo0fnAxHW6e+T0fcA5gAAAlEqhbw8AQbnzswDmA9+j10UjtwrZgvY3C28n3DEJZFZdWqKJNP6YQkqrow/5EXZpVSzz1/dheo+DTzwmcl0bUm0H8qGObR4wYoQlwu/o1lBVATKPuWRfeqBVEqYleo2ZEr9/HgB5EuA5uxpAi7s85rV639GsXzilcA+DojrRP1ExbgXMM62lVC+7rWmXnDYBrJM5lnPvqcbDyETBuF8o17KabtZ/BjXGwrqOEkMBAmRM6K+D3D8IRg7m4LqJtlA68I5BGPHz4cE1MQSTjOoTfQgw6+wtqfCFmP/zwQ+0zce1BWjkGTOE5od+n4b7ODgyYX3vttVqdNjp84LoKgYdyMLtOMk7Q68ixf6zANV4V3lbXSyP6dc9qcMB4Ldev5+r1EgOmGAB1Qn48UAAEPwYDsByk2Buv8zhXcL+G+fR9gPsXDMrACwBiHL+ZKrz+k2BB4R0iUItpNU2/SOF/Y422Orqoj6iidhhR8nAAdUcQPGbmJpiG11FLpdbBYrraBgo/YDCFU8UW5kW9E15TL7r657Ru3Vr7Hz94qKF64IEHAt4GfJbV+qufpf+P6YhAGufVX9eXiTowHGN10MW4zGCAGwjcCED0IpKKOliM6KuGXxi9xYg6xJb6A2JmyhYoiIQioo7IAlps+QtawUHIodYN0Xmr0XQrjOLSCl10WtU260DE+zJyUc/pc889N3c6jjtEr5PjjOgJTNpwo+ak7Y26TrgmYBus9rc+aGfMBtCvJf6c03brD1Crbie88ZkYdDJGWZyiXyetrpHYL0aDvGBmlpjdtFpF7wkhBQPuMXDvABGMawMEK6KtqjErIuDIZoGgVclP/2l90BjXMwwao34cEVVcB+Hh4+Q+DesFM0l4ZajXKjUiHyj6tQmZSBhcN8P4e+P0eqkvG9dAp2LdCH6rrWrEjezYsSPgwArEtF6TjXp+s8EQiG5kQBm3BYMpwMzrBNtekO3bSORCc7UQgVE2fQQSIPUTF1uk5+oXAggHpG7qQltNs0UUR794IkKGi7tuElHYIPqGdVFdIvGDhtFcRPR04Qxhggub0QgMEWuYWaip4lgmpiH9VwUpuRATushBOhVEjtMUKav1h/BXndL1dDV8ji5ecGGGyMB01dgLEUikwarrj7Ql/IgZHemx7bhRQKpXMMENAKJyGKzAZyBVWk2h039Q1T7juiALFtg/GLWHGYlxdFgF3wM1zV0H+3TLli3auR5Iv2psC1LXfD3g4KqD44vvH8zm1GOK7yAGKqxuWHRwHDHybTyn8Rz73Nhz1QhuBnADgu01ixKo6OeSuk640cNNAa4jyAgwPvRzF6796nT9hsGfc9oK3dkbA3B26OdjoE6w2AasLwaW1MwG3Pjq5orGrINQAqOgYPSRJ4TkP90c1y+UvSDijfRv9VqAa7Hxt2/16tWODDR9gUFE3JPB/BIDi7oZK6bh2ojrqBVYL2TVqIIX9zNGV3OziLE6HRhfQ/o7fleQWm/224CHntHjL/m9lqup5k4egaaao5c3RDfODdyjqp13VLB8DJIYU8f188Mo1nF/iPtdXv9JMGDE209w02vmuogbYYx4qiOESFlBahMu1HCkhNhTW4phxFSv2UTkEOkwSHFBfQnqcfRWB6hxhGiHuEOKJ26ccdHFDTxuxJ2OIvoCy0MUDuDChZFD3YUdNb369iH9Fa0jUKMJF0v8EMANEkJQFTmIzGI7EJ1F+yOMUkNoPfbYY1p6lipG8aOFaffcc48kJydrN/UQlYgMQiTo6apYH0TlkNpsBG2AUNcDsAzcrOvrj5FM/SKMFFk4gmKEHOuNSCuOD34wUTdtTEnCemFeuJoiMotjgHVQI8xwzMTNAI4p1hWfB3GHGjOkowUz1Vz/IYTwgJs1fhDwOcbXcU5iXbEf8PlIUYaLdDDB/vMFzhWIZLSVwn7BeY0fPdRd4aYF577xhsDJ9wzCzF8QJX3zzTe1dD98n3C+QcghHRHrAOdr9fuAwTGsHx4A+/Hpp5/Wvtf4GxF7XdwPGzbM44cZy0f6OtLrILIh7iHQsVycf/oAFZzWEcHGgBCyRfA9Qr0gjimuIRjY0MH3DccQ3RCQ9YF9gPlx3uMGFNkHvtIdnZ7TOJcxaILMBAhgZFfg+4hIE95rllavghsgXdTbZRrYHStcP2644QbtOodrCCJDuNnGujg594IF0t5x7UKKKyGkcIGIxDUFtdL4fcNvrwquF7im4xqGexf8tuPaiWhzIK7Z6BiBAXSIWwQTIJbx24vfMj1KiuXjuo3rJdqMITtKv2bC5Ry/yVgvBChw3UVKOH67sZ5YJq4vKng/Irb43cbriOrjeq9nJeH3AdMwEIztwm8Mot0YBEd0FsvHvQ3u6TCojP+NGQBOwYA1th/X8kC9YbCuTso0zdDv43SXdNzrIXVdTa3H7xjuhZEFge4XeI6HDn4X9d9GlHrhXhu/g/jt02u8ca+GrAD85hgHbVBeFqx7bVLMsbVeI47clvH48MMPc+fF8xEjRmguxY0aNdKcLuFoDkdhI2vWrNFcKBMTEzVHYbhPmjk2njhxwjVy5EhX3bp1teXB1feyyy5zbdy40cMp+PXXX/d6rxPXTl8u0UYHza1bt7quvvpqzaUZDuxwC122bJnpcuESDtdobF/16tVdDzzwgOZMbATT8Brmwbx4j9GNvU6dOpbbAgdQq/U37lO4bt58882aCyjcrM877zxL1/g///xTex3zYX6879ChQ17zwWUV64ZjhPWHy/Vbb73lcopTV3OdDz74INdBPykpyet1uKvCbb1s2bKuChUqaI6dcNA3ng+BuJrbYXQ1x3rAdbZz586uKlWqaG7dWB/M88UXXwT8PcsPcK+GEzuOKb57V155pZcjuO7CanZMxo8frx1fHGccb8yjuuzqrt74DGwrHNkrVaqkOZz/+uuvHvNt2bLF1b9/f801FY7pWCc4eaMrgJnD6pEjR7TvCZzhcS3AOdmpUyfNbfz06dOOtt/JOQ1n70suuURzrMUxg5s6nHFxTsOJ1wndunXTtk3F32sVjhXcxLGucPrFteaff/7xmEd36cW+Mfte4Vpp9lm4TjtZt48//ljb10aHdkJI4YBrML6rLVu29HotLS3N9cgjj2jXVFw3OnbsqF1HcI+A64G/90efffaZq1evXtq1ENd8dFu49tprXatXr/aYb8+ePZq7Oe5hcL3Q51OvrWPGjHHVr19fu9a3aNFC+03Tr18q6LyATgq4v8Jr6m/quHHjtOs/fleM9zdz587Vrne4rmMdsA/wHG7oTq6XVtx0002m+9rsOgqwn81c2QPB7p7AqWu68RgvX75ccztHhxgci4YNG7qGDRvm1WUIPPPMM5qTvpXjOSH+EIV/Clv8RxpII8KIGlwwSfBAn06MvGL00dhHORJAOjDqptWsAUKKMkgJh4kcsmdUV/GiBox3YHCJKAkhhBQ3EGVGdB9Rb9XfJNJB9icyMJGxF4iXDSFGWONNigxIscc4USSKbkIiEdSM42YNaZlFFRgpoZwAKaGEEFIcQZo4SrSK23UQpY7wH4KfDSHBgMKbEEJIyLJ/UBOut/8qiqC+G4aXercGQggpjvzvf//TBlLz4w5f1MDvFjKdAjGAJcQMppoTEiYw1ZwQQgghhJDIhMKbEEIIIYQQQggJIUw1J4QQQgghhBBCQgiFNyGEEEIIIYQQEkJKSAQaIezfv1/Kli2rGfsQQggh4Qy6NcCwCCZ00dHFezycv+GEEEIi9Tc84oQ3RHedOnUKezUIIYQQv9izZ4/Url27WO81/oYTQgiJ1N/wiBPeiHTrG1+uXLnCXh1CCCHEluTkZG3AWP/9Ks7wN5wQQkik/oZHnPDW08shuim8CSGEFBVYHsXfcEIIIZH7G168i8kIIYQQQgghhJAQQ+FNCCGEEEIIIYSEEApvQgghhBBCCCEkhERcjTchhJDQtHlKT0/nrg2A2NhYiYmJ4b4jhBBCijEU3oQQQmyB4N6xY4cmvklglC9fXqpXr04DNUIIIaSYQuFNCCHEEpfLJQcOHNAitmiXER3NCiV/919qaqocPnxYe16jRg2ebYQQQkgxhMKbEEKIJZmZmZpwrFmzppQqVYp7KgBKliyp/Q/xXbVqVaadE0IIIcUQhi4IIYRYkpWVpf0fFxfHvZQP9EGLjIwM7kdCCCGkGBIxwnvChAnSsmVL6dKlS2GvCiGERBxRUVGFvQpFGu4/QgghpPBZsfuETFm+V/u/oImYVPMRI0Zoj+TkZElMTCzs1SGEEEIIIYQQEiaMmbZB3pu7Pff58B4N5Yl+LQrs8yMm4h0SVn8n8kEvkVkvF/aaEEIIKSTq168v48aN4/4nhBBCiigrdp/wEN0Azwsy8h0xEe+QkHJYZP9ykUqNC3tNCCGE+EHPnj2lffv2QRHMS5YskdKlS3P/E0IIIUWU9fuTTafvOJoiHepWKJB1oPC2IyonIcDlNhcihBASOW2+YBxXooTvn8EqVaoUyDoRQgghJPgs23Vcxv+1xfS1BpULbmCdqeZ2RMW4/3dlF8zRIIQQkm9uvfVWmTt3rowfP14zNcNj0qRJ2v/Tp0+Xzp07S3x8vMyfP1+2bdsmV111lVSrVk3KlCmjGXTOnDnTNtUcy/noo49kwIABmlt5kyZN5Oeff+aRI4QQQsKIzKxseXPGZhn83r9y+FSalE3wHGy/p0fDAot2A0a87dBdfCm8CSHEfTl0ueRMRuFkAZWMjXHkDg7BvXnzZmndurW8+OKL2rR169Zp/z/22GPyxhtvSMOGDaV8+fKyd+9e6d+/v7z00kuSkJAgn332mVxxxRWyadMmqVu3ruVnvPDCC/Laa6/J66+/Lm+//bbccMMNsmvXLqlYsWIQt5gQQgghgbD7WKo8OHmFLN99Uns+sEMteeGqVrL18GktvRyR7oIU3YDC21GqOSPehBACILpbPju9UHbG+hf7SKk43z9b6GyBvuOIRlevXl2btnHjRu1/CPFLL700d95KlSpJu3btcp9DgE+dOlWLYN933322UfXrrrtO+/uVV17RxPfixYulb9+++dpGQgghhOQvQDB1xT559qd1cjotU4tyv3R1a7mqfS3tdYjtghbcOhTejoS3q2COBiGEkJCCNHOVlJQULXr966+/yv79+yUzM1POnDkju3fvtl1O27Ztc/+G8VrZsmXl8OHDIVtvQgghhNiTdCZDnv5xrfyyar/2/Jz6FWXskHZSu0IpCQcovJ0I72yaqxFCiJ7ujchzYX12fjG6kz/66KNa3TfSzxs3biwlS5aUa665RtLT022XExsb6/EcKfDZ2cyOIoQQQgqDxTuOy0OTV8q+k2ckJjpKHrqkidzTs7H2d7hA4W3D39uOy4Wa/fxJaVlwx4QQQsIWCEwn6d6FDVLN4VruCxisIW0cRmng9OnTsnPnzgJYQ0IIIYTkl4ysbBk/c4tMnLNVsl0i9SqVknFD2hdaOrkd4X/3VIhkC83VCCGkKAIn8kWLFmkiGm7lVtFoRLmnTJmiGaphUOGZZ55h5JoQQggpAuw8miIjJ6+UVXvcBmqDO9WW565sJWXiw1Pisp2YHWwnRgghRZJHHnlEYmJipGXLllofbqua7TfffFMqVKggXbt21cR3nz59pGPHjgW+voQQQghxbqD27dI90v+t+ZroLpdQQiZc31FeH9wubEU3CN81Cwei3eMSUUJzNUIIKUo0bdpU/v33X49pSCk3i4zPmjXLY9qIESM8nhtTz/GDb+TkSfdoOyGEEEJCR1Jqhvxn6hr5bc0B7fm5DSrKm0PaS83yJcN+t1N42xCVY64WxXZihBBCCCGEEFJo/LvtmIz6dqUcSDorJaKjZFTvpnJ390ZhZaBmB4W3DVF6xNtFV3NCCCGEEEIICSUrdp+QHUdTpEHl0rkGaemZ2TJ2xmZ5f942rcszXhs/tL20rV2+SB0MCm8n7cSYak4IIYQQQgghIWPMtA3y3tztuc+H92gogzvXkQe/WSlr9iVp04Z2qSPPXtGySHRYMVL01rgAYao5IYQQQgghhIQ+0v2eIroBnn/6z05Jy8yW8qViZczANtK3dY0ieygovO2IjtH+ixLzNjSEEEIIIYQQQvLHjqMpptMhui9oXEn+N7i9VE9MsE1JD3covB1FvOlqTgghhBBCCCGhoEHl0qbTb+1aT569vJVEKwZqZinpT/RrEfYHhn28nZirMeJNCCGEEEIIISGhZc1y0q52ose0azrWkuevbO0huq1S0jE93AlL4T1gwACpUKGCXHPNNYW7ImwnRgghhBBCCCEhY+vhUzJw4gJZtddtoNa1USX55s5z5Y1r2ztOSbeaHk6EZar5Aw88ILfffrt89tlnhboeUazxJoQQQgghhJCg43K55MtFu+Wl39bL2YxsqVg6Tl4d1FYubVnN75R0q+nhRFhGvHv16iVly5Yt7NVgjTchhBRT6tevL+PGjSvs1SCEEEIikmOn0+TOz5fK0z+u1UR3tyaV5Y+R3WxFN4CRGmq6Ve7p0bBIGKz5LbznzZsnV1xxhdSsWVOioqLkxx9/9Jpn4sSJ0qBBA0lISJBOnTrJ/PnzpSjCiDchhBBCCCGEBI95m49I3/HzZeaGwxIXEy3PXN5SPrvtHKlaLs+13A4YqU29t6uMvbad9v/jRcBYLaBU85SUFGnXrp3cdtttMmjQIK/XJ0+eLA8++KAmvi+44AJ5//33pV+/frJ+/XqpW7euNg/EeFpamtd7//zzT03Qh525movtxAghhBBCCCEkUM5mZMlrf2yST/7ZoT1vUrWMvHVdB2lRo5zfy0KEuyhEufMlvCGi8bBi7Nixcscdd8iwYcO050jVmz59urz77rsyevRobdqyZcukKBAVpffxZjsxQggpKmDA98UXX5Q9e/ZIdM4AKrjyyis1485nn31WRo0aJQsXLtQGk1u0aKH9Pl1yySWFut6EEEJIpLL50Cl54OsVsvHgKe35LefXkyf7t5CEWLfeKg4EtcY7PT1dE9W9e/f2mI7nCxYskFCAyHlycrLHI1hE5VjXR7OdGCGEuHG5RNJTCueBz3bA4MGD5ejRozJ79uzcaSdOnNAGgW+44QY5ffq09O/fX2bOnCkrVqyQPn36aCVUu3fv5lEmhBBCgmyg9tmCnXLF239rortymTj59NYu8sJVrYuV6A66qzludLKysqRaNc+ieDw/ePCg4+XgJmj58uVaJKJ27doydepU6dKli+m8iFK88MILEtIab6aaE0KIm4xUkVcKqSToP/tF4ny7llasWFH69u0rX331lVx88cXatO+++06bjucxMTFayZTOSy+9pP3O/Pzzz3LfffeFdBMIIYSQ4sKRU2ny2PerZPamI9rzns2qyOvXtJMqZeOlOBISV3OYrhlHOozT7EBU4siRI5Kamip79+61FN3gySeflKSkpNwHUguDRZTex5up5oQQUqRAZPuHH37I9RP58ssvZejQoZroxqDuY489Ji1btpTy5ctLmTJlZOPGjYx4E0IIIUFi9sbD0m/8PE10x5WIlheubKVFuour6A56xLty5craTY0xun348GGvKHiwiI+P1x4hgX28CSHEk9hS7shzYX22Q5A6np2dLb/99ps2eIvuGvAgAY8++qg2wPvGG29I48aNpWTJknLNNddo5VKEEEIIyZ+B2phpG2XSgp3a8+bVy8r4oR2kWfXCbxUdUcI7Li5OcyyfMWOGDBgwIHc6nl911VVS1NBNeaKZak4IIW6QveQg3buwgZgeOHCgFuneunWrNG3aVPt9AhDht956a+7vFGq+d+503yAQQgghJDA2HEiWkd+skM2HTmvPb7+ggTzWt1mxq+UOmvDGDQpuYnR27NghK1eu1Grn0C4MTrE33XSTdO7cWc4//3z54IMPtPS94cOHSyiZMGGC9kCNebCgqzkhhBTtdHNEvtetWyc33nhj7nREuadMmaK9hjKoZ555RouOE0IIIcR/srNdWoR7zB8bJT0zWyqXiZc3BreVns2qcnfmR3gvXbpUevXqlfscQhvccsstMmnSJBkyZIgcO3ZMa+Vy4MABad26tfz+++9Sr149CSUjRozQHnA1T0xMDG4fb7qaE0JIkeOiiy7SBoU3bdok119/fe70N998U26//Xbp2rWrViL1+OOPB7UjBiGEEFJcOJx8Vh75frXM2+w2ULukRVV5dVBbqVSm+NZyB0149+zZUzNLs+Pee+/VHkWd3FRzmqsRQkiRA54j+/d716PXr19fZs2a5TENA7cqTD0nhBBC7Jm5/pA89sNqOZ6SLvElouXpy1vKjefW9ctUuzgR1BrviCO3nZiz3rGEEEIIIYQQEsmcSc+Sl39fL/+3cLf2vEWNcvL2de2lcVUaqNlB4e2gj3e0BK9unBBCCCGEEEKKIuv2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M2fcemy2a9fOqlmzpunNXKcaNWpYnTp1ysLZA5mlvcxXrlzZ/H85f/68w7ULmafHhh4jEyZMsIoUKWI/fvRYQsa4r8y+czfcd18P9yLxzsDs2bOtyMhIKyAgwKpbt26e715fT27OJh1WJOUQPNHR0WYYnsDAQOupp54yN6wp/fXXX9brr79uhYaGWgULFjQ3oGfPnrXyitSJNzFL3+rVq63q1aubfUmH85s3b57D98QuLU0Kdf8qX768FRQUZEVFRVljx461EhMTiVsqGzdudHpO04cX7ty/fv/9d6tnz55WcHCwmfT99evXXTxj4EHo9Sm9axcyT48NZ7HUYwn3x31l9py74b77eriXn/7HEyXpAAAAAACANt4AAAAAAHgUPYwAAAAAAOBBJN4AAAAAAHgQiTcAAAAAAB5E4g0AAAAAgAeReAMAAAAAQOINAAAAAEDuRIk3AAAAAAAeROINAAAA5BF37tyRypUryw8//CA5VYMGDWTFihXe3gzArUi8AQAAAA8bP3681K5d2+txnjdvnkRGRkqTJk0kp3r77bdl9OjRkpyc7O1NAdyGxBsAAADIIe7evevR9c+cOVMGDBgg2VGynlUdO3aU+Ph4+fbbb926TYA3kXgDAAAA9/Hpp59KiRIlJDEx0WH+888/L717987wtwsWLJAJEybIgQMHxM/Pz0w6T+n7uXPnynPPPSeFCxeW9957z3xXrFgxh3V89dVXZtmUVq9eLfXq1ZOgoCCJiooy/0ZSUlK627F37145efKkSWxt4uLizHq1aneLFi2kUKFCUqtWLdmxY4fDb7/88kt57LHHJDAwUCpUqCDTpk1z+F7n6bb37dtXQkJCZODAgfa/Y82aNfLoo4+adb/wwgty69YtWbhwoflN8eLFZciQIXLv3j37uvLlyycdOnSQpUuXZhhXIDch8QYAAADuo2vXriY5XLVqlX3etWvXTFL5yiuvZPjbl156Sd58802TuF66dMlMOs8mOjraJN4HDx6Ufv36ufT/QkuDe/XqJUOHDpUjR45ITEyMSXQnTpyY7m+2bNkijzzyiBQtWjTNd2PHjpW33npL9u/fb5bp3r27PYnfs2ePvPjii9KtWzezjVptXquD2x4e2EyZMkWqV69ultfv1e3bt+Wjjz6SZcuWydq1a2XTpk3SpUsX+frrr8302WefmervX3zxhcO6GjZsKFu3bnUpFkBukN/bGwAAAADkdAULFpQePXpIbGysScLV4sWLJSIiQpo3b37f3xYpUkTy588v4eHhab7X9bqacNtogq3toPv06WM+a4n3f/7zHxk5cqRJ5J3R0u0yZco4/U6TbltJuJac60MCLR2vUqWKTJ8+XVq1amVPpjUx12RfE20t4bZp2bKlWY/Ntm3bTNX5OXPmSKVKlcw8LfHWZPvy5csmJtWqVTMl7Rs3bnR4GFG2bFk5e/asaeft709ZIXI/9mIAAADABVp9+rvvvpMLFy6Yz5qEa+KZugp4ZtWvXz/Tv9FS5Xfffdckr7ZJt09L07WU2Zm//vrLVEt3pmbNmvb3pUuXNq9Xrlwxr0ePHk3TGZt+PnHihEMVcWd/h1YvtyXdqlSpUqaKuW5vynm2fyvlwwpNulNX7QdyK0q8AQAAABfUqVPHtH/W9t5PP/20qXat7awflLbtTklLeC3LyrDTNU1KtWRaq22nll5yHRYWZrbZmQIFCtjf2x4k2HoV121J/XAh9fY5+ztSr9e2bmfzUvdg/scff5ikXRNwwBeQeAMAAAAu0h7BZ8yYYUq9W7duLeXKlXPpdwEBAQ6lwxl56KGHJCEhwXRCZktmte11SnXr1pXjx4+bMbkz8+BAq307S6QzotXBtdp4Stu3bzdVzrUjNE84dOiQ+RsBX0FVcwAAAMBFPXv2NEn3/PnzM9UuW6tXnz592iTQ2ilbRlWoGzVqZEp7x4wZY9pZL1myJE1HZu+8844pedeOzg4fPmyqgy9fvlzGjRuX7nq1LbUm87p8ZmjHcOvXrzdtyH/55RfTI/msWbMc2nO7m3as1rZtW4+tH8huJN4AAACAi7RHcB1CTNsod+7c2eW46W/atWtnkl8t0c5oqKzQ0FBZtGiR6fW7Ro0aZllNsFPSqu7ao/q6deukQYMG0rhxY9MJWmRkZLrr1eHQtGq6dgqXGVry/Pnnn5ueybXXck36tX15yo7V3EkfbGiJ+v16iwdyEz/LWQMNAAAAAE61adNGqlataobJym20jbdWkdeS9ODgYMmJRowYIfHx8WaYMcBXUOINAAAAuEA7/NJS3w0bNsjgwYNzZcy0BH3y5MlmaLGcqmTJkqZaO+BLKPEGAAAAXGynff36dTOeder2zTru9ZkzZ5z+LiYmxrQNB5B3kXgDAAAAD0iT7tRDfqUcpzqnVusGkD1IvAEAAAAA8CDaeAMAAAAA4EEk3gAAAAAAeBCJNwAAAAAAHkTiDQAAAACAB5F4AwAAAADgQSTeAAAAAAB4EIk3AAAAAAAeROINAAAAAIB4zv8Dv2yeffeSqmQAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[DONE] best val MSE (norm) = 5.731e-03 @ epoch 640\n",
      "\n",
      "\n",
      "Interrupt request received\n",
      "iter=0  t=0.00s  best_y=896051  x=[50, -20, 1, -10, 7, 30, 0, 2, -30, -11]\n",
      "iter=1  t=0.04s  best_y=888580  x=[50, -20, 1, -8, 7, 30, 0, 2, -30, -13]\n",
      "iter=2  t=0.07s  best_y=881109  x=[50, -20, 1, -6, 7, 30, 0, 2, -30, -15]\n",
      "iter=3  t=0.11s  best_y=873638  x=[50, -20, 1, -4, 7, 30, 0, 2, -30, -17]\n",
      "iter=4  t=0.14s  best_y=867501  x=[50, -20, 1, -2, 7, 30, 0, 2, -30, -19]\n",
      "iter=5  t=0.18s  best_y=862218  x=[50, -20, 1, 0, 7, 30, 0, 2, -30, -21]\n",
      "iter=6  t=0.21s  best_y=856934  x=[50, -20, 1, 2, 7, 30, 0, 2, -30, -23]\n",
      "iter=7  t=0.25s  best_y=851650  x=[50, -20, 1, 4, 7, 30, 0, 2, -30, -25]\n",
      "iter=8  t=0.28s  best_y=846366  x=[50, -20, 1, 6, 7, 30, 0, 2, -30, -27]\n",
      "iter=9  t=0.32s  best_y=841083  x=[50, -20, 1, 8, 7, 30, 0, 2, -30, -29]\n",
      "iter=10  t=0.35s  best_y=835800  x=[50, -20, 1, 10, 7, 30, 0, 2, -30, -31]\n",
      "iter=11  t=0.39s  best_y=830516  x=[50, -20, 1, 12, 7, 30, 0, 2, -30, -33]\n",
      "iter=12  t=0.42s  best_y=825232  x=[50, -20, 1, 14, 7, 30, 0, 2, -30, -35]\n",
      "iter=13  t=0.46s  best_y=819948  x=[50, -20, 1, 16, 7, 30, 0, 2, -30, -37]\n",
      "iter=14  t=0.50s  best_y=814665  x=[50, -20, 1, 18, 7, 30, 0, 2, -30, -39]\n",
      "iter=15  t=0.53s  best_y=809382  x=[50, -20, 1, 20, 7, 30, 0, 2, -30, -41]\n",
      "iter=16  t=0.57s  best_y=804098  x=[50, -20, 1, 22, 7, 30, 0, 2, -30, -43]\n",
      "iter=17  t=0.60s  best_y=798814  x=[50, -20, 1, 24, 7, 30, 0, 2, -30, -45]\n",
      "iter=18  t=0.64s  best_y=793530  x=[50, -20, 1, 26, 7, 30, 0, 2, -30, -47]\n",
      "iter=19  t=0.67s  best_y=788247  x=[50, -20, 1, 28, 7, 30, 0, 2, -30, -49]\n",
      "iter=20  t=0.71s  best_y=782963  x=[50, -20, 1, 30, 7, 30, 0, 2, -30, -51]\n",
      "iter=21  t=0.74s  best_y=777680  x=[50, -20, 1, 32, 7, 30, 0, 2, -30, -53]\n",
      "iter=22  t=0.78s  best_y=772396  x=[50, -20, 1, 34, 7, 30, 0, 2, -30, -55]\n",
      "iter=23  t=0.81s  best_y=767112  x=[50, -20, 1, 36, 7, 30, 0, 2, -30, -57]\n",
      "iter=24  t=0.85s  best_y=761829  x=[50, -20, 1, 38, 7, 30, 0, 2, -30, -59]\n",
      "iter=25  t=0.89s  best_y=756546  x=[50, -20, 1, 40, 7, 30, 0, 2, -30, -61]\n",
      "iter=26  t=0.92s  best_y=751262  x=[50, -20, 1, 42, 7, 30, 0, 2, -30, -63]\n",
      "iter=27  t=0.96s  best_y=745978  x=[50, -20, 1, 44, 7, 30, 0, 2, -30, -65]\n",
      "iter=28  t=0.99s  best_y=740694  x=[50, -20, 1, 46, 7, 30, 0, 2, -30, -67]\n",
      "iter=29  t=1.03s  best_y=735411  x=[50, -20, 1, 48, 7, 30, 0, 2, -30, -69]\n",
      "iter=30  t=1.06s  best_y=730127  x=[50, -20, 1, 50, 7, 30, 0, 2, -30, -71]\n",
      "iter=31  t=1.10s  best_y=724844  x=[50, -20, 1, 52, 7, 30, 0, 2, -30, -73]\n",
      "iter=32  t=1.13s  best_y=719560  x=[50, -20, 1, 54, 7, 30, 0, 2, -30, -75]\n",
      "iter=33  t=1.17s  best_y=714276  x=[50, -20, 1, 56, 7, 30, 0, 2, -30, -77]\n",
      "iter=34  t=1.20s  best_y=708993  x=[50, -20, 1, 58, 7, 30, 0, 2, -30, -79]\n",
      "iter=35  t=1.24s  best_y=703709  x=[50, -20, 1, 60, 7, 30, 0, 2, -30, -81]\n",
      "iter=36  t=1.27s  best_y=698426  x=[50, -20, 1, 62, 7, 30, 0, 2, -30, -83]\n",
      "iter=37  t=1.31s  best_y=693142  x=[50, -20, 1, 64, 7, 30, 0, 2, -30, -85]\n",
      "iter=38  t=1.35s  best_y=687858  x=[50, -20, 1, 66, 7, 30, 0, 2, -30, -87]\n",
      "iter=39  t=1.38s  best_y=682575  x=[50, -20, 1, 68, 7, 30, 0, 2, -30, -89]\n",
      "iter=40  t=1.42s  best_y=677291  x=[50, -20, 1, 70, 7, 30, 0, 2, -30, -91]\n",
      "iter=41  t=1.45s  best_y=672008  x=[50, -20, 1, 72, 7, 30, 0, 2, -30, -93]\n",
      "iter=42  t=1.49s  best_y=666724  x=[50, -20, 1, 74, 7, 30, 0, 2, -30, -95]\n",
      "iter=43  t=1.52s  best_y=661440  x=[50, -20, 1, 76, 7, 30, 0, 2, -30, -97]\n",
      "iter=44  t=1.56s  best_y=656157  x=[50, -20, 1, 78, 7, 30, 0, 2, -30, -99]\n",
      "iter=45  t=1.59s  best_y=651620  x=[50, -20, 1, 80, 7, 30, 0, 2, -31, -100]\n",
      "iter=46  t=1.63s  best_y=647829  x=[50, -20, 1, 82, 7, 30, 0, 2, -33, -100]\n",
      "iter=47  t=1.66s  best_y=644038  x=[50, -20, 1, 84, 7, 30, 0, 2, -35, -100]\n",
      "iter=48  t=1.69s  best_y=640247  x=[50, -20, 1, 86, 7, 30, 0, 2, -37, -100]\n",
      "iter=49  t=1.72s  best_y=636456  x=[50, -20, 1, 88, 7, 30, 0, 2, -39, -100]\n",
      "iter=50  t=1.75s  best_y=632665  x=[50, -20, 1, 90, 7, 30, 0, 2, -41, -100]\n",
      "iter=51  t=1.79s  best_y=628874  x=[50, -20, 1, 92, 7, 30, 0, 2, -43, -100]\n",
      "iter=52  t=1.82s  best_y=625083  x=[50, -20, 1, 94, 7, 30, 0, 2, -45, -100]\n",
      "iter=53  t=1.85s  best_y=621292  x=[50, -20, 1, 96, 7, 30, 0, 2, -47, -100]\n",
      "iter=54  t=1.89s  best_y=617502  x=[50, -20, 1, 98, 7, 30, 0, 2, -49, -100]\n",
      "iter=55  t=1.92s  best_y=613711  x=[50, -20, 1, 100, 7, 30, 0, 2, -51, -100]\n",
      "iter=56  t=1.95s  best_y=610981  x=[50, -20, 1, 100, 7, 32, 0, 2, -53, -100]\n",
      "iter=57  t=1.98s  best_y=608251  x=[50, -20, 1, 100, 7, 34, 0, 2, -55, -100]\n",
      "iter=58  t=2.01s  best_y=605522  x=[50, -20, 1, 100, 7, 36, 0, 2, -57, -100]\n",
      "iter=59  t=2.05s  best_y=602792  x=[50, -20, 1, 100, 7, 38, 0, 2, -59, -100]\n",
      "iter=60  t=2.08s  best_y=600062  x=[50, -20, 1, 100, 7, 40, 0, 2, -61, -100]\n",
      "iter=61  t=2.11s  best_y=597333  x=[50, -20, 1, 100, 7, 42, 0, 2, -63, -100]\n",
      "iter=62  t=2.14s  best_y=594603  x=[50, -20, 1, 100, 7, 44, 0, 2, -65, -100]\n",
      "iter=63  t=2.17s  best_y=591873  x=[50, -20, 1, 100, 7, 46, 0, 2, -67, -100]\n",
      "iter=64  t=2.20s  best_y=589143  x=[50, -20, 1, 100, 7, 48, 0, 2, -69, -100]\n",
      "iter=65  t=2.24s  best_y=586414  x=[50, -20, 1, 100, 7, 50, 0, 2, -71, -100]\n",
      "iter=66  t=2.27s  best_y=583684  x=[50, -20, 1, 100, 7, 52, 0, 2, -73, -100]\n",
      "iter=67  t=2.30s  best_y=580954  x=[50, -20, 1, 100, 7, 54, 0, 2, -75, -100]\n",
      "iter=68  t=2.33s  best_y=578225  x=[50, -20, 1, 100, 7, 56, 0, 2, -77, -100]\n",
      "iter=69  t=2.36s  best_y=575495  x=[50, -20, 1, 100, 7, 58, 0, 2, -79, -100]\n",
      "iter=70  t=2.40s  best_y=572766  x=[50, -20, 1, 100, 7, 60, 0, 2, -81, -100]\n",
      "iter=71  t=2.43s  best_y=570036  x=[50, -20, 1, 100, 7, 62, 0, 2, -83, -100]\n",
      "iter=72  t=2.46s  best_y=567306  x=[50, -20, 1, 100, 7, 64, 0, 2, -85, -100]\n",
      "iter=73  t=2.49s  best_y=564576  x=[50, -20, 1, 100, 7, 66, 0, 2, -87, -100]\n",
      "iter=74  t=2.52s  best_y=561846  x=[50, -20, 1, 100, 7, 68, 0, 2, -89, -100]\n",
      "iter=75  t=2.56s  best_y=559116  x=[50, -20, 1, 100, 7, 70, 0, 2, -91, -100]\n",
      "iter=76  t=2.59s  best_y=556388  x=[50, -20, 1, 100, 7, 72, 0, 2, -93, -100]\n",
      "iter=77  t=2.62s  best_y=553658  x=[50, -20, 1, 100, 7, 74, 0, 2, -95, -100]\n",
      "iter=78  t=2.65s  best_y=550928  x=[50, -20, 1, 100, 7, 76, 0, 2, -97, -100]\n",
      "iter=79  t=2.68s  best_y=548198  x=[50, -20, 1, 100, 7, 78, 0, 2, -99, -100]\n",
      "iter=80  t=2.71s  best_y=545705  x=[49, -20, 1, 100, 7, 80, 0, 2, -100, -100]\n",
      "iter=81  t=2.74s  best_y=543449  x=[47, -20, 1, 100, 7, 82, 0, 2, -100, -100]\n",
      "iter=82  t=2.77s  best_y=541192  x=[45, -20, 1, 100, 7, 84, 0, 2, -100, -100]\n",
      "iter=83  t=2.80s  best_y=538936  x=[43, -20, 1, 100, 7, 86, 0, 2, -100, -100]\n",
      "iter=84  t=2.83s  best_y=536680  x=[41, -20, 1, 100, 7, 88, 0, 2, -100, -100]\n",
      "iter=85  t=2.86s  best_y=534423  x=[39, -20, 1, 100, 7, 90, 0, 2, -100, -100]\n",
      "iter=86  t=2.89s  best_y=532166  x=[37, -20, 1, 100, 7, 92, 0, 2, -100, -100]\n",
      "iter=87  t=2.92s  best_y=529910  x=[35, -20, 1, 100, 7, 94, 0, 2, -100, -100]\n",
      "iter=88  t=2.95s  best_y=527654  x=[33, -20, 1, 100, 7, 96, 0, 2, -100, -100]\n",
      "iter=89  t=2.98s  best_y=525398  x=[31, -20, 1, 100, 7, 98, 0, 2, -100, -100]\n",
      "iter=90  t=3.01s  best_y=523141  x=[29, -20, 1, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=91  t=3.04s  best_y=520963  x=[27, -20, 3, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=92  t=3.07s  best_y=518786  x=[25, -20, 5, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=93  t=3.10s  best_y=516608  x=[23, -20, 7, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=94  t=3.12s  best_y=514429  x=[21, -20, 9, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=95  t=3.15s  best_y=512252  x=[19, -20, 11, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=96  t=3.18s  best_y=510074  x=[17, -20, 13, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=97  t=3.21s  best_y=507896  x=[15, -20, 15, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=98  t=3.24s  best_y=505598  x=[13, -20, 17, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=99  t=3.27s  best_y=503212  x=[11, -20, 19, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=100  t=3.30s  best_y=500826  x=[9, -20, 21, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=101  t=3.33s  best_y=498439  x=[7, -20, 23, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=102  t=3.36s  best_y=496053  x=[5, -20, 25, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=103  t=3.38s  best_y=493666  x=[3, -20, 27, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=104  t=3.41s  best_y=491280  x=[1, -20, 29, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=105  t=3.44s  best_y=488894  x=[-1, -20, 31, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=106  t=3.47s  best_y=486508  x=[-3, -20, 33, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=107  t=3.50s  best_y=484122  x=[-5, -20, 35, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=108  t=3.53s  best_y=481736  x=[-7, -20, 37, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=109  t=3.56s  best_y=479349  x=[-9, -20, 39, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=110  t=3.59s  best_y=476963  x=[-11, -20, 41, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=111  t=3.62s  best_y=474577  x=[-13, -20, 43, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=112  t=3.65s  best_y=472190  x=[-15, -20, 45, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=113  t=3.68s  best_y=469804  x=[-17, -20, 47, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=114  t=3.71s  best_y=467418  x=[-19, -20, 49, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=115  t=3.74s  best_y=465032  x=[-21, -20, 51, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=116  t=3.77s  best_y=462646  x=[-23, -20, 53, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=117  t=3.80s  best_y=460259  x=[-25, -20, 55, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=118  t=3.83s  best_y=457873  x=[-27, -20, 57, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=119  t=3.85s  best_y=455486  x=[-29, -20, 59, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=120  t=3.88s  best_y=453100  x=[-31, -20, 61, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=121  t=3.91s  best_y=450714  x=[-33, -20, 63, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=122  t=3.94s  best_y=448328  x=[-35, -20, 65, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=123  t=3.97s  best_y=445942  x=[-37, -20, 67, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=124  t=4.00s  best_y=443556  x=[-39, -20, 69, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=125  t=4.03s  best_y=441169  x=[-41, -20, 71, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=126  t=4.06s  best_y=438782  x=[-43, -20, 73, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=127  t=4.09s  best_y=436396  x=[-45, -20, 75, 100, 7, 100, 0, 2, -100, -100]\n",
      "iter=128  t=4.12s  best_y=434073  x=[-46, -20, 77, 100, 6, 100, 0, 2, -100, -100]\n",
      "iter=129  t=4.15s  best_y=431748  x=[-47, -20, 79, 100, 5, 100, 0, 2, -100, -100]\n",
      "iter=130  t=4.18s  best_y=429374  x=[-49, -20, 81, 100, 5, 100, 0, 2, -100, -100]\n",
      "iter=131  t=4.21s  best_y=427038  x=[-50, -20, 83, 100, 4, 100, 0, 2, -100, -100]\n",
      "iter=132  t=4.23s  best_y=424715  x=[-51, -20, 85, 100, 3, 100, 0, 2, -100, -100]\n",
      "iter=133  t=4.26s  best_y=422391  x=[-52, -20, 87, 100, 2, 100, 0, 2, -100, -100]\n",
      "iter=134  t=4.29s  best_y=420004  x=[-54, -20, 89, 100, 2, 100, 0, 2, -100, -100]\n",
      "iter=135  t=4.32s  best_y=417681  x=[-55, -20, 91, 100, 1, 100, 0, 2, -100, -100]\n",
      "iter=136  t=4.35s  best_y=415356  x=[-56, -20, 93, 100, 0, 100, 0, 2, -100, -100]\n",
      "iter=137  t=4.38s  best_y=412970  x=[-58, -20, 95, 100, 0, 100, 0, 2, -100, -100]\n",
      "iter=138  t=4.41s  best_y=410647  x=[-59, -20, 97, 100, -1, 100, 0, 2, -100, -100]\n",
      "iter=139  t=4.44s  best_y=408323  x=[-60, -20, 99, 100, -2, 100, 0, 2, -100, -100]\n",
      "iter=140  t=4.47s  best_y=406198  x=[-61, -20, 100, 100, -3, 100, 1, 2, -100, -100]\n",
      "iter=141  t=4.49s  best_y=404211  x=[-63, -20, 100, 100, -3, 100, 3, 2, -100, -100]\n",
      "iter=142  t=4.52s  best_y=402286  x=[-64, -20, 100, 100, -4, 100, 5, 2, -100, -100]\n",
      "iter=143  t=4.55s  best_y=400360  x=[-65, -20, 100, 100, -5, 100, 7, 2, -100, -100]\n",
      "iter=144  t=4.58s  best_y=398436  x=[-66, -20, 100, 100, -6, 100, 9, 2, -100, -100]\n",
      "iter=145  t=4.61s  best_y=396510  x=[-67, -20, 100, 100, -7, 100, 11, 2, -100, -100]\n",
      "iter=146  t=4.64s  best_y=394585  x=[-68, -20, 100, 100, -8, 100, 13, 2, -100, -100]\n",
      "iter=147  t=4.66s  best_y=392629  x=[-70, -20, 100, 100, -8, 100, 15, 2, -100, -100]\n",
      "iter=148  t=4.69s  best_y=390672  x=[-71, -20, 100, 100, -9, 100, 17, 2, -100, -100]\n",
      "iter=149  t=4.72s  best_y=388748  x=[-72, -20, 100, 100, -10, 100, 19, 2, -100, -100]\n",
      "iter=150  t=4.75s  best_y=386822  x=[-73, -20, 100, 100, -11, 100, 21, 2, -100, -100]\n",
      "iter=151  t=4.78s  best_y=384897  x=[-74, -20, 100, 100, -12, 100, 23, 2, -100, -100]\n",
      "iter=152  t=4.80s  best_y=382972  x=[-75, -20, 100, 100, -13, 100, 25, 2, -100, -100]\n",
      "iter=153  t=4.83s  best_y=381046  x=[-76, -20, 100, 100, -14, 100, 27, 2, -100, -100]\n",
      "iter=154  t=4.86s  best_y=379059  x=[-78, -20, 100, 100, -14, 100, 29, 2, -100, -100]\n",
      "iter=155  t=4.89s  best_y=377134  x=[-79, -20, 100, 100, -15, 100, 31, 2, -100, -100]\n",
      "iter=156  t=4.92s  best_y=374574  x=[-79, -20, 100, 100, -17, 100, 33, 2, -100, -100]\n",
      "iter=157  t=4.95s  best_y=371749  x=[-79, -20, 100, 100, -19, 100, 35, 2, -100, -100]\n",
      "iter=158  t=4.98s  best_y=368922  x=[-79, -20, 100, 100, -21, 100, 37, 2, -100, -100]\n",
      "iter=159  t=5.01s  best_y=366096  x=[-79, -20, 100, 100, -23, 100, 39, 2, -100, -100]\n",
      "iter=160  t=5.03s  best_y=363270  x=[-79, -20, 100, 100, -25, 100, 41, 2, -100, -100]\n",
      "iter=161  t=5.06s  best_y=360444  x=[-79, -20, 100, 100, -27, 100, 43, 2, -100, -100]\n",
      "iter=162  t=5.09s  best_y=357618  x=[-79, -20, 100, 100, -29, 100, 45, 2, -100, -100]\n",
      "iter=163  t=5.12s  best_y=354792  x=[-79, -20, 100, 100, -31, 100, 47, 2, -100, -100]\n",
      "iter=164  t=5.15s  best_y=351966  x=[-79, -20, 100, 100, -33, 100, 49, 2, -100, -100]\n",
      "iter=165  t=5.18s  best_y=349139  x=[-79, -20, 100, 100, -35, 100, 51, 2, -100, -100]\n",
      "iter=166  t=5.20s  best_y=346313  x=[-79, -20, 100, 100, -37, 100, 53, 2, -100, -100]\n",
      "iter=167  t=5.23s  best_y=343487  x=[-79, -20, 100, 100, -39, 100, 55, 2, -100, -100]\n",
      "iter=168  t=5.26s  best_y=340660  x=[-79, -20, 100, 100, -41, 100, 57, 2, -100, -100]\n",
      "iter=169  t=5.29s  best_y=337834  x=[-79, -20, 100, 100, -43, 100, 59, 2, -100, -100]\n",
      "iter=170  t=5.32s  best_y=335008  x=[-79, -20, 100, 100, -45, 100, 61, 2, -100, -100]\n",
      "iter=171  t=5.34s  best_y=332182  x=[-79, -20, 100, 100, -47, 100, 63, 2, -100, -100]\n",
      "iter=172  t=5.37s  best_y=329356  x=[-79, -20, 100, 100, -49, 100, 65, 2, -100, -100]\n",
      "iter=173  t=5.40s  best_y=326530  x=[-79, -20, 100, 100, -51, 100, 67, 2, -100, -100]\n",
      "iter=174  t=5.43s  best_y=323704  x=[-79, -20, 100, 100, -53, 100, 69, 2, -100, -100]\n",
      "iter=175  t=5.46s  best_y=320877  x=[-79, -20, 100, 100, -55, 100, 71, 2, -100, -100]\n",
      "iter=176  t=5.49s  best_y=318051  x=[-79, -20, 100, 100, -57, 100, 73, 2, -100, -100]\n",
      "iter=177  t=5.51s  best_y=315224  x=[-79, -20, 100, 100, -59, 100, 75, 2, -100, -100]\n",
      "iter=178  t=5.54s  best_y=312398  x=[-79, -20, 100, 100, -61, 100, 77, 2, -100, -100]\n",
      "iter=179  t=5.57s  best_y=309572  x=[-79, -20, 100, 100, -63, 100, 79, 2, -100, -100]\n",
      "iter=180  t=5.60s  best_y=306746  x=[-79, -20, 100, 100, -65, 100, 81, 2, -100, -100]\n",
      "iter=181  t=5.63s  best_y=303920  x=[-79, -20, 100, 100, -67, 100, 83, 2, -100, -100]\n",
      "iter=182  t=5.65s  best_y=301094  x=[-79, -20, 100, 100, -69, 100, 85, 2, -100, -100]\n",
      "iter=183  t=5.68s  best_y=298268  x=[-79, -20, 100, 100, -71, 100, 87, 2, -100, -100]\n",
      "iter=184  t=5.71s  best_y=295441  x=[-79, -20, 100, 100, -73, 100, 89, 2, -100, -100]\n",
      "iter=185  t=5.74s  best_y=292615  x=[-79, -20, 100, 100, -75, 100, 91, 2, -100, -100]\n",
      "iter=186  t=5.77s  best_y=289788  x=[-79, -20, 100, 100, -77, 100, 93, 2, -100, -100]\n",
      "iter=187  t=5.80s  best_y=286963  x=[-79, -20, 100, 100, -79, 100, 95, 2, -100, -100]\n",
      "iter=188  t=5.82s  best_y=284136  x=[-79, -20, 100, 100, -81, 100, 97, 2, -100, -100]\n",
      "iter=189  t=5.85s  best_y=281306  x=[-79, -20, 100, 100, -83, 100, 99, 2, -100, -100]\n",
      "iter=190  t=5.88s  best_y=278490  x=[-79, -20, 100, 100, -85, 100, 100, 3, -100, -100]\n",
      "iter=191  t=5.91s  best_y=276030  x=[-79, -20, 100, 100, -87, 100, 100, 5, -100, -100]\n",
      "iter=192  t=5.94s  best_y=273568  x=[-79, -20, 100, 100, -89, 100, 100, 7, -100, -100]\n",
      "iter=193  t=5.97s  best_y=271108  x=[-79, -20, 100, 100, -91, 100, 100, 9, -100, -100]\n",
      "iter=194  t=6.00s  best_y=268646  x=[-79, -20, 100, 100, -93, 100, 100, 11, -100, -100]\n",
      "iter=195  t=6.03s  best_y=266186  x=[-79, -20, 100, 100, -95, 100, 100, 13, -100, -100]\n",
      "iter=196  t=6.05s  best_y=263724  x=[-79, -20, 100, 100, -97, 100, 100, 15, -100, -100]\n",
      "iter=197  t=6.08s  best_y=261264  x=[-79, -20, 100, 100, -99, 100, 100, 17, -100, -100]\n",
      "iter=198  t=6.11s  best_y=258818  x=[-80, -20, 100, 100, -100, 100, 100, 19, -100, -100]\n",
      "iter=199  t=6.14s  best_y=256268  x=[-82, -20, 100, 100, -100, 100, 100, 21, -100, -100]\n",
      "iter=200  t=6.16s  best_y=253565  x=[-84, -20, 100, 100, -100, 100, 100, 23, -100, -100]\n",
      "iter=201  t=6.19s  best_y=250862  x=[-86, -20, 100, 100, -100, 100, 100, 25, -100, -100]\n",
      "iter=202  t=6.22s  best_y=248160  x=[-88, -20, 100, 100, -100, 100, 100, 27, -100, -100]\n",
      "iter=203  t=6.24s  best_y=245456  x=[-90, -20, 100, 100, -100, 100, 100, 29, -100, -100]\n",
      "iter=204  t=6.27s  best_y=242754  x=[-92, -20, 100, 100, -100, 100, 100, 31, -100, -100]\n",
      "iter=205  t=6.30s  best_y=240051  x=[-94, -20, 100, 100, -100, 100, 100, 33, -100, -100]\n",
      "iter=206  t=6.32s  best_y=237348  x=[-96, -20, 100, 100, -100, 100, 100, 35, -100, -100]\n",
      "iter=207  t=6.35s  best_y=234646  x=[-98, -20, 100, 100, -100, 100, 100, 37, -100, -100]\n",
      "iter=208  t=6.38s  best_y=231942  x=[-100, -20, 100, 100, -100, 100, 100, 39, -100, -100]\n",
      "iter=209  t=6.40s  best_y=230776  x=[-100, -22, 100, 100, -100, 100, 100, 41, -100, -100]\n",
      "iter=210  t=6.43s  best_y=229504  x=[-100, -24, 100, 100, -100, 100, 100, 43, -100, -100]\n",
      "iter=211  t=6.45s  best_y=227576  x=[-100, -26, 100, 100, -100, 100, 100, 45, -100, -100]\n",
      "iter=212  t=6.48s  best_y=225648  x=[-100, -28, 100, 100, -100, 100, 100, 47, -100, -100]\n",
      "iter=213  t=6.50s  best_y=223721  x=[-100, -30, 100, 100, -100, 100, 100, 49, -100, -100]\n",
      "iter=214  t=6.53s  best_y=221794  x=[-100, -32, 100, 100, -100, 100, 100, 51, -100, -100]\n",
      "iter=215  t=6.55s  best_y=219866  x=[-100, -34, 100, 100, -100, 100, 100, 53, -100, -100]\n",
      "iter=216  t=6.58s  best_y=217938  x=[-100, -36, 100, 100, -100, 100, 100, 55, -100, -100]\n",
      "iter=217  t=6.60s  best_y=216010  x=[-100, -38, 100, 100, -100, 100, 100, 57, -100, -100]\n",
      "iter=218  t=6.63s  best_y=214082  x=[-100, -40, 100, 100, -100, 100, 100, 59, -100, -100]\n",
      "iter=219  t=6.65s  best_y=212154  x=[-100, -42, 100, 100, -100, 100, 100, 61, -100, -100]\n",
      "iter=220  t=6.68s  best_y=210227  x=[-100, -44, 100, 100, -100, 100, 100, 63, -100, -100]\n",
      "iter=221  t=6.70s  best_y=208299  x=[-100, -46, 100, 100, -100, 100, 100, 65, -100, -100]\n",
      "iter=222  t=6.73s  best_y=206364  x=[-100, -48, 100, 100, -100, 100, 100, 67, -100, -100]\n",
      "iter=223  t=6.76s  best_y=204396  x=[-100, -50, 100, 100, -100, 100, 100, 69, -100, -100]\n",
      "iter=224  t=6.78s  best_y=202428  x=[-100, -52, 100, 100, -100, 100, 100, 71, -100, -100]\n",
      "iter=225  t=6.80s  best_y=200460  x=[-100, -54, 100, 100, -100, 100, 100, 73, -100, -100]\n",
      "iter=226  t=6.83s  best_y=198493  x=[-100, -56, 100, 100, -100, 100, 100, 75, -100, -100]\n",
      "iter=227  t=6.85s  best_y=196526  x=[-100, -58, 100, 100, -100, 100, 100, 77, -100, -100]\n",
      "iter=228  t=6.88s  best_y=194558  x=[-100, -60, 100, 100, -100, 100, 100, 79, -100, -100]\n",
      "iter=229  t=6.90s  best_y=192590  x=[-100, -62, 100, 100, -100, 100, 100, 81, -100, -100]\n",
      "iter=230  t=6.93s  best_y=190622  x=[-100, -64, 100, 100, -100, 100, 100, 83, -100, -100]\n",
      "iter=231  t=6.95s  best_y=188655  x=[-100, -66, 100, 100, -100, 100, 100, 85, -100, -100]\n",
      "iter=232  t=6.98s  best_y=186688  x=[-100, -68, 100, 100, -100, 100, 100, 87, -100, -100]\n",
      "iter=233  t=7.00s  best_y=184720  x=[-100, -70, 100, 100, -100, 100, 100, 89, -100, -100]\n",
      "iter=234  t=7.03s  best_y=182752  x=[-100, -72, 100, 100, -100, 100, 100, 91, -100, -100]\n",
      "iter=235  t=7.05s  best_y=180784  x=[-100, -74, 100, 100, -100, 100, 100, 93, -100, -100]\n",
      "iter=236  t=7.08s  best_y=178718  x=[-100, -76, 100, 100, -100, 100, 100, 95, -100, -100]\n",
      "iter=237  t=7.10s  best_y=176030  x=[-100, -78, 100, 100, -100, 100, 100, 97, -100, -100]\n",
      "iter=238  t=7.13s  best_y=173341  x=[-100, -80, 100, 100, -100, 100, 100, 99, -100, -100]\n",
      "iter=239  t=7.15s  best_y=171996  x=[-100, -81, 100, 100, -100, 100, 100, 100, -100, -100]\n",
      "STOP: local minimum. best_y=171996 x=[-100, -81, 100, 100, -100, 100, 100, 100, -100, -100]\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, 779 columns and 20107 nonzeros\n",
      "Model fingerprint: 0x3271f02d\n",
      "Variable types: 513 continuous, 266 integer (256 binary)\n",
      "Coefficient statistics:\n",
      "  Matrix range     [1e-06, 3e+01]\n",
      "  Objective range  [1e+07, 1e+07]\n",
      "  Bounds range     [1e-01, 1e+02]\n",
      "  RHS range        [1e-03, 3e+01]\n",
      "Presolve removed 923 rows and 496 columns\n",
      "Presolve time: 0.03s\n",
      "Presolved: 359 rows, 283 columns, 3652 nonzeros\n",
      "Variable types: 188 continuous, 95 integer (85 binary)\n",
      "\n",
      "Root relaxation: objective -2.807283e+06, 134 iterations, 0.00 seconds (0.00 work units)\n",
      "\n",
      "    Nodes    |    Current Node    |     Objective Bounds      |     Work\n",
      " Expl Unexpl |  Obj  Depth IntInf | Incumbent    BestBd   Gap | It/Node Time\n",
      "\n",
      "     0     0 -2807282.5    0   53          - -2807282.5      -     -    0s\n",
      "H    0     0                    202875.45648 -2807282.5  1484%     -    0s\n",
      "     0     0 -1604113.2    0   71 202875.456 -1604113.2   891%     -    0s\n",
      "     0     0 -1499504.5    0   71 202875.456 -1499504.5   839%     -    0s\n",
      "     0     0 -1486598.9    0   73 202875.456 -1486598.9   833%     -    0s\n",
      "     0     0 -1462703.9    0   72 202875.456 -1462703.9   821%     -    0s\n",
      "     0     0 -1461572.5    0   72 202875.456 -1461572.5   820%     -    0s\n",
      "     0     0 -1335489.0    0   69 202875.456 -1335489.0   758%     -    0s\n",
      "     0     0 -1257632.9    0   72 202875.456 -1257632.9   720%     -    0s\n",
      "     0     0 -1246085.4    0   73 202875.456 -1246085.4   714%     -    0s\n",
      "     0     0 -1246085.4    0   76 202875.456 -1246085.4   714%     -    0s\n",
      "     0     0 -1246085.4    0   76 202875.456 -1246085.4   714%     -    0s\n",
      "     0     0 -1246085.4    0   76 202875.456 -1246085.4   714%     -    0s\n",
      "     0     0 -1023575.2    0   76 202875.456 -1023575.2   605%     -    0s\n",
      "     0     0 -778520.29    0   76 202875.456 -778520.29   484%     -    0s\n",
      "     0     2 -778520.29    0   76 202875.456 -778520.29   484%     -    0s\n",
      "* 1017   503              42    197758.97506 -778520.29   494%  18.7    0s\n",
      "* 1147   503              43    145194.43239 -778520.29   636%  19.0    0s\n",
      "\n",
      "Cutting planes:\n",
      "  Learned: 4\n",
      "  Gomory: 5\n",
      "  Cover: 27\n",
      "  Implied bound: 63\n",
      "  Clique: 1\n",
      "  MIR: 208\n",
      "  Flow cover: 19\n",
      "  Inf proof: 11\n",
      "  RLT: 16\n",
      "  Relax-and-lift: 31\n",
      "  BQP: 2\n",
      "  PSD: 3\n",
      "\n",
      "Explored 3986 nodes (81118 simplex iterations) in 0.53 seconds (1.02 work units)\n",
      "Thread count was 14 (of 14 available processors)\n",
      "\n",
      "Solution count 3: 145194 197759 202875 \n",
      "\n",
      "Optimal solution found (tolerance 1.00e-04)\n",
      "Best objective 1.451944323941e+05, best bound 1.451944323941e+05, gap 0.0000%\n",
      "[CHECK MLP] obj(x_ip)=145194  ip_y=145194  rel_err=4.656e-07\n",
      "\n",
      "--- Dataset stats (quadratic) ---\n",
      "  X: shape=(2000, 10)  mean(mean)=-2.39  std(mean)=579  min=-1e+03  max=1e+03\n",
      "  y: shape=(2000,)  mean=4.77e+06  std=1.35e+07  min=-5.19e+07  max=5.46e+07\n",
      "\n",
      "\n",
      "=== Run: quadratic | DFN ===\n",
      "  data: N=2000  train/val/test=1400/300/300  dim=10\n",
      "  model: params=16,725 layers=[16, 128, 16] p_list=[1, 1] alpha=0.005 beta=-2.0\n",
      "  train: device=cpu  epochs=1000  batch=8  lr=0.1  wd=0  seed=0\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA90AAAGGCAYAAABmGOKbAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjgsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvwVt1zgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAsJFJREFUeJzs3Qd4k1UXB/B/k+7N3nvvrYIg4GC6QBkKTnAgKrjFrZ+IAwUHy4VbUQHFCQrIEGXI3nuvsrrbNON7zk3TJmmSJm1m+/89T2jzJiRvbtI377n33HPDTCaTCURERERERETkdRrvPyQRERERERERMegmIiIiIiIi8iGOdBMRERERERH5CINuIiIiIiIiIh9h0E1ERERERETkIwy6iYiIiIiIiHyEQTcRERERERGRjzDoJiIiIiIiIvIRBt1EREREREREPsKg202ffPIJwsLCnF7++usvBNLBgwfVfkyePLnEj7F7927ccMMNqFChAmJjY3HxxRdjwYIFxf6/kSNHque++uqri9xWv359h+117733Onwsg8GAqlWrYsqUKer6ypUrMXr0aHTq1AlRUVHq/8prdebdd99F8+bN1X0bNGiAF198EXl5eUXud/r0adx+++2oXLmyeq1du3bF4sWLHT7mn3/+qW6X+8n95f/J/3fHCy+8oNogEKSt5PnduZ9c5HU58tJLLxXcx7rtTSYTvvnmG/To0UO9Z9HR0ahduzb69u2LDz/80OFzOLo4e15PfP3117jssstQrVo19d7XrFkT11xzDVatWuXW//f0c3bo0CHceeed6nnk/rVq1cKgQYNs7nP06FGMHz8ePXv2RHJysnpMOY74iruf6Z9//hm33nor2rRpg4iICLVfntq3b5963f/88w9CkRzrIiMjsX79+kDvChG5IMfVmJgYXLhwwel9RowYoY5lp06d8vr3o7+98sor+OGHH4ps3759u9pfV99LvvLZZ5+hSpUqSE9P99tzuvv9mZaWhokTJ6JXr16oXr064uPj1Xfba6+9hpycnCL337t3L2655RbUrVtXfa4aNWqEhx9+GGfPnrW5n9zn+uuv9+lrpPKHQbeHZs+erU407S8dO3ZEKJMDuZyk79q1CzNnzsR3332nDrJy0Jk7d67T//fLL7+oL4jExESn97n00kuLtNcTTzzh8L7Lly9HSkoKBg8erK5L0CBB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+IiIiAKFQTcRERG5bdGiRdi7d6+61K5du8iSd0RERGSL+WBERETktttvv10F144uREREVBSDbiIiIiIiIiIfYdBNRERERERE5CMMuomIiIiIiIh8hEE3ERERERERkY+EXPVyS6GWtLS0QO8KERFRiVi+w8pz8TF+nxMRUXn5Pg+5oDs9PV39rFOnTqB3hYiIqNTfaUlJSeWyFfl9TkRE5eX7PMwUYt3sRqMRx48fR0JCAsLCwrzSOyEB/JEjR5CYmOiVfSzP2J5sy2DFzybbM5jIV698QdesWRMaTfmc6VXa73P+TXsf25TtGez4GWWbhur3eciNdMuLqV27ttcfVwJuBt1sz2DEzybbM5jx81ly5XWE29vf5/wMeh/blO0Z7PgZZZuG2vd5+exeJyIiIiIiIvIDBt1EREREREREPlLug+6oqCg8//zz6ieVHtvTe9iW3sX2ZHtS2cK/abZpsONnlG0aCvg59Y+QK6RGREREREREFCrK/Ug3ERERERERka8w6CYiIiIiIiLyEQbdRERERERERD5SroPu6dOno0GDBoiOjkanTp2wYsWKQO9S0Jk0aRK6dOmChIQEVK1aFddffz127dplcx8pC/DCCy+oReFjYmLQq1cvbNu2zeY+ubm5eOCBB1C5cmXExcXh2muvxdGjR1HeSfuGhYVh/PjxBdvYnp45duwYRo4ciUqVKiE2Nhbt27fHf//9x/YsAb1ej2eeeUYdF+VvuWHDhnjppZdgNBrZnhTUDh48iFGjRhV8dhs1aqSKpOp0ukDvWsiaOHEiunXrpo6rycnJgd6dkMTzTO9Zvnw5rrnmGnWuKedNP/zwgxcfvfxx5/yevKvcBt1z5sxRgc7TTz+NDRs2oEePHujfvz8OHz4c6F0LKsuWLcPYsWPx77//4o8//lAn5X369EFmZmbBfV5//XW89dZbeO+997B27VpUr14dV111FdLT0wvuI209f/58fPPNN1i5ciUyMjJw9dVXw2AwoLyStnr//ffRtm1bm+1sT/edP38el156KSIiIvDbb79h+/btePPNN21OENme7nvttdcwc+ZM9be8Y8cO1XZvvPEG3n33XbYnBbWdO3eqzqFZs2apTt8pU6aoz/JTTz0V6F0LWdJhMWTIEIwZMybQuxKSeJ7pXXLe2a5dO/X9RP45vycvM5VTF110kenee++12da8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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "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=2000, dim=10, eigen_min=-1, eigen_max=15.0,\n",
    "    x_min=-1000, x_max=1000, noise_std=0.1, seed=0\n",
    ")\n",
    "in_dim = int(dataset_params[\"dim\"])\n",
    "\n",
    "train_base = dict(\n",
    "    epochs=1000,\n",
    "    batch_size=8,\n",
    "    val_frac=0.15,\n",
    "    test_frac=0.15,\n",
    "    seed=0,\n",
    "    device=(\"cuda\" if torch.cuda.is_available() else \"cpu\"),\n",
    "    eps=1e-8,\n",
    "    weight_decay=0.0,\n",
    "    plot_every=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=[16, 128, 16], p_list=[1, 1],\n",
    "    seed=0, alpha=5e-3, beta=-2.0\n",
    ")\n",
    "\n",
    "# ---- DFN (fixed A = I) ----\n",
    "dfnA_layer_sizes = [5, 400, 6]\n",
    "assert dfnA_layer_sizes[0] + dfnA_layer_sizes[-1] == in_dim + 1, \"Need |L1|+|LK| = dim+1 for A_fixed=I\"\n",
    "\n",
    "dfn_Afix_params = dict(\n",
    "    input_dim=in_dim,\n",
    "    layer_sizes=dfnA_layer_sizes,\n",
    "    p_list=[1, 1],\n",
    "    seed=0,\n",
    "    alpha=5e-3,\n",
    "    beta=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
}
