{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "96398989-280e-41b3-bc11-99b4b51e9439",
   "metadata": {},
   "source": [
    "# General Helpers"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "767d34c9-66bd-4986-a324-4daeec80aee1",
   "metadata": {},
   "source": [
    "### Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "bef302bd-4ca0-4269-9ba1-ac6d05ef5819",
   "metadata": {},
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "import sys, json, time, math, random, tempfile\n",
    "from pathlib import Path\n",
    "from typing import Optional, Tuple, Dict, List\n",
    "import numpy as np\n",
    "import cppimport\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from IPython.display import Image, display\n",
    "from dataclasses import dataclass\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3cd5a10c-7768-4318-8244-beaf6c4c84d1",
   "metadata": {},
   "source": [
    "### Python Implementation of LEMON \n",
    "The following defines an importable Python module `lemon_mcf` which runs LEMON solvers (written in C++)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "b4658685-aab5-4c45-aa6d-36e862bbdc9a",
   "metadata": {},
   "outputs": [],
   "source": [
    "repo = Path().resolve().parent\n",
    "sys.path.insert(0, str(repo))\n",
    "lemon_mcf = cppimport.imp(\"lemon_mcf\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "800144cd-efc4-47e8-95d6-e75cad74419d",
   "metadata": {},
   "source": [
    "The module has the function `lemon_mcf.solve_mcf` which solves the minimum cost flow problem. The function can be described as follows:\n",
    "\n",
    "**Signature**\n",
    "    out = lemon_mcf.solve_mcf(n, src, dst, cost, cap, supply, tol=1e-9)\n",
    "    \n",
    "**Inputs (NumPy arrays)**\n",
    "- `n` : `int` \u2014 number of nodes (indexed `0..n-1`)\n",
    "- `src`, `dst` : `np.ndarray` shape `(m,)`, dtype `int64` \u2014 directed edges `src[i] -> dst[i]`\n",
    "- `cost`, `cap` : `np.ndarray` shape `(m,)`, dtype `float64` \u2014 per-edge cost and capacities\n",
    "- `supply` : `np.ndarray` shape `(n,)`, dtype `float64` \u2014 node supplies/demands (`>0` supply, `<0` demand)\n",
    "- `tol` : `float` \u2014 tolerance for capacity-status flags\n",
    "\n",
    "**Output (`dict`)**\n",
    "- `out[\"status\"]` : `int`\n",
    "- `out[\"flow\"]` : `np.ndarray` shape `(m,)`, dtype `float64`\n",
    "- `out[\"potential\"]` : `np.ndarray` shape `(n,)`, dtype `float64` (node potentials; defined up to an additive constant)\n",
    "- `out[\"reduced_cost\"]` : `np.ndarray` shape `(m,)`, dtype `float64`, computed as `cost[i] + potential[src[i]] - potential[dst[i]]`\n",
    "- (optional) capacity-status flag: boolean arrays indicating whether each arc is at its capacity\n",
    "- `out[\"total_cost\"]` : `float` (objective value)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1ff805be-adf0-44c9-a12c-d4d2fb077251",
   "metadata": {},
   "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"
     ]
    }
   ],
   "source": [
    "# ----- Example of Usage ------ #\n",
    "\n",
    "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)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5edb874-0d6f-4c0a-a1c0-8707a04bd0c7",
   "metadata": {},
   "source": [
    "It also has the function `lemon_mcf.max_flow` which solves the maximum s-t flow problem. The function can be described as follows:\n",
    "\n",
    "**Signature**\n",
    "    out = lemon_mcf.max_flow(n, src, dst, cap, s, t)\n",
    "\n",
    "**Inputs (NumPy arrays)**\n",
    "- `n` : `int` \u2014 number of nodes (indexed `0..n-1`)\n",
    "- `src`, `dst` : `np.ndarray` shape `(m,)`, dtype `int64` \u2014 directed edges `src[i] -> dst[i]`\n",
    "- `cap` : `np.ndarray` shape `(m,)`, dtype `float64` \u2014 per-edge capacities (nonnegative)\n",
    "- `s` : `int` \u2014 source node index\n",
    "- `t` : `int` \u2014 sink node index\n",
    "\n",
    "**Output (`dict`)**\n",
    "- `out[\"value\"]` : `float` \u2014 maximum flow value from `s` to `t`\n",
    "- `out[\"flow\"]` : `np.ndarray` shape `(m,)`, dtype `float64` \u2014 per-edge flow values (in the same order as the input edges)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "84d7091d-b4dd-46db-84b2-e580dd826951",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'value': 4.0, 'flow': array([2., 2., 2., 2.])}\n"
     ]
    }
   ],
   "source": [
    "# ----- Example of Usage ------ #\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": "b2d023ed-4551-4a12-895b-5499538ccfe3",
   "metadata": {},
   "source": [
    "# Generating Datasets"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9c19ed1a-276c-4f7b-930b-15ee5dd1a087",
   "metadata": {},
   "source": [
    "## Synthetic Quadratic "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e668b358-356c-4b0f-8f52-3fa79709f650",
   "metadata": {},
   "source": [
    "#### Quadratic Instance Generator"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d51f4f1d-d52d-4167-8546-975f7056157a",
   "metadata": {},
   "source": [
    "The main function can be described as follows:\n",
    "\n",
    "**Signature**\n",
    "    data = generate_synthetic_convex_quadratic_dataset(d, m, M, X, sigma2, K, seed=None)\n",
    "\n",
    "**Inputs**\n",
    "- `d` : `int` \u2014 dimension (`d > 0`)\n",
    "- `m`, `M` : `float` \u2014 eigenvalue bounds (`0 < m <= M`)\n",
    "- `X` : `int` \u2014 integer box bound (samples are uniform over `[-X, X]^d`, coordinate-wise)\n",
    "- `sigma2` : `float` \u2014 noise variance (`sigma2 >= 0`)\n",
    "- `K` : `int` \u2014 number of samples (`K > 0`)\n",
    "- `seed` : `int` or `None` \u2014 RNG seed\n",
    "\n",
    "**Process**\n",
    "1. Sample eigenvalues `\u03bb_1,...,\u03bb_d ~ Uniform([m, M])` i.i.d.\n",
    "2. Sample an (approximately) Haar-uniform orthonormal matrix `U \u2208 R^{d\u00d7d}` via QR of a Gaussian matrix.\n",
    "3. Form `Q = U diag(\u03bb) U^T` (positive definite since `m > 0`).\n",
    "4. Sample an integer optimizer `x*` uniformly from the integer box `[-X, X]^d`.\n",
    "5. Sample `K` integer covariates `x^1,...,x^K` uniformly from `[-X, X]^d`.\n",
    "6. Sample i.i.d. noise `\u03b5^k ~ N(0, sigma2)` for `k=1,...,K`.\n",
    "7. Set targets `y^k = (x^k - x*)^T Q (x^k - x*) + \u03b5^k`.\n",
    "\n",
    "**Output (`dict`)**\n",
    "- `data[\"U\"]` : `np.ndarray` shape `(d, d)` \u2014 sampled orthonormal matrix\n",
    "- `data[\"lambdas\"]` : `np.ndarray` shape `(d,)` \u2014 sampled eigenvalues\n",
    "- `data[\"Q\"]` : `np.ndarray` shape `(d, d)` \u2014 sampled quadratic matrix `Q = U diag(lambdas) U^T`\n",
    "- `data[\"x_star\"]` : `np.ndarray` shape `(d,)`, dtype `int` \u2014 sampled optimizer `x*`\n",
    "- `data[\"X_samples\"]` : `np.ndarray` shape `(K, d)`, dtype `int` \u2014 sampled inputs `x^k`\n",
    "- `data[\"y\"]` : `np.ndarray` shape `(K,)`, dtype `float` \u2014 targets `y^k`\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "edd30065-8973-4a0f-8315-99a96ac42355",
   "metadata": {},
   "outputs": [],
   "source": [
    "def _haar_orthonormal_matrix(d: int, rng: np.random.Generator) -> np.ndarray:\n",
    "    \"\"\"\n",
    "    Sample an (approximately) Haar-uniform random orthonormal matrix U in R^{dxd}\n",
    "    using QR decomposition of a standard Gaussian matrix, with a sign correction.\n",
    "    \"\"\"\n",
    "    A = rng.standard_normal((d, d))\n",
    "    U, R = np.linalg.qr(A)\n",
    "    s = np.sign(np.diag(R))\n",
    "    s[s == 0] = 1.0\n",
    "    U = U * s  # multiply columns by sign\n",
    "    return U\n",
    "\n",
    "def generate_synthetic_convex_quadratic_dataset(\n",
    "    d: int,\n",
    "    m: float,\n",
    "    M: float,\n",
    "    X: int,\n",
    "    sigma2: float,\n",
    "    K: int,\n",
    "    seed: int | None = None\n",
    "):\n",
    "    if d <= 0:\n",
    "        raise ValueError(\"d must be positive.\")\n",
    "    if not (M >= m):\n",
    "        raise ValueError(\"Require M >= m.\")\n",
    "    if m <= 0:\n",
    "        raise ValueError(\"Require m > 0 so that Q is positive definite.\")\n",
    "    if K <= 0:\n",
    "        raise ValueError(\"K must be positive.\")\n",
    "    if sigma2 < 0:\n",
    "        raise ValueError(\"sigma2 must be nonnegative.\")\n",
    "    if not isinstance(X, (int, np.integer)):\n",
    "        raise ValueError(\"X must be an integer.\")\n",
    "    X = int(X)\n",
    "    if X < 0:\n",
    "        raise ValueError(\"X must be a nonnegative integer.\")\n",
    "\n",
    "    rng = np.random.default_rng(seed)\n",
    "\n",
    "    # (i) sample eigenvalues\n",
    "    lambdas = rng.uniform(m, M, size=d)\n",
    "\n",
    "    # (ii) sample orthonormal matrix U\n",
    "    U = _haar_orthonormal_matrix(d, rng)\n",
    "\n",
    "    # Build Q = U diag(lambdas) U^T\n",
    "    Q = U @ np.diag(lambdas) @ U.T\n",
    "\n",
    "    # Sample integer vectors uniformly from [-X, X]^d\n",
    "    def sample_int_box(num: int) -> np.ndarray:\n",
    "        return rng.integers(-X, X + 1, size=(num, d), dtype=int)\n",
    "\n",
    "    # (iii) sample x*\n",
    "    x_star = sample_int_box(1).reshape(-1)  # shape (d,)\n",
    "\n",
    "    # (iv) sample K vectors x^1,...,x^K\n",
    "    X_samples = sample_int_box(K)  # shape (K, d)\n",
    "\n",
    "    # (v) sample Gaussian noises\n",
    "    sigma = float(np.sqrt(sigma2))\n",
    "    eps = rng.normal(loc=0.0, scale=sigma, size=K)\n",
    "\n",
    "    # (vi) compute y^k\n",
    "    diffs = X_samples - x_star  # shape (K, d)\n",
    "    quad = np.einsum(\"bi,ij,bj->b\", diffs, Q, diffs)  # shape (K,)\n",
    "    y = quad + eps\n",
    "\n",
    "    return {\n",
    "        \"U\": U,\n",
    "        \"lambdas\": lambdas,\n",
    "        \"Q\": Q,\n",
    "        \"x_star\": x_star,\n",
    "        \"X_samples\": X_samples,\n",
    "        \"y\": y,\n",
    "    }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "666e79e4-8daa-483c-995d-6a3b92367b88",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "x_star: [ 7  4  4 -2  8]\n",
      "X_samples shape: (100, 5)\n",
      "y shape: (100,)\n",
      "min eigenvalue(Q) ~ 0.5413190888213227\n"
     ]
    }
   ],
   "source": [
    "# --- Example of Usage --- #\n",
    "data = generate_synthetic_convex_quadratic_dataset(\n",
    "    d=5, m=0.5, M=3.0, X=10, sigma2=0.25, K=100, seed=0\n",
    ")\n",
    "\n",
    "print(\"x_star:\", data[\"x_star\"])\n",
    "print(\"X_samples shape:\", data[\"X_samples\"].shape)\n",
    "print(\"y shape:\", data[\"y\"].shape)\n",
    "print(\"min eigenvalue(Q) ~\", np.min(np.linalg.eigvalsh(data[\"Q\"])))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f403d314-b3f6-4933-b466-a6b3363839c7",
   "metadata": {},
   "source": [
    "#### Full Dataset Generator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "302dbd89-36d9-45e5-a1a4-d34846a4a6fc",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "4c49e91f-881b-4020-8878-c22cf217c00c",
   "metadata": {},
   "source": [
    "## Synthetic Min Cost Flow"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1f773417-3500-4b5a-b931-59eacf5b61c2",
   "metadata": {},
   "source": [
    "#### NETGEN Instance Generator"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d6c6656d-e5ac-45fd-975d-398e4fbe472b",
   "metadata": {},
   "source": [
    "The main function can be described as follows:\n",
    "\n",
    "**Signature**\n",
    "    u, v, cap, cost, b = netgen_instance_arrays(nodes, arcs, sources, sinks, cost_bounds, cap_bounds, total_supply=1000, seed=None)\n",
    "\n",
    "**Inputs**\n",
    "- `nodes` : `int` \u2014 number of nodes (`nodes > 0`)\n",
    "- `arcs` : `int` \u2014 number of directed arcs/edges to generate (`arcs > 0`)\n",
    "- `sources` : `int` \u2014 number of supply (source) nodes (`sources >= 0`)\n",
    "- `sinks` : `int` \u2014 number of demand (sink) nodes (`sinks >= 0`)\n",
    "- `cost_bounds` : `tuple[int, int]` \u2014 arc cost bounds `(min_cost, max_cost)` with `min_cost <= max_cost`\n",
    "- `cap_bounds` : `tuple[int, int]` \u2014 arc capacity bounds `(min_cap, max_cap)` with `min_cap <= max_cap`\n",
    "- `total_supply` : `int` \u2014 total supply injected into the network (`total_supply >= 0`)\n",
    "- `seed` : `int` or `None` \u2014 RNG seed\n",
    "\n",
    "**Output (arrays)**\n",
    "Returns a 5-tuple:\n",
    "\n",
    "- `u` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 tail (source) node index for each arc (0-based)\n",
    "- `v` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 head (destination) node index for each arc (0-based)  \n",
    "  (arc `i` is `u[i] -> v[i]`)\n",
    "- `cap` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 capacity for each arc\n",
    "- `cost` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 per-unit cost for each arc\n",
    "- `b` : `np.ndarray` shape `(nodes,)`, dtype `int` \u2014 flow balance for each node"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "7fb8ebaa-0e13-4c0b-bbf6-00c0b60c4547",
   "metadata": {},
   "outputs": [],
   "source": [
    "def _parse_dimacs_mcf(path: Path) -> tuple[int, Dict[int, int], List[tuple[int, int, int, int]]]:\n",
    "    n = None\n",
    "    supply: Dict[int, int] = {}\n",
    "    arcs: List[tuple[int, int, int, int]] = []  # (u, v, cap, cost)\n",
    "\n",
    "    for raw in path.read_text(encoding=\"utf-8\").splitlines():\n",
    "        line = raw.strip()\n",
    "        if not line or line.startswith(\"c\"):\n",
    "            continue\n",
    "        parts = line.split()\n",
    "        if parts[0] == \"p\":\n",
    "            n = int(parts[2])\n",
    "        elif parts[0] == \"n\":\n",
    "            supply[int(parts[1])] = int(parts[2])\n",
    "        elif parts[0] == \"a\":\n",
    "            u, v, ignored, cap, cost = map(int, parts[1:])\n",
    "            if ignored != 0:\n",
    "                raise ValueError(f\"Unexpected nonzero ignored field in DIMACS arc line: value={ignored}\")\n",
    "            arcs.append((u, v, cap, cost))\n",
    "\n",
    "    if n is None:\n",
    "        raise ValueError(\"Missing 'p min N M' line.\")\n",
    "    return n, supply, arcs\n",
    "\n",
    "\n",
    "def netgen_instance_arrays(\n",
    "    *,\n",
    "    nodes: int,\n",
    "    arcs: int,\n",
    "    sources: int,\n",
    "    sinks: int,\n",
    "    cost_bounds: Tuple[int, int],\n",
    "    cap_bounds: Tuple[int, int],\n",
    "    total_supply: int = 1000,\n",
    "    seed: Optional[int] = None,\n",
    "):\n",
    "    \"\"\"\n",
    "    Returns:\n",
    "      u, v, cap, cost, b\n",
    "    where:\n",
    "      - u,v,cap,cost are arrays of length m\n",
    "      - b is an array of length n with supply(+)/demand(-), sum(b)=0\n",
    "      - nodes are 0-based indices\n",
    "    \"\"\"\n",
    "    if nodes <= 0 or arcs <= 0:\n",
    "        raise ValueError(\"nodes and arcs must be positive.\")\n",
    "    if sources < 0 or sinks < 0 or sources + sinks > nodes:\n",
    "        raise ValueError(\"Need sources+sinks <= nodes.\")\n",
    "    minc, maxc = cost_bounds\n",
    "    mincap, maxcap = cap_bounds\n",
    "    if minc > maxc or mincap > maxcap:\n",
    "        raise ValueError(\"Bad bounds: min must be <= max.\")\n",
    "    if seed is None:\n",
    "        seed = random.randint(1, 2_000_000_000)\n",
    "\n",
    "    try:\n",
    "        import pynetgen  # type: ignore\n",
    "    except ImportError as e:\n",
    "        raise RuntimeError(\n",
    "            \"pynetgen is not installed.\"\n",
    "        ) from e\n",
    "\n",
    "    tmpdir = Path(tempfile.mkdtemp(prefix=\"netgen_\"))\n",
    "    dimacs_path = tmpdir / \"instance.dimacs\"\n",
    "\n",
    "    pynetgen.netgen_generate(\n",
    "        seed=seed,\n",
    "        nodes=nodes,\n",
    "        sources=sources,\n",
    "        sinks=sinks,\n",
    "        density=arcs,\n",
    "        mincost=minc,\n",
    "        maxcost=maxc,\n",
    "        supply=total_supply,\n",
    "        tsources=0,\n",
    "        tsinks=0,\n",
    "        hicost=0,\n",
    "        capacitated=100,\n",
    "        mincap=mincap,\n",
    "        maxcap=maxcap,\n",
    "        rng=1,\n",
    "        type=None,\n",
    "        fname=str(dimacs_path),\n",
    "    )\n",
    "\n",
    "    n, supply_1b, arc_list_1b = _parse_dimacs_mcf(dimacs_path)\n",
    "    m = len(arc_list_1b)\n",
    "\n",
    "    u = np.empty(m, dtype=np.int64)\n",
    "    v = np.empty(m, dtype=np.int64)\n",
    "    cap = np.empty(m, dtype=np.int64)\n",
    "    cost = np.empty(m, dtype=np.int64)\n",
    "\n",
    "    for i, (uu, vv, c, w) in enumerate(arc_list_1b):\n",
    "        u[i] = uu - 1\n",
    "        v[i] = vv - 1\n",
    "        cap[i] = c\n",
    "        cost[i] = w\n",
    "\n",
    "    b = np.zeros(n, dtype=np.int64)\n",
    "    for node_1b, val in supply_1b.items():\n",
    "        b[node_1b - 1] = val\n",
    "\n",
    "    if b.sum() != 0:\n",
    "        raise ValueError(f\"Unbalanced instance: sum(b)={b.sum()} (should be 0).\")\n",
    "\n",
    "    return u, v, cap, cost, b"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "51948156-ee64-4380-9cd9-f31f811dd585",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=200, m=1500, balance sum(b)=0\n",
      "#sources=10, #sinks=10\n",
      "sources (node, supply): [(0, 1054), (1, 864), (2, 372), (3, 49), (4, 299), (5, 917), (6, 565), (7, 45), (8, 711), (9, 124)]\n",
      "sinks   (node, demand): [(190, -546), (191, -1003), (192, -42), (193, -49), (194, -1021), (195, -621), (196, -432), (197, -768), (198, -33), (199, -485)]\n",
      "\n",
      "First edges:\n",
      "   i      u      v     cap    cost\n",
      "-----------------------------------\n",
      "   0      0    118    1054      23\n",
      "   1      0    175      85       8\n",
      "   2      0    109      97      13\n",
      "   3      0    134      56      41\n",
      "   4      0     95      93      26\n",
      "   5      0    131      96       6\n",
      "   6      0     50      17       2\n",
      "   7      0     48      60      42\n",
      "   8      0     47      77      31\n",
      "   9      0    187      20      36\n",
      "\n",
      "cap stats: 1 108.12266666666666 1054\n",
      "cost stats: 1 24.924 50\n"
     ]
    }
   ],
   "source": [
    "# ------ Example of usage ------ #\n",
    "\n",
    "u, v, cap, cost, b = netgen_instance_arrays(\n",
    "    nodes=200,\n",
    "    arcs=1500,\n",
    "    sources=10,\n",
    "    sinks=10,\n",
    "    cost_bounds=(1, 50),\n",
    "    cap_bounds=(1, 100),\n",
    "    total_supply=5000,\n",
    "    seed=7,\n",
    ")\n",
    "\n",
    "n = len(b)\n",
    "m = len(u)\n",
    "print(f\"n={n}, m={m}, balance sum(b)={int(b.sum())}\")\n",
    "\n",
    "src = np.where(b > 0)[0]\n",
    "snk = np.where(b < 0)[0]\n",
    "print(f\"#sources={len(src)}, #sinks={len(snk)}\")\n",
    "print(\"sources (node, supply):\", [(int(i), int(b[i])) for i in src[:10]])\n",
    "print(\"sinks   (node, demand):\", [(int(i), int(b[i])) for i in snk[:10]])\n",
    "\n",
    "k = 10\n",
    "k = min(k, m)\n",
    "print(\"\\nFirst edges:\")\n",
    "print(f\"{'i':>4}  {'u':>5}  {'v':>5}  {'cap':>6}  {'cost':>6}\")\n",
    "print(\"-\" * 35)\n",
    "for i in range(k):\n",
    "    print(f\"{i:>4}  {int(u[i]):>5}  {int(v[i]):>5}  {int(cap[i]):>6}  {int(cost[i]):>6}\")\n",
    "\n",
    "print(\"\\ncap stats:\", int(cap.min()), float(cap.mean()), int(cap.max()))\n",
    "print(\"cost stats:\", int(cost.min()), float(cost.mean()), int(cost.max()))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "273eeb7d-38f9-4cdb-9c43-1c3e920c792d",
   "metadata": {},
   "source": [
    "#### NETGEN Dataset Generator"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "723aab3f-b920-44d6-a891-dc87f6a89b61",
   "metadata": {},
   "source": [
    "The main function can be described as follows:\n",
    "\n",
    "**Signature**\n",
    "    data = generate_netgen_mcf_dataset(n, family, K, tilde_C, sigma2, seed=None, cap_range=(1,1000), cost_range=(1,10000), total_supply=1000, verbose=False)\n",
    "\n",
    "**Inputs**\n",
    "- `n` : `int` \u2014 number of nodes in the NETGEN network (`n > 0`)\n",
    "- `family` : `str` \u2014 family: `\"sparse\"` or `\"dense\"`\n",
    "  - `\"sparse\"` uses `m = 8n` arcs\n",
    "  - `\"dense\"` uses `m = ceil(n * sqrt(n))` arcs\n",
    "- `K` : `int` \u2014 number of balance/label samples to generate (`K > 0`)\n",
    "- `tilde_C` : `int` \u2014 max auxiliary cost used when generating balances (`tilde_C > 0`)\n",
    "- `sigma2` : `float` \u2014 label noise variance (`sigma2 >= 0`)\n",
    "- `seed` : `int` or `None` \u2014 RNG seed\n",
    "- `cap_range` : `(int, int)` \u2014 integer capacity range for NETGEN arcs (default `(1,1000)`)\n",
    "- `cost_range` : `(int, int)` \u2014 integer cost range for NETGEN arcs (default `(1,10000)`)\n",
    "- `total_supply` : `int` \u2014 total supply in the feasible balance NETGEN produces (needed to generate network but balances are discarded)\n",
    "- `verbose` : `bool` \u2014 print `F_max` and basic diagnostics if `True`\n",
    "\n",
    "**Process**\n",
    "1. Set the number of supply and demand nodes to  \n",
    "   `|S| = |T| = floor(sqrt(n))`.\n",
    "2. Choose the number of arcs `m` based on `family`:\n",
    "   - If `\"sparse\"`: `m = 8n`\n",
    "   - If `\"dense\"`:  `m = ceil(n * sqrt(n))`\n",
    "3. Generate a NETGEN minimum-cost flow instance on `n` nodes with:\n",
    "   - arc capacities `u_e ~ Unif({1,...,1000})`\n",
    "   - arc costs `c_e ~ Unif({1,...,10000})`\n",
    "   NETGEN also returns a feasible balance vector `b0`; discard its magnitudes but use its sign pattern to\n",
    "   identify the node sets:\n",
    "   - `S = {v : b0(v) > 0}` (supplies)\n",
    "   - `T = {v : b0(v) < 0}` (demands)\n",
    "4. Augment the network by adding a super-source `s` and super-sink `t`:\n",
    "   - add arcs `(s,v)` for all `v \u2208 S` with capacity `+\u221e`\n",
    "   - add arcs `(v,t)` for all `v \u2208 T` with capacity `+\u221e`\n",
    "5. Compute `F_max`, the value of a maximum `s\u2013t` flow in the augmented network.\n",
    "6. For each sample `k = 1,...,K`:\n",
    "   1. Sample a target flow value `F^(k) ~ Unif({0,1,...,F_max})`.\n",
    "   2. Sample auxiliary integer arc costs on the augmented network  \n",
    "      `tilde_c_e^(k) ~ Unif({1,...,tilde_C})`.\n",
    "   3. Solve a minimum-cost `s\u2013t` flow of value `F^(k)` on the augmented network under costs `tilde_c^(k)`,\n",
    "      obtaining a flow `f^(k)`.\n",
    "   4. Define the balance vector `b^(k) \u2208 Z^n` on the original node set by:\n",
    "      - `b^(k)(v) = f_sv^(k)` for `v \u2208 S`\n",
    "      - `b^(k)(v) = -f_vt^(k)` for `v \u2208 T`\n",
    "      - `b^(k)(v) = 0` otherwise\n",
    "   5. Solve the original minimum-cost flow problem on `G(V,E)` with capacities `u`, costs `c`,\n",
    "      and balance `b^(k)` to obtain the optimal cost `z^(k)`.\n",
    "   6. Sample noise `\u03b5^(k) ~ N(0, sigma2)` and set the label `y^(k) = z^(k) + \u03b5^(k)`.\n",
    "\n",
    "**Output (`dict`)**\n",
    "- `data[\"n\"]` : `int` \u2014 number of nodes\n",
    "- `data[\"m\"]` : `int` \u2014 number of arcs in the original NETGEN graph\n",
    "- `data[\"u\"]` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 arc tail nodes (0-indexed)\n",
    "- `data[\"v\"]` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 arc head nodes (0-indexed)\n",
    "- `data[\"cap\"]` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 arc capacities\n",
    "- `data[\"cost\"]` : `np.ndarray` shape `(m,)`, dtype `int` \u2014 arc costs\n",
    "- `data[\"S_nodes\"]` : `np.ndarray` shape `(|S|,)`, dtype `int` \u2014 supply node indices\n",
    "- `data[\"T_nodes\"]` : `np.ndarray` shape `(|T|,)`, dtype `int` \u2014 demand node indices\n",
    "- `data[\"X_samples\"]` : `np.ndarray` shape `(K, n)`, dtype `int` \u2014 sampled balance vectors `b^(k)`\n",
    "- `data[\"y\"]` : `np.ndarray` shape `(K,)`, dtype `float` \u2014 noisy labels `y^(k)`\n",
    "- `data[\"z\"]` : `np.ndarray` shape `(K,)`, dtype `float` \u2014 clean optimal costs `z^(k)` (before noise)\n",
    "- `data[\"F_max\"]` : `int` \u2014 maximum `s\u2013t` flow value in the augmented network\n",
    "- `data[\"F_targets\"]` : `np.ndarray` shape `(K,)`, dtype `int` \u2014 sampled target values `F^(k)`\n",
    "- `data[\"tilde_C\"]` : `int` \u2014 auxiliary cost bound used\n",
    "- `data[\"sigma2\"]` : `float` \u2014 noise variance used\n",
    "- `data[\"seed\"]` : `int` or `None` \u2014 RNG seed used\n",
    "- `data[\"family\"]` : `str` \u2014 `\"sparse\"` or `\"dense\"`"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8d6820d3-b93a-4172-a39e-4c96595e5999",
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import numpy as np\n",
    "\n",
    "def generate_netgen_mcf_dataset(\n",
    "    n: int,\n",
    "    family: str,                 # \"sparse\" or \"dense\"\n",
    "    K: int,\n",
    "    tilde_C: int,\n",
    "    sigma2: float,\n",
    "    seed: int | None = None,\n",
    "    cap_range: tuple[int, int] = (1, 1000),\n",
    "    cost_range: tuple[int, int] = (1, 10000),\n",
    "    total_supply: int = 1000,\n",
    "    verbose: bool = False,\n",
    "):\n",
    "    if n <= 0:\n",
    "        raise ValueError(\"n must be positive.\")\n",
    "    if K <= 0:\n",
    "        raise ValueError(\"K must be positive.\")\n",
    "    if tilde_C <= 0:\n",
    "        raise ValueError(\"tilde_C must be positive.\")\n",
    "    if sigma2 < 0:\n",
    "        raise ValueError(\"sigma2 must be nonnegative.\")\n",
    "    if family not in {\"sparse\", \"dense\"}:\n",
    "        raise ValueError(\"family must be 'sparse' or 'dense'.\")\n",
    "\n",
    "    rng = np.random.default_rng(seed)\n",
    "\n",
    "    # --- NETGEN sizes ---\n",
    "    S_cnt = int(math.isqrt(n))                 # floor(sqrt(n))\n",
    "    T_cnt = int(math.isqrt(n))                 # floor(sqrt(n))\n",
    "    if family == \"sparse\":\n",
    "        m_target = 8 * n\n",
    "    else:\n",
    "        m_target = int(math.ceil(n * math.sqrt(n)))\n",
    "\n",
    "    # --- generate NETGEN instance (graph + a feasible b we will discard) ---\n",
    "    u, v, cap, cost, b0 = netgen_instance_arrays(\n",
    "        nodes=n,\n",
    "        arcs=m_target,\n",
    "        sources=S_cnt,\n",
    "        sinks=T_cnt,\n",
    "        cost_bounds=cost_range,\n",
    "        cap_bounds=cap_range,\n",
    "        total_supply=total_supply,\n",
    "        seed=None if seed is None else int(seed),\n",
    "    )\n",
    "\n",
    "    # Identify supply/demand node sets from NETGEN's b (we discard magnitudes, keep sets)\n",
    "    S_nodes = np.where(b0 > 0)[0]\n",
    "    T_nodes = np.where(b0 < 0)[0]\n",
    "\n",
    "    if len(S_nodes) != S_cnt or len(T_nodes) != T_cnt:\n",
    "        raise RuntimeError(\n",
    "            f\"NETGEN returned |S|={len(S_nodes)}, |T|={len(T_nodes)} but expected \"\n",
    "            f\"{S_cnt} and {T_cnt}. (Check generator settings.)\"\n",
    "        )\n",
    "\n",
    "    # --- build augmented network with super-source s and super-sink t ---\n",
    "    s = n\n",
    "    t = n + 1\n",
    "    N_aug = n + 2\n",
    "\n",
    "    # \"infinite\" capacity for super arcs: big enough not to bind\n",
    "    INF = float(np.sum(cap, dtype=np.int64) + 1)\n",
    "\n",
    "    # original arcs\n",
    "    src_aug = [u.astype(np.int64)]\n",
    "    dst_aug = [v.astype(np.int64)]\n",
    "    cap_aug = [cap.astype(np.float64)]\n",
    "\n",
    "    # s -> v for v in S\n",
    "    src_s = np.full(len(S_nodes), s, dtype=np.int64)\n",
    "    dst_s = S_nodes.astype(np.int64)\n",
    "    cap_s = np.full(len(S_nodes), INF, dtype=np.float64)\n",
    "\n",
    "    # v -> t for v in T\n",
    "    src_t = T_nodes.astype(np.int64)\n",
    "    dst_t = np.full(len(T_nodes), t, dtype=np.int64)\n",
    "    cap_t = np.full(len(T_nodes), INF, dtype=np.float64)\n",
    "\n",
    "    src_aug = np.concatenate(src_aug + [src_s, src_t])\n",
    "    dst_aug = np.concatenate(dst_aug + [dst_s, dst_t])\n",
    "    cap_aug = np.concatenate(cap_aug + [cap_s, cap_t])\n",
    "\n",
    "    # Remember arc indices for extracting b^(k)\n",
    "    m0 = len(u)\n",
    "    idx_s_to_v = {int(vv): m0 + i for i, vv in enumerate(S_nodes)}\n",
    "    idx_v_to_t = {int(vv): m0 + len(S_nodes) + i for i, vv in enumerate(T_nodes)}\n",
    "\n",
    "    # --- compute F_max via max s-t flow on augmented network ---\n",
    "    out_mf = lemon_mcf.max_flow(N_aug, src_aug, dst_aug, cap_aug, s, t)\n",
    "    F_max = int(out_mf[\"value\"])\n",
    "    if verbose:\n",
    "        print(f\"[augmented] F_max = {F_max}\")\n",
    "\n",
    "    # --- generate K balances and labels ---\n",
    "    B = np.zeros((K, n), dtype=np.int64)\n",
    "    z = np.zeros(K, dtype=np.float64)\n",
    "    y = np.zeros(K, dtype=np.float64)\n",
    "    F_targets = np.zeros(K, dtype=np.int64)\n",
    "\n",
    "    for k in range(K):\n",
    "        # sample target value F^(k)\n",
    "        Fk = int(rng.integers(0, F_max + 1))\n",
    "        F_targets[k] = Fk\n",
    "\n",
    "        # sample auxiliary integer costs on ALL augmented arcs\n",
    "        aux_cost = rng.integers(1, tilde_C + 1, size=len(src_aug), dtype=np.int64).astype(np.float64)\n",
    "\n",
    "        # min-cost s-t flow of value Fk via supplies\n",
    "        supply_aug = np.zeros(N_aug, dtype=np.float64)\n",
    "        supply_aug[s] = float(Fk)\n",
    "        supply_aug[t] = -float(Fk)\n",
    "\n",
    "        out_aux = lemon_mcf.solve_mcf(\n",
    "            N_aug,\n",
    "            src_aug,\n",
    "            dst_aug,\n",
    "            aux_cost,\n",
    "            cap_aug,\n",
    "            supply_aug,\n",
    "        )\n",
    "        if out_aux[\"status\"] != 1:\n",
    "            raise RuntimeError(f\"Aux min-cost flow failed at k={k} with status={out_aux['status']}.\")\n",
    "\n",
    "        flow_aug = out_aux[\"flow\"]  # float array per arc\n",
    "\n",
    "        # build balance b^(k) on original nodes\n",
    "        bk = np.zeros(n, dtype=np.int64)\n",
    "        for vv in S_nodes:\n",
    "            bk[int(vv)] = int(round(float(flow_aug[idx_s_to_v[int(vv)]])))\n",
    "        for vv in T_nodes:\n",
    "            bk[int(vv)] = -int(round(float(flow_aug[idx_v_to_t[int(vv)]])))\n",
    "\n",
    "        B[k] = bk\n",
    "\n",
    "        # solve original min-cost flow to get z^(k)\n",
    "        out_orig = lemon_mcf.solve_mcf(\n",
    "            n,\n",
    "            u.astype(np.int64),\n",
    "            v.astype(np.int64),\n",
    "            cost.astype(np.float64),\n",
    "            cap.astype(np.float64),\n",
    "            bk.astype(np.float64),\n",
    "        )\n",
    "        if out_orig[\"status\"] != 1:\n",
    "            raise RuntimeError(f\"Original min-cost flow failed at k={k} with status={out_orig['status']}.\")\n",
    "\n",
    "        zk = float(out_orig[\"total_cost\"])\n",
    "        z[k] = zk\n",
    "\n",
    "        # noisy label\n",
    "        eps = float(rng.normal(0.0, math.sqrt(sigma2)))\n",
    "        y[k] = zk + eps\n",
    "\n",
    "    return {\n",
    "        # graph\n",
    "        \"n\": n,\n",
    "        \"m\": len(u),\n",
    "        \"u\": u.astype(np.int64),\n",
    "        \"v\": v.astype(np.int64),\n",
    "        \"cap\": cap.astype(np.int64),\n",
    "        \"cost\": cost.astype(np.int64),\n",
    "        \"S_nodes\": S_nodes.astype(np.int64),\n",
    "        \"T_nodes\": T_nodes.astype(np.int64),\n",
    "\n",
    "        # dataset (quadratic-like)\n",
    "        \"X_samples\": B,     # balances b^(k)\n",
    "        \"y\": y,             # noisy labels\n",
    "\n",
    "        # extras (often useful)\n",
    "        \"z\": z,             # clean optimal costs\n",
    "        \"F_max\": F_max,\n",
    "        \"F_targets\": F_targets,\n",
    "        \"tilde_C\": tilde_C,\n",
    "        \"sigma2\": float(sigma2),\n",
    "        \"seed\": seed,\n",
    "        \"family\": family,\n",
    "    }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "891009e5-34d9-4d9e-a62e-ef7321e4ddc1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[augmented] F_max = 46135\n",
      "X_samples shape: (100, 200)\n",
      "y shape: (100,)\n",
      "F_max: 46135\n"
     ]
    }
   ],
   "source": [
    "# ---- Example of Usage ---- #\n",
    "\n",
    "data = generate_netgen_mcf_dataset(\n",
    "    n=200,\n",
    "    family=\"sparse\",\n",
    "    K=100,\n",
    "    tilde_C=100,\n",
    "    sigma2=0.25,\n",
    "    seed=0,\n",
    "    verbose=True,\n",
    ")\n",
    "\n",
    "print(\"X_samples shape:\", data[\"X_samples\"].shape)  # (K, n)\n",
    "print(\"y shape:\", data[\"y\"].shape)                  # (K,)\n",
    "print(\"F_max:\", data[\"F_max\"])\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "da97f236-a558-46cc-9803-8ce740c9c2a0",
   "metadata": {},
   "source": [
    "#### Full Dataset Generator "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "06e9b3dc-4978-4afe-a58b-b58f2c5c505a",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "e1cdd06e-81d1-4f27-a39b-81d540a21241",
   "metadata": {},
   "source": [
    "# Models Definition"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b131da93-331f-46ef-91e4-1ced802f0e27",
   "metadata": {},
   "source": [
    "### DFN"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0d617ce9-24ab-4015-9fd0-6bc983d61fab",
   "metadata": {},
   "source": [
    "#### Helper Functions and Modules"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "07be67a0-5e8e-4e5f-9e2d-5acda3ab83be",
   "metadata": {},
   "source": [
    "STE op:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "0d7ccba7-9593-421d-a6ef-778b9dd9c437",
   "metadata": {},
   "outputs": [],
   "source": [
    "class RoundSTEFunc(torch.autograd.Function):\n",
    "    \"\"\"Straight-through rounding (kept for reference; no longer used by default).\"\"\"\n",
    "\n",
    "    @staticmethod\n",
    "    def forward(ctx, x: torch.Tensor) -> torch.Tensor:\n",
    "        return torch.round(x)\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_out: torch.Tensor) -> torch.Tensor:\n",
    "        return grad_out\n",
    "\n",
    "\n",
    "def round_ste(x: torch.Tensor) -> torch.Tensor:\n",
    "    return RoundSTEFunc.apply(x)\n",
    "\n",
    "\n",
    "def make_int_param(raw: torch.Tensor, nonneg: bool = False, eps: float = 0.0) -> torch.Tensor:\n",
    "    \"\"\"Turn a raw parameter into the value used in the forward pass.\n",
    "\n",
    "    NOTE: Despite the name, this now returns **continuous** parameters.\n",
    "      - If nonneg=True: uses softplus to enforce non-negativity.\n",
    "      - If nonneg=False: returns the raw value.\n",
    "\n",
    "    eps can be used to keep parameters strictly positive if desired.\n",
    "    \"\"\"\n",
    "    return (F.softplus(raw) + eps) if nonneg else raw\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ab29e386-ed1c-4bff-aae5-830ec56acfa2",
   "metadata": {},
   "source": [
    "Min Cost Flow op:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "980806c5-11ab-4527-b164-a88981fca9ff",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MinCostFlowValue(torch.autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(ctx, n_nodes: int, cost: torch.Tensor, cap: torch.Tensor, b: torch.Tensor):\n",
    "        \"\"\"Solve a continuous min-cost-flow LP with Gurobi.\n",
    "\n",
    "        n_nodes: int\n",
    "        cost: (2*n*(n-1),) float tensor (nonnegative expected)\n",
    "        cap:  (2*n*(n-1),) float tensor (nonnegative expected)\n",
    "        b:    (n,) float tensor (net supplies; must sum to 0 for feasibility)\n",
    "\n",
    "        Returns: scalar tensor (optimal total cost)\n",
    "\n",
    "        Arc order: for i=0..n-1, j runs over [0..n-1] excluding i, in increasing j.\n",
    "        Copy of arcs repeats the same order.\n",
    "        \"\"\"\n",
    "        n = int(n_nodes)\n",
    "        m = n * (n - 1)\n",
    "        if cost.dim() != 1 or cost.numel() != 2 * m:\n",
    "            raise ValueError(f\"cost must be (2*n*(n-1),) = ({2*m},), got {tuple(cost.shape)}\")\n",
    "        if cap.dim() != 1 or cap.numel() != 2 * m:\n",
    "            raise ValueError(f\"cap must be (2*n*(n-1),) = ({2*m},), got {tuple(cap.shape)}\")\n",
    "        if b.dim() != 1 or b.numel() != n:\n",
    "            raise ValueError(f\"b must be (n,) = ({n},), got {tuple(b.shape)}\")\n",
    "\n",
    "        # build all directed arcs i->j with i!=j (no self loops), then duplicate (copy0, copy1)\n",
    "        src_base = np.repeat(np.arange(n, dtype=np.int64), n - 1)\n",
    "        dst_base = np.concatenate([np.delete(np.arange(n, dtype=np.int64), i) for i in range(n)])\n",
    "        src = np.concatenate([src_base, src_base])\n",
    "        dst = np.concatenate([dst_base, dst_base])\n",
    "\n",
    "        cost_np = cost.detach().cpu().contiguous().numpy().astype(np.float64, copy=False)\n",
    "        cap_np  = cap.detach().cpu().contiguous().numpy().astype(np.float64, copy=False)\n",
    "        b_np    = b.detach().cpu().contiguous().numpy().astype(np.float64, copy=False)\n",
    "\n",
    "        # Floating-point safety: for pure circulation constraints, feasibility requires sum(b)=0.\n",
    "        # Even if the caller enforces this in float32, rounding can leave a nonzero residual.\n",
    "        # DFN uses the last node as an auto-balance node; re-enforce it here in float64.\n",
    "        b_np = b_np.copy()\n",
    "        b_np[-1] = -b_np[:-1].sum()\n",
    "\n",
    "\n",
    "        try:\n",
    "            import gurobipy as gp\n",
    "            from gurobipy import GRB\n",
    "        except Exception as e:\n",
    "            raise RuntimeError(\"gurobipy is required for continuous LP solving. Make sure Gurobi is installed and licensed.\") from e\n",
    "\n",
    "        model = gp.Model(\"mcf_lp\")\n",
    "        model.Params.OutputFlag = 0\n",
    "        model.Params.LogToConsole = 0\n",
    "\n",
    "        # Variables: flow on each arc (continuous)\n",
    "        f = model.addVars(2 * m, lb=0.0, ub=GRB.INFINITY, vtype=GRB.CONTINUOUS, name=\"f\")\n",
    "\n",
    "        # Capacity constraints f_k <= cap_k (explicit so we can read duals)\n",
    "        cap_constr = [model.addConstr(f[k] <= float(cap_np[k]), name=f\"cap[{k}]\") for k in range(2 * m)]\n",
    "\n",
    "        # Flow conservation: sum_out f - sum_in f == b_i\n",
    "        out_arcs = [[] for _ in range(n)]\n",
    "        in_arcs  = [[] for _ in range(n)]\n",
    "        for k in range(2 * m):\n",
    "            out_arcs[int(src[k])].append(k)\n",
    "            in_arcs[int(dst[k])].append(k)\n",
    "\n",
    "        bal_constr = []\n",
    "        for i in range(n):\n",
    "            expr = gp.quicksum(f[k] for k in out_arcs[i]) - gp.quicksum(f[k] for k in in_arcs[i])\n",
    "            bal_constr.append(model.addConstr(expr == float(b_np[i]), name=f\"bal[{i}]\") )\n",
    "\n",
    "        # Objective\n",
    "        model.setObjective(gp.quicksum(float(cost_np[k]) * f[k] for k in range(2 * m)), GRB.MINIMIZE)\n",
    "\n",
    "        model.optimize()\n",
    "        if model.Status != GRB.OPTIMAL:\n",
    "            raise RuntimeError(f\"Gurobi did not find an optimal solution (status={model.Status}).\")\n",
    "\n",
    "        flow = np.array([f[k].X for k in range(2 * m)], dtype=np.float64)\n",
    "\n",
    "        # Duals:\n",
    "        # - bal_constr[i].Pi is the dual for equality (free)\n",
    "        # - cap_constr[k].Pi is the dual for <= (should be <= 0 in minimization)\n",
    "        bal_pi = np.array([c.Pi for c in bal_constr], dtype=np.float64)          # (n,)\n",
    "        cap_pi = np.array([c.Pi for c in cap_constr], dtype=np.float64)          # (2*m,)\n",
    "\n",
    "        # Match the old convention where we stored \"potential\" and used grad_b = mean(pot) - pot.\n",
    "        # Setting pot = -bal_pi makes grad_b become a centered version of bal_pi.\n",
    "        pot = -bal_pi\n",
    "\n",
    "        ctx.n = n\n",
    "        ctx.flow = flow\n",
    "        ctx.pot = pot\n",
    "        ctx.cap_pi = cap_pi\n",
    "\n",
    "        return cost.new_tensor(float(model.ObjVal))\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_out: torch.Tensor):\n",
    "        # grads for (n_nodes, cost, cap, b) => (None, grad_cost, grad_cap, grad_b)\n",
    "        device, dtype = grad_out.device, grad_out.dtype\n",
    "\n",
    "        flow   = torch.as_tensor(ctx.flow,   device=device, dtype=dtype)\n",
    "        pot    = torch.as_tensor(ctx.pot,    device=device, dtype=dtype)\n",
    "        cap_pi = torch.as_tensor(ctx.cap_pi, device=device, dtype=dtype)\n",
    "\n",
    "        grad_cost = flow\n",
    "        grad_cap  = cap_pi\n",
    "        grad_b    = pot.mean() - pot\n",
    "\n",
    "        grad_cost = grad_cost * grad_out\n",
    "        grad_cap  = grad_cap  * grad_out\n",
    "        grad_b    = grad_b    * grad_out\n",
    "\n",
    "        return None, grad_cost, grad_cap, grad_b\n",
    "\n",
    "\n",
    "def min_cost_flow_value(n_nodes: int, cost: torch.Tensor, cap: torch.Tensor, b: torch.Tensor):\n",
    "    return MinCostFlowValue.apply(n_nodes, cost, cap, b)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b8c13547-8c12-4413-a9d3-368d015cd778",
   "metadata": {},
   "source": [
    "#### The DFN Module"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "2bd3916f-ee07-4ef5-8b4e-05ba9438ee93",
   "metadata": {},
   "outputs": [],
   "source": [
    "class DFN(nn.Module):\n",
    "    def __init__(\n",
    "        self,\n",
    "        input_dim: int,\n",
    "        num_nodes: int,\n",
    "        n_var: int | None = None,\n",
    "        n_cst: int | None = None,\n",
    "        big_cost: float = 1e3,\n",
    "        big_cap: float = 1e9,\n",
    "    ):\n",
    "        super().__init__()\n",
    "        n = int(num_nodes)\n",
    "        n_aug = n + 1                      # +1 auto-balance node\n",
    "\n",
    "        k = int(math.sqrt(n)) if n_var is None else int(n_var)\n",
    "        l = int(math.sqrt(n)) if n_cst is None else int(n_cst)\n",
    "        if k + l > n:\n",
    "            raise ValueError(\"Need n_var + n_cst <= num_nodes.\")\n",
    "\n",
    "        self.n, self.n_aug = n, n_aug\n",
    "        self.n_var, self.n_cst = k, l\n",
    "        self.big_cost, self.big_cap = float(big_cost), float(big_cap)\n",
    "\n",
    "        # complete digraph on (n+1) nodes without self-loops\n",
    "        self.m0 = n_aug * (n_aug - 1)\n",
    "\n",
    "        # learn copy-0 (copy-1 is fixed to big bounds)\n",
    "        # NOTE: parameters are now continuous; costs/caps are made nonnegative via softplus.\n",
    "        self.cap_raw  = nn.Parameter(torch.randn(self.m0) + 1)\n",
    "        self.cost_raw = nn.Parameter(torch.randn(self.m0) + 1)\n",
    "\n",
    "        self.A_raw = nn.Parameter(torch.randn(self.n_var, input_dim) + 1)\n",
    "        self.b_raw = nn.Parameter(torch.randn(self.n_cst) + 1)\n",
    "\n",
    "        # alpha removed; keep beta as an intercept\n",
    "        self.beta  = nn.Parameter(torch.tensor(0.0))\n",
    "\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        # continuous params\n",
    "        cap0  = make_int_param(self.cap_raw,  nonneg=True)   # (m0,)\n",
    "        cost0 = make_int_param(self.cost_raw, nonneg=True)   # (m0,)\n",
    "        cap1  = cap0.new_full(cap0.shape, self.big_cap)\n",
    "        cost1 = cost0.new_full(cost0.shape, self.big_cost)\n",
    "        cap   = torch.cat([cap0,  cap1],  dim=0)             # (2*m0,)\n",
    "        cost  = torch.cat([cost0, cost1], dim=0)             # (2*m0,)\n",
    "\n",
    "        A = make_int_param(self.A_raw, nonneg=False)         # (n_var, input_dim)\n",
    "        b = make_int_param(self.b_raw, nonneg=False)         # (n_cst,)\n",
    "\n",
    "        def solve_one(x1: torch.Tensor) -> torch.Tensor:\n",
    "            # balances on first n nodes: [Ax, b, 0...], then extra node balances sum to 0\n",
    "            ax = A @ x1                                      # (n_var,)\n",
    "            zeros = x1.new_zeros(self.n - self.n_var - self.n_cst)\n",
    "            b_core = torch.cat([ax, b, zeros], dim=0)         # (n,)\n",
    "            b_full = torch.cat([b_core, (-b_core.sum()).view(1)], dim=0)  # (n+1,)\n",
    "            obj = min_cost_flow_value(self.n_aug, cost, cap, b_full)\n",
    "            return obj + self.beta\n",
    "\n",
    "        if x.dim() == 1:\n",
    "            return solve_one(x)\n",
    "\n",
    "        return torch.stack([solve_one(xi) for xi in x], dim=0)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fb0c0c1d-e1d6-4259-ad3b-55cd5b258432",
   "metadata": {},
   "source": [
    "### Gradient Boosted Trees"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "72c356e0-d483-442b-a32b-49f0d7eab840",
   "metadata": {},
   "source": [
    "### MLP"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "df43dee6-8978-44d8-a576-cceea5fa46a1",
   "metadata": {},
   "source": [
    "### ICNN"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e74105b4-7d14-4b4d-9261-160fc4b44827",
   "metadata": {},
   "source": [
    "### Max-Affine"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e8a3d912-fa0a-4acc-a12f-10a089f213de",
   "metadata": {},
   "source": [
    "# Training on Data"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "609e32f6-7edd-40dd-9764-e06a592d0e73",
   "metadata": {},
   "source": [
    "# Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "04f08bbc-0c5b-4556-9837-1437d56239c3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "88a70431-932d-40db-8ad1-d21fd24d60ec",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "e6c539a0-3a00-4fa0-ab8d-6c8b5ec310f0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[data] X integer? True  X min/max: -50/49\n",
      "[data] Q eig min/max: 0.273/35.937  y mean/std: -0.000/1.000\n",
      "\n",
      "[DFN] cap0 min/mean/max: 0.857/1.32/1.91\n",
      "[DFN] cost0 min/mean/max: 0.379/0.695/1.09\n",
      "[DFN] A   min/mean/max: -0.509/-0.00715/0.622\n",
      "[DFN] b   min/mean/max: -0.45/-0.0475/0.466\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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WgRZXbzXrFkT3K6++mq3ILK+ZrgWAAAkqoTM8In66fXs2dMFbArmJk2a5FqyeH31NNyqLt2vvvqqu63948ePd4+7+eabXRGH5ut5HbfV10ZNE0MVK1bM/Ru+HwAAIJEkbMDXrVs32717t6uqVeNlBWVz5851S6KI9oX25FOzQt1/xx132PPPP+8aL6tz96muywcAAJAoEnqljXjDShsAgHjAShtZa+rUqW5xiL1796bL82XrlTYAAIC/9O7d23XO0Kb59SrEbNmypb388svBQkqP1tr1jvW2ChUqJLt/+fLlSR6nQKx58+aW3RDwAQCAuNGmTRs3LUudMz788ENXPHn77bdbhw4dXO/cUN60Lm/T0qmhlA275557MvkTxCcCPgAAsgPN4Dp6MPO3NM4cU7sztVIrX768NWjQwO6991577733XPCnodJQRYoUccd6m1bKCtW/f3+X4Zs7d25Mr60sorKEWo411FdffeWyhT///HNwedd69epZoUKFXBeP2267zQ4cOGDxLGGLNgAAQBocO2Q2qlzmn7J7t5rlLXRaT3HFFVdY/fr1bdasWdavX7+YH6dhXXXpGDZsmMsc5syZcp5L93fv3t2mT58e7Pohb7zxhusIUq1ateBxKvzU82tunQK+u+++2yZMmGDxigwfAACIe+qXq2HeUBquLVy4cHBTEBbu/vvvd0GZgrhYXH/99W6p1V9++SWY9ZsxY4ZbjSt0HqCGmlVEoWB05MiR9tZbb1k8I8MHAEB2kKfg/2XbsuJ100Gk5VPvuusuV+jhibQ0qoZ577zzTnvwwQddS7fUXHDBBS64VJ/eoUOH2qJFi9zKXNdee23wmIULF9qoUaNs/fr1rkpWcwtVSXvw4EE3zBuPyPABAJAdKFjS0GpmbymscZ8WGzZscBm1UArwqlevHty8BRPCDRkyxA4fPhzzkKuyfBrGFf3bunXrYDCpzF+7du1c/993333XVq1a5fr7yrFjxyxeEfABAIC49sknn7jlT091sYTChQvbAw88YI8++qjLyKXmuuuuc6+nYO6dd95xAaBn5cqVLqP31FNP2SWXXGLnnHOObd2aBZnTNCLgAwAAcePIkSO2fft2tzyqqmM1dNqpUyfXluXGG2885ee95ZZbXINib0nVlCiT2KRJE+vbt68L7vT6nrPPPtvte+6551zV7muvvZasqjce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fK5cubIbbv3999/t1VdfTdKWRS1Z1JpFBRqq0g1ty6LijMsuu8weffRR69Spk7333nt2//3305YFAABk77V0RQUUWkdXLU80303tURQwadg1o6iFipZ0GzFihG3bts0Fc3PnznXBnmjfr7/+GjxeDZp1/x133OHaxZQrV86effbZYLAnyuTNmDHDBXkPPPCAnX322S6wjKUHHwAAgG8zfBs3bnRtWJRNq1mzpity0Pw9Vax+8MEHLmjKLujDBwAAfNmHb9CgQS6oU4XqV199ZatXr3aZNWXUdB8AAACy1mkP6Wp92uXLl7s2LB5VvT722GOuchcAAABZ67QzfGpLsn///mT7Dxw44KppAQAAkOABn1bWuOWWW+yLL75w8/e0KeOnCtirr746fd4lAAAAsi7gU6Wr5vCpNUr+/PndpmrX6tWr27hx40736QEAAJDVc/iKFSvm+tWpWldtWZTh09q6CvgAAACQoAGfllJLyaeffhr8euzYsafyEgAAAMjKgE+tV2KhJswAAABIwIBv4cKF6f9OAAAAEJ9FGwAAAIhvBHwAAAA+R8AHAADgcwR8AAAAPkfABwAA4HMEfAAAAD5HwAcAAOBzBHwAAAA+R8AHAADgcwR8AAAAPkfABwAA4HMEfAAAAD5HwAcAAOBzBHwAAAA+R8AHAADgcwR8AAAAPkfABwAA4HMEfAAAAD6XkAHfnj17rGfPnnbGGWe4TV/v3bs3xccEAgEbPny4lStXzgoUKGDNmze3b7/9Nnj/n3/+af/617+sZs2aVrBgQatUqZINGjTI/vrrr0z4RAAAABknIQO+6667ztasWWPz5s1zm75W0JeSMWPG2NixY238+PH25ZdfWpkyZaxly5a2f/9+d//WrVvd9uSTT9q6dets6tSp7rn79u2bSZ8KAAAgY+QIKPWVQDZs2GB16tSx5cuXW6NGjdw+fd24cWP77rvvXIYunD6iMnuDBw+2e+65x+07cuSIlS5d2h5//HHr379/xNd6++237YYbbrCDBw9a7ty5U31v+/btcxlHZQWLFi162p8VAAAgPWKPhMvwLVu2zH0wL9iTSy65xO1bunRpxMds2rTJtm/fbq1atQruy5cvn11++eVRHyPeyYsW7Clo1IkO3QAAAOJNwgV8CtxKlSqVbL/26b5ojxFl9ELpdrTH7N6920aOHBk1+yejR48O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kT777DPr2LGjlStXzgWXc+bMiXjchAkTrGrVqpY/f3678MILbfHixTG/xo8//miFCxe2q6++2ho0aGCjRo1Kx08AAACQ+RKqrFZz6erXr299+vSxrl27Rjxm5syZNnjwYBf0NW3a1F588UVr27atrV+/3mUpRUFgpJ6B8+fPt2PHjrkAcc2aNVaqVClr06aNXXTRRdayZcsM/3wAAAAZIU1DuqtXr7YzzzzT3R4/frzdeOONKY4XpzcFbtpSMnbsWOvbt6/169fP3R43bpx99NFHNnHiRBs9erTbt2rVqqiPr1ChggvwKlas6G63a9fOBX+RAj4FjaGBo8bRAQAAEnZI97fffrMTJ04Eb9977722a9cuiydHjx51wVyrVq2S7NftpUuXxvQcCvb++OMP27Nnj508edINI9euXTvisQogNVHS27wgEQAAICEDvnAxFvdmKgWgCkpLly6dZL9ub9++PabnUDWu5u1ddtlldt5551mNGjWsQ4cOEY8dNmyYq4rxti1btqTL5wAAAPDVHD6t0vHwww+neMyXX35pDRs2jPk5VdARHpyG7zvdoWPJly+f2wAAAHwT8E2ZMsVVsHrr6E6dOtVKliyZ5Ji0tmUZOHCgde/ePcVjUur7F0rvJVeuXMmyeTt27EiW9QMAAMgu0lS0MXny5ODtMmXK2GuvvXbaffgUpIUHjacqb968rgJ3wYIFbkUQj2536tQpXV4DAADAtwGf+vBltQMHDtjGjRuDt9X7TxW0ao7stVzR2r49e/Z0Q8CNGze2SZMm2a+//moDBgzIwncOAACQjefwpcXKlSvdGrceBXfSq1cvN7ws3bp1s927d9uIESNs27ZtVrduXZs7d65Vrlw5y943AABAQqylK2pTosBq1qxZLuOnIVytaPGPf/zDZdXSUhjhR6ylCwAA4jH2iLkti+JCLTemhsa///671atXz84991z75ZdfrHfv3knmzAEAACABh3SV2VMT4o8//jjJsKp88skn1rlzZ3v11Vfd6hsAAACIHzFn+N588023ukZ4sCdXXHGFDR061KZPn57e7w8AAACZFfCtXbvW2rRpE/V+NSr++uuvT/f9AAAAIKsCvj///DPF5sW6T+vP+oGWVzv//PPdpjmLAAAA2WIOn9aoVSAUjVa40OobflCsWDHX3w8AACBbBXyq0lU1brS1Y48cOZKe7wsAAACZPaSr5salSpVyvV4ibbovoyt0VSXcsWNHK1eunOv5N2fOnIjHTZgwwfUHzJ8/v1tqbfHixWnuaaPHXXrppbZo0aJ0evcAAABxnuF75ZVXLKsdPHjQ6tevb3369LGuXbtGPGbmzJk2ePBgF/Q1bdrUXnzxRVdQsn79+uDyawrmImUk58+f74JJNZXWv9988421b9/e1q1bl2IzQwAAAN+stBFPlOGbPXu26/8XqlGjRtagQQObOHFicF/t2rXdcaNHj07z6yhYHDlypFubN5yCxtDAUZnBihUrptrtGgAAIC5X2kgER48etVWrVlmrVq2S7NftpUuXxvQcqjT2grjffvvNZQarVasW8VgFkKHD2gr2AAAA4o2vAr5du3a5auLw9jG6vX379pieY8OGDS6bp6HjDh062DPPPGMlSpSIeOywYcNcRO1tW7ZsSZfPAQAAkCVz+DLK8OHD7eGHH07xmC+//DLikGpKw72hNGodvi+aJk2auDl7sVDFcrSqZQAAgHiR5QHfwIEDrXv37ikeU6VKlZieq2TJkq4fYHg2b8eOHSk2jQYAAPCzLA/4FKRpSw958+Z1FbgLFiywLl26BPfrdqdOndLlNQAAABJNlgd8aXHgwAHbuHFj8PamTZvcihiaY+e1XBkyZIj17NnTDQE3btzYJk2aZL/++qsNGDAgC985AABA1kmogG/lypXWokWL4G0Fd15T6KlTp7qvu3XrZrt377YRI0bYtm3brG7dujZ37lyrXLlylr1vAACArJSwffgSuRcOAABAesiWffgAAACQHAEfAACAzxHwAQAA+BwBHwAAgM8R8EVx6NAhV9l75513Zu4VAQAASGcEfFE8+uij1qhRo/Q+3wAAAJmOgC+CH3/80b777jtr165d5l8RAACA7BzwffbZZ9axY0crV66c5ciRw+bMmRPxuAkTJljVqlUtf/78bqm1xYsXp+l1NIw7evTodHrXAAAAWSuhAr6DBw9a/fr1bfz48VGPmTlzpg0ePNjuu+8+W716tTVr1szatm3rllfzKAjUChzh29atW+29996zc845x20AAAB+kLArbSjDN3v2bOvcuXOS/Zp316BBA5s4cWJwX+3atd1xsWTthg0bZq+//rrlypXLrd177Ngx+/e//20PPvhgsmOPHDnittBu1xUrVmSlDQAAkCmy5UobR48e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",
      "text/plain": [
       "<Figure size 700x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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seBCXKltcsV2SdE2WYO2yTqj3axtomDKRFevXpXOnTvLwIEDxdnNnTvX7DC++uqrsn79emnZsqW0adNGDh48KM7s4sWLUr9+ffnqq6/E1axcuVIGDx4sa9askWXLlsn169eldevW5j05u7Jly8r7778vISEh5nLvvfdKu3btZMuWLeJKgoODZdKkSab4cyW1a9eWY8eOxV42bdpk9SLBhbhK2+Sq7ZKibbIGbZO1artT26Sx4shcgYGBNm9vb6de7U2aNLENGDAg3m01atSwjRgxwuYq9Ou9aNEim6sKDw8372HlypU2V1SoUCHbt99+a3MV58+ft1WtWtW2bNky2913320bOnSozRWMGjXKVr9+fasXA27A2dsmd2iXFG2TtWibMscoN2ubGGFCor2NoaGhZnQjLr2+evVq1lgmiYiIMP8WLlzYpdZ5dHS0zJkzx/So6tQ8V6Gje4888ojcf//94mp27dplpqHqVKVu3brJ3r17rV4kIF3RLjkP2qbMRdvkHLJZvQBwPqdOnTI7vSVKlIh3u14/fvy4ZcvlSbQTctiwYXLnnXdKnTp1xBXoULsWSFeuXJF8+fLJokWLpFatWuIKtMBbt26dmZLnapo2bSrTpk2TatWqyYkTJ+Sdd94xx6PodMgiRYpYvXhAuqBdcg60TZmLtsl5MMKUAQe1JbzoMR2uSJc94YYy4W3IGEOGDJGwsDCZPXu2y6zi6tWry4YNG8wxWHosRO/evWXr1q3i7A4dOiRDhw6VGTNmmAPJXY0ew9GpUyepW7euGR37+eefze1Tp061etFgIXdtm2iXrEXblHlom5wLI0zpsPHQKTC3UqFCBXElRYsWFS8vr5tGk8LDw28adUL6e+aZZ2Tx4sUmVUkPWHUVOXLkkCpVqpj/9/X1NaM1X3zxhUycOFGcmU4/1e+2Jm456Airrn8ND4mKijK/B1eRN29eUzzpND14Lndrm2iXrEfblLlom5wLBVM6bMT14k50x1d3HjWprUOHDrG363VNPkPG0BE8bZB0KtuKFSvM8Siu/n602HB29913303JPQEBAVKjRg15+eWXXapYUrrOt23bZhLE4LncrW2iXbIObZM1aJucCwVTJtLo0zNnzph/tQdbpy8p7ZXXYz6ciR4/06tXLzNSoMelaNSyLveAAQPEmV24cEF2794de33fvn1mPWtwQrly5cTZD+ycNWuW/PDDD+ZcTI4RPm9vb8mdO7c4s5EjR5qpYT4+PnL+/Hkz71qLviVLloiz03Wd8DgxHaXR439c4fix4cOHm3OP6fdbR8r0GKbIyEgzJRJwp7bJVdslRdtkDdom6wx3t7bJ6pg+T9K7d28TJ5rwsnz5cpszGjdunK18+fK2HDly2Bo1auQS8da6LhNbx7runV1iy60Xjfp1dn369In9rhQrVsx233332ZYuXWpzVa4UK961a1dbqVKlbNmzZ7eVLl3a1rFjR9uWLVusXiy4EFdqm1yxXVK0TdagbbJOVzdrm7Lof6wu2gAAAADAGZGSBwAAAABJoGACAAAAgCRQMAEAAABAEiiYAAAAACAJFEwAAAAAkAQKJgAAAABIAgUTAAAAACSBggkAAAAAkkDBBAAAAABJoGACnNDo0aOlQYMGVi8GAAAG7RI8GQUT4MKuXbtm9SIAABCLdgnuiIIJyCDTpk2TIkWKSFRUVLzbO3XqJE8++WSSjwsKCpI333xTNm7cKFmyZDEXvU3p/3/99dfSrl07yZs3r7zzzjvmbwULFoz3HN9//725b1w//vijNG7cWHLlyiWVKlUyr3H9+vV0fc8AAOdFuwSkDQUTkEE6d+4s0dHRsnjx4tjbTp06JT/99JMEBAQk+biuXbvKCy+8ILVr15Zjx46Zi97mMGrUKFMwbdq0Sfr06ZOiZfntt9+kZ8+e8uyzz8rWrVtl4sSJptB69913b/NdAgBcBe0SkDYUTEAGyZ07tzzxxBMSGBgYe9vMmTOlbNmy0qpVq1s+Ll++fJItWzYpWbKkuehtDvqcWijpKFH58uVTtCxaGI0YMUJ69+5tHvfAAw/I22+/bQonAIBnoF0C0iZbGh8HIAWeeuop8fPzkyNHjkiZMmVM8eTv73/TdLnU8PX1TfVjQkNDJTg4ON6Iko5+XblyRS5duiR58uRJ8/IAAFwH7RKQehRMQAZq2LCh1K9f38wbf/DBB800Oj2W6HbosUtxZc2aVWw22y0Pur1x44Y5Zqljx443PZ8e0wQA8Ay0S0DqUTABGaxfv37y2WefmVGm+++/X3x8fJJ9TI4cOcwIUEoUK1ZMzp8/LxcvXowtpjZs2BDvPo0aNZIdO3ZIlSpV0vguAADugnYJSB2OYQIyWI8ePUyx9M0336Q4pKFChQqyb98+U/hoUETCpL24mjZtaqbUjRw5Unbv3i2zZs2KTdVzeOONN8wol55HY8uWLbJt2zaZO3euvPbaa7f9/gAAroV2CUgdCiYggxUoUMBEiWuQQ/v27VP0GL3/Qw89JPfcc48ZQZo9e3aS9y1cuLDMmDFDfvnlF6lbt665rxZGcel0QE3nW7ZsmTmmqlmzZvLpp5+mODQCAOA+aJeA1MliS3jwA4B0p6l0NWvWlLFjx7J2AQCWo10CUo6CCchAZ86ckaVLl5rpD3r+o+rVq7O+AQCWoV0CUo/QByADadjC2bNn5YMPPohXLOlJaQ8cOJDoY/TcSFpgAQBAuwRYjxEmwAJaLCWM/nYoUaKE5M+fP9OXCQDguWiXgKRRMAEAAABAEkjJAwAAAIAkUDABAAAAQBIomAAAAAAgCRRMAAAAAJAECiYAAAAASAIFEwAAAAAkgYIJAAAAACRx/wPSuUFJfDBgMwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "ename": "RuntimeError",
     "evalue": "Gurobi did not find an optimal solution (status=3).",
     "output_type": "error",
     "traceback": [
      "\u001b[31m---------------------------------------------------------------------------\u001b[39m",
      "\u001b[31mRuntimeError\u001b[39m                              Traceback (most recent call last)",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[17]\u001b[39m\u001b[32m, line 275\u001b[39m\n\u001b[32m    273\u001b[39m \u001b[38;5;66;03m# ---- DFN step ----\u001b[39;00m\n\u001b[32m    274\u001b[39m dfn.train()\n\u001b[32m--> \u001b[39m\u001b[32m275\u001b[39m pred_dfn = \u001b[43mdfn\u001b[49m\u001b[43m(\u001b[49m\u001b[43mXb\u001b[49m\u001b[43m)\u001b[49m\n\u001b[32m    276\u001b[39m loss_dfn = F.mse_loss(pred_dfn, yb)\n\u001b[32m    277\u001b[39m opt_dfn.zero_grad(set_to_none=\u001b[38;5;28;01mTrue\u001b[39;00m)\n",
      "\u001b[36mFile \u001b[39m\u001b[32m/opt/anaconda3/envs/dfn/lib/python3.11/site-packages/torch/nn/modules/module.py:1736\u001b[39m, in \u001b[36mModule._wrapped_call_impl\u001b[39m\u001b[34m(self, *args, **kwargs)\u001b[39m\n\u001b[32m   1734\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m._compiled_call_impl(*args, **kwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[32m   1735\u001b[39m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[32m-> \u001b[39m\u001b[32m1736\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[43m.\u001b[49m\u001b[43m_call_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[43m*\u001b[49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m*\u001b[49m\u001b[43m*\u001b[49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[36mFile \u001b[39m\u001b[32m/opt/anaconda3/envs/dfn/lib/python3.11/site-packages/torch/nn/modules/module.py:1747\u001b[39m, in \u001b[36mModule._call_impl\u001b[39m\u001b[34m(self, *args, **kwargs)\u001b[39m\n\u001b[32m   1742\u001b[39m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[32m   1743\u001b[39m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[32m   1744\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m._backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m._backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m._forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m._forward_pre_hooks\n\u001b[32m   1745\u001b[39m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[32m   1746\u001b[39m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[32m-> \u001b[39m\u001b[32m1747\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[43m*\u001b[49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m*\u001b[49m\u001b[43m*\u001b[49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[32m   1749\u001b[39m result = \u001b[38;5;28;01mNone\u001b[39;00m\n\u001b[32m   1750\u001b[39m called_always_called_hooks = \u001b[38;5;28mset\u001b[39m()\n",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[13]\u001b[39m\u001b[32m, line 62\u001b[39m, in \u001b[36mDFN.forward\u001b[39m\u001b[34m(self, x)\u001b[39m\n\u001b[32m     59\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m x.dim() == \u001b[32m1\u001b[39m:\n\u001b[32m     60\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m solve_one(x)\n\u001b[32m---> \u001b[39m\u001b[32m62\u001b[39m \u001b[38;5;28;01mreturn\u001b[39;00m torch.stack(\u001b[43m[\u001b[49m\u001b[43msolve_one\u001b[49m\u001b[43m(\u001b[49m\u001b[43mxi\u001b[49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mxi\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mx\u001b[49m\u001b[43m]\u001b[49m, dim=\u001b[32m0\u001b[39m)\n",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[13]\u001b[39m\u001b[32m, line 62\u001b[39m, in \u001b[36m<listcomp>\u001b[39m\u001b[34m(.0)\u001b[39m\n\u001b[32m     59\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m x.dim() == \u001b[32m1\u001b[39m:\n\u001b[32m     60\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m solve_one(x)\n\u001b[32m---> \u001b[39m\u001b[32m62\u001b[39m \u001b[38;5;28;01mreturn\u001b[39;00m torch.stack([\u001b[43msolve_one\u001b[49m\u001b[43m(\u001b[49m\u001b[43mxi\u001b[49m\u001b[43m)\u001b[49m \u001b[38;5;28;01mfor\u001b[39;00m xi \u001b[38;5;129;01min\u001b[39;00m x], dim=\u001b[32m0\u001b[39m)\n",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[13]\u001b[39m\u001b[32m, line 56\u001b[39m, in \u001b[36mDFN.forward.<locals>.solve_one\u001b[39m\u001b[34m(x1)\u001b[39m\n\u001b[32m     54\u001b[39m b_core = torch.cat([ax, b, zeros], dim=\u001b[32m0\u001b[39m)         \u001b[38;5;66;03m# (n,)\u001b[39;00m\n\u001b[32m     55\u001b[39m b_full = torch.cat([b_core, (-b_core.sum()).view(\u001b[32m1\u001b[39m)], dim=\u001b[32m0\u001b[39m)  \u001b[38;5;66;03m# (n+1,)\u001b[39;00m\n\u001b[32m---> \u001b[39m\u001b[32m56\u001b[39m obj = \u001b[43mmin_cost_flow_value\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m.\u001b[49m\u001b[43mn_aug\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcost\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcap\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mb_full\u001b[49m\u001b[43m)\u001b[49m\n\u001b[32m     57\u001b[39m \u001b[38;5;28;01mreturn\u001b[39;00m obj + \u001b[38;5;28mself\u001b[39m.beta\n",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[12]\u001b[39m\u001b[32m, line 110\u001b[39m, in \u001b[36mmin_cost_flow_value\u001b[39m\u001b[34m(n_nodes, cost, cap, b)\u001b[39m\n\u001b[32m    109\u001b[39m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[34mmin_cost_flow_value\u001b[39m(n_nodes: \u001b[38;5;28mint\u001b[39m, cost: torch.Tensor, cap: torch.Tensor, b: torch.Tensor):\n\u001b[32m--> \u001b[39m\u001b[32m110\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mMinCostFlowValue\u001b[49m\u001b[43m.\u001b[49m\u001b[43mapply\u001b[49m\u001b[43m(\u001b[49m\u001b[43mn_nodes\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcost\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcap\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mb\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[36mFile \u001b[39m\u001b[32m/opt/anaconda3/envs/dfn/lib/python3.11/site-packages/torch/autograd/function.py:575\u001b[39m, in \u001b[36mFunction.apply\u001b[39m\u001b[34m(cls, *args, **kwargs)\u001b[39m\n\u001b[32m    572\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m torch._C._are_functorch_transforms_active():\n\u001b[32m    573\u001b[39m     \u001b[38;5;66;03m# See NOTE: [functorch vjp and autograd interaction]\u001b[39;00m\n\u001b[32m    574\u001b[39m     args = _functorch.utils.unwrap_dead_wrappers(args)\n\u001b[32m--> \u001b[39m\u001b[32m575\u001b[39m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43msuper\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m.\u001b[49m\u001b[43mapply\u001b[49m\u001b[43m(\u001b[49m\u001b[43m*\u001b[49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m*\u001b[49m\u001b[43m*\u001b[49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[32m    577\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m is_setup_ctx_defined:\n\u001b[32m    578\u001b[39m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mRuntimeError\u001b[39;00m(\n\u001b[32m    579\u001b[39m         \u001b[33m\"\u001b[39m\u001b[33mIn order to use an autograd.Function with functorch transforms \u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m    580\u001b[39m         \u001b[33m\"\u001b[39m\u001b[33m(vmap, grad, jvp, jacrev, ...), it must override the setup_context \u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m    581\u001b[39m         \u001b[33m\"\u001b[39m\u001b[33mstaticmethod. For more details, please see \u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m    582\u001b[39m         \u001b[33m\"\u001b[39m\u001b[33mhttps://pytorch.org/docs/main/notes/extending.func.html\u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m    583\u001b[39m     )\n",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[12]\u001b[39m\u001b[32m, line 68\u001b[39m, in \u001b[36mMinCostFlowValue.forward\u001b[39m\u001b[34m(ctx, n_nodes, cost, cap, b)\u001b[39m\n\u001b[32m     66\u001b[39m model.optimize()\n\u001b[32m     67\u001b[39m \u001b[38;5;28;01mif\u001b[39;00m model.Status != GRB.OPTIMAL:\n\u001b[32m---> \u001b[39m\u001b[32m68\u001b[39m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mRuntimeError\u001b[39;00m(\u001b[33mf\u001b[39m\u001b[33m\"\u001b[39m\u001b[33mGurobi did not find an optimal solution (status=\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mmodel.Status\u001b[38;5;132;01m}\u001b[39;00m\u001b[33m).\u001b[39m\u001b[33m\"\u001b[39m)\n\u001b[32m     70\u001b[39m flow = np.array([f[k].X \u001b[38;5;28;01mfor\u001b[39;00m k \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[32m2\u001b[39m * m)], dtype=np.float64)\n\u001b[32m     72\u001b[39m \u001b[38;5;66;03m# Duals:\u001b[39;00m\n\u001b[32m     73\u001b[39m \u001b[38;5;66;03m# - bal_constr[i].Pi is the dual for equality (free)\u001b[39;00m\n\u001b[32m     74\u001b[39m \u001b[38;5;66;03m# - cap_constr[k].Pi is the dual for <= (should be <= 0 in minimization)\u001b[39;00m\n",
      "\u001b[31mRuntimeError\u001b[39m: Gurobi did not find an optimal solution (status=3)."
     ]
    },
    {
     "data": {
      "image/png": "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WgRZXbzXrFkT3K6++mq3ILK+ZrgWAAAkqoTM8In66fXs2dMFbArmJk2a5FqyeH31NNyqLt2vvvqqu63948ePd4+7+eabXRGH5ut5HbfV10ZNE0MVK1bM/Ru+HwAAIJEkbMDXrVs32717t6uqVeNlBWVz5851S6KI9oX25FOzQt1/xx132PPPP+8aL6tz96muywcAAJAoEnqljXjDShsAgHjAShtZa+rUqW5xiL1796bL82XrlTYAAIC/9O7d23XO0Kb59SrEbNmypb388svBQkqP1tr1jvW2ChUqJLt/+fLlSR6nQKx58+aW3RDwAQCAuNGmTRs3LUudMz788ENXPHn77bdbhw4dXO/cUN60Lm/T0qmhlA275557MvkTxCcCPgAAsgPN4Dp6MPO3NM4cU7sztVIrX768NWjQwO6991577733XPCnodJQRYoUccd6m1bKCtW/f3+X4Zs7d25Mr60sorKEWo411FdffeWyhT///HNwedd69epZoUKFXBeP2267zQ4cOGDxLGGLNgAAQBocO2Q2qlzmn7J7t5rlLXRaT3HFFVdY/fr1bdasWdavX7+YH6dhXXXpGDZsmMsc5syZcp5L93fv3t2mT58e7Pohb7zxhusIUq1ateBxKvzU82tunQK+u+++2yZMmGDxigwfAACIe+qXq2HeUBquLVy4cHBTEBbu/vvvd0GZgrhYXH/99W6p1V9++SWY9ZsxY4ZbjSt0HqCGmlVEoWB05MiR9tZbb1k8I8MHAEB2kKfg/2XbsuJ100Gk5VPvuusuV+jhibQ0qoZ577zzTnvwwQddS7fUXHDBBS64VJ/eoUOH2qJFi9zKXNdee23wmIULF9qoUaNs/fr1rkpWcwtVSXvw4EE3zBuPyPABAJAdKFjS0GpmbymscZ8WGzZscBm1UArwqlevHty8BRPCDRkyxA4fPhzzkKuyfBrGFf3bunXrYDCpzF+7du1c/993333XVq1a5fr7yrFjxyxeEfABAIC49sknn7jlT091sYTChQvbAw88YI8++qjLyKXmuuuuc6+nYO6dd95xAaBn5cqVLqP31FNP2SWXXGLnnHOObd2aBZnTNCLgAwAAcePIkSO2fft2tzyqqmM1dNqpUyfXluXGG2885ee95ZZbXINib0nVlCiT2KRJE+vbt68L7vT6nrPPPtvte+6551zV7muvvZasqjce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fK5cubIbbv3999/t1VdfTdKWRS1Z1JpFBRqq0g1ty6LijMsuu8weffRR69Spk7333nt2//3305YFAABk77V0RQUUWkdXLU80303tURQwadg1o6iFipZ0GzFihG3bts0Fc3PnznXBnmjfr7/+GjxeDZp1/x133OHaxZQrV86effbZYLAnyuTNmDHDBXkPPPCAnX322S6wjKUHHwAAgG8zfBs3bnRtWJRNq1mzpity0Pw9Vax+8MEHLmjKLujDBwAAfNmHb9CgQS6oU4XqV199ZatXr3aZNWXUdB8AAACy1mkP6Wp92uXLl7s2LB5VvT722GOuchcAAABZ67QzfGpLsn///mT7Dxw44KppAQAAkOABn1bWuOWWW+yLL75w8/e0KeOnCtirr746fd4lAAAAsi7gU6Wr5vCpNUr+/PndpmrX6tWr27hx40736QEAAJDVc/iKFSvm+tWpWldtWZTh09q6CvgAAACQoAGfllJLyaeffhr8euzYsafyEgAAAMjKgE+tV2KhJswAAABIwIBv4cKF6f9OAAAAEJ9FGwAAAIhvBHwAAAA+R8AHAADgcwR8AAAAPkfABwAA4HMEfAAAAD5HwAcAAOBzBHwAAAA+R8AHAADgcwR8AAAAPkfABwAA4HMEfAAAAD5HwAcAAOBzBHwAAAA+R8AHAADgcwR8AAAAPkfABwAA4HMEfAAAAD6XkAHfnj17rGfPnnbGGWe4TV/v3bs3xccEAgEbPny4lStXzgoUKGDNmze3b7/9Nnj/n3/+af/617+sZs2aVrBgQatUqZINGjTI/vrrr0z4RAAAABknIQO+6667ztasWWPz5s1zm75W0JeSMWPG2NixY238+PH25ZdfWpkyZaxly5a2f/9+d//WrVvd9uSTT9q6dets6tSp7rn79u2bSZ8KAAAgY+QIKPWVQDZs2GB16tSx5cuXW6NGjdw+fd24cWP77rvvXIYunD6iMnuDBw+2e+65x+07cuSIlS5d2h5//HHr379/xNd6++237YYbbrCDBw9a7ty5U31v+/btcxlHZQWLFi162p8VAAAgPWKPhMvwLVu2zH0wL9iTSy65xO1bunRpxMds2rTJtm/fbq1atQruy5cvn11++eVRHyPeyYsW7Clo1IkO3QAAAOJNwgV8CtxKlSqVbL/26b5ojxFl9ELpdrTH7N6920aOHBk1+yejR48O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kT777DPr2LGjlStXzgWXc+bMiXjchAkTrGrVqpY/f3678MILbfHixTG/xo8//miFCxe2q6++2ho0aGCjRo1Kx08AAACQ+RKqrFZz6erXr299+vSxrl27Rjxm5syZNnjwYBf0NW3a1F588UVr27atrV+/3mUpRUFgpJ6B8+fPt2PHjrkAcc2aNVaqVClr06aNXXTRRdayZcsM/3wAAAAZIU1DuqtXr7YzzzzT3R4/frzdeOONKY4XpzcFbtpSMnbsWOvbt6/169fP3R43bpx99NFHNnHiRBs9erTbt2rVqqiPr1ChggvwKlas6G63a9fOBX+RAj4FjaGBo8bRAQAAEnZI97fffrMTJ04Eb9977722a9cuiydHjx51wVyrVq2S7NftpUuXxvQcCvb++OMP27Nnj508edINI9euXTvisQogNVHS27wgEQAAICEDvnAxFvdmKgWgCkpLly6dZL9ub9++PabnUDWu5u1ddtlldt5551mNGjWsQ4cOEY8dNmyYq4rxti1btqTL5wAAAPDVHD6t0vHwww+neMyXX35pDRs2jPk5VdARHpyG7zvdoWPJly+f2wAAAHwT8E2ZMsVVsHrr6E6dOtVKliyZ5Ji0tmUZOHCgde/ePcVjUur7F0rvJVeuXMmyeTt27EiW9QMAAMgu0lS0MXny5ODtMmXK2GuvvXbaffgUpIUHjacqb968rgJ3wYIFbkUQj2536tQpXV4DAADAtwGf+vBltQMHDtjGjRuDt9X7TxW0ao7stVzR2r49e/Z0Q8CNGze2SZMm2a+//moDBgzIwncOAACQjefwpcXKlSvdGrceBXfSq1cvN7ws3bp1s927d9uIESNs27ZtVrduXZs7d65Vrlw5y943AABAQqylK2pTosBq1qxZLuOnIVytaPGPf/zDZdXSUhjhR6ylCwAA4jH2iLkti+JCLTemhsa///671atXz84991z75ZdfrHfv3knmzAEAACABh3SV2VMT4o8//jjJsKp88skn1rlzZ3v11Vfd6hsAAACIHzFn+N588023ukZ4sCdXXHGFDR061KZPn57e7w8AAACZFfCtXbvW2rRpE/V+NSr++uuvT/f9AAAAIKsCvj///DPF5sW6T+vP+oGWVzv//PPdpjmLAAAA2WIOn9aoVSAUjVa40OobflCsWDHX3w8AACBbBXyq0lU1brS1Y48cOZKe7wsAAACZPaSr5salSpVyvV4ibbovoyt0VSXcsWNHK1eunOv5N2fOnIjHTZgwwfUHzJ8/v1tqbfHixWnuaaPHXXrppbZo0aJ0evcAAABxnuF75ZVXLKsdPHjQ6tevb3369LGuXbtGPGbmzJk2ePBgF/Q1bdrUXnzxRVdQsn79+uDyawrmImUk58+f74JJNZXWv9988421b9/e1q1bl2IzQwAAAN+stBFPlOGbPXu26/8XqlGjRtagQQObOHFicF/t2rXdcaNHj07z6yhYHDlypFubN5yCxtDAUZnBihUrptrtGgAAIC5X2kgER48etVWrVlmrVq2S7NftpUuXxvQcqjT2grjffvvNZQarVasW8VgFkKHD2gr2AAAA4o2vAr5du3a5auLw9jG6vX379pieY8OGDS6bp6HjDh062DPPPGMlSpSIeOywYcNcRO1tW7ZsSZfPAQAAkCVz+DLK8OHD7eGHH07xmC+//DLikGpKw72hNGodvi+aJk2auDl7sVDFcrSqZQAAgHiR5QHfwIEDrXv37ikeU6VKlZieq2TJkq4fYHg2b8eOHSk2jQYAAPCzLA/4FKRpSw958+Z1FbgLFiywLl26BPfrdqdOndLlNQAAABJNlgd8aXHgwAHbuHFj8PamTZvcihiaY+e1XBkyZIj17NnTDQE3btzYJk2aZL/++qsNGDAgC985AABA1kmogG/lypXWokWL4G0Fd15T6KlTp7qvu3XrZrt377YRI0bYtm3brG7dujZ37lyrXLlylr1vAACArJSwffgSuRcOAABAesiWffgAAACQHAEfAACAzxHwAQAA+BwBHwAAgM8R8EVx6NAhV9l75513Zu4VAQAASGcEfFE8+uij1qhRo/Q+3wAAAJmOgC+CH3/80b777jtr165d5l8RAACA7BzwffbZZ9axY0crV66c5ciRw+bMmRPxuAkTJljVqlUtf/78bqm1xYsXp+l1NIw7evTodHrXAAAAWSuhAr6DBw9a/fr1bfz48VGPmTlzpg0ePNjuu+8+W716tTVr1szatm3rllfzKAjUChzh29atW+29996zc845x20AAAB+kLArbSjDN3v2bOvcuXOS/Zp316BBA5s4cWJwX+3atd1xsWTthg0bZq+//rrlypXLrd177Ngx+/e//20PPvhgsmOPHDnittBu1xUrVmSlDQAAkCmy5UobR48e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",
      "text/plain": [
       "<Figure size 700x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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seBCXKltcsV2SdE2WYO2yTqj3axtomDKRFevXpXOnTvLwIEDxdnNnTvX7DC++uqrsn79emnZsqW0adNGDh48KM7s4sWLUr9+ffnqq6/E1axcuVIGDx4sa9askWXLlsn169eldevW5j05u7Jly8r7778vISEh5nLvvfdKu3btZMuWLeJKgoODZdKkSab4cyW1a9eWY8eOxV42bdpk9SLBhbhK2+Sq7ZKibbIGbZO1artT26Sx4shcgYGBNm9vb6de7U2aNLENGDAg3m01atSwjRgxwuYq9Ou9aNEim6sKDw8372HlypU2V1SoUCHbt99+a3MV58+ft1WtWtW2bNky2913320bOnSozRWMGjXKVr9+fasXA27A2dsmd2iXFG2TtWibMscoN2ubGGFCor2NoaGhZnQjLr2+evVq1lgmiYiIMP8WLlzYpdZ5dHS0zJkzx/So6tQ8V6Gje4888ojcf//94mp27dplpqHqVKVu3brJ3r17rV4kIF3RLjkP2qbMRdvkHLJZvQBwPqdOnTI7vSVKlIh3u14/fvy4ZcvlSbQTctiwYXLnnXdKnTp1xBXoULsWSFeuXJF8+fLJokWLpFatWuIKtMBbt26dmZLnapo2bSrTpk2TatWqyYkTJ+Sdd94xx6PodMgiRYpYvXhAuqBdcg60TZmLtsl5MMKUAQe1JbzoMR2uSJc94YYy4W3IGEOGDJGwsDCZPXu2y6zi6tWry4YNG8wxWHosRO/evWXr1q3i7A4dOiRDhw6VGTNmmAPJXY0ew9GpUyepW7euGR37+eefze1Tp061etFgIXdtm2iXrEXblHlom5wLI0zpsPHQKTC3UqFCBXElRYsWFS8vr5tGk8LDw28adUL6e+aZZ2Tx4sUmVUkPWHUVOXLkkCpVqpj/9/X1NaM1X3zxhUycOFGcmU4/1e+2Jm456Airrn8ND4mKijK/B1eRN29eUzzpND14Lndrm2iXrEfblLlom5wLBVM6bMT14k50x1d3HjWprUOHDrG363VNPkPG0BE8bZB0KtuKFSvM8Siu/n602HB29913303JPQEBAVKjRg15+eWXXapYUrrOt23bZhLE4LncrW2iXbIObZM1aJucCwVTJtLo0zNnzph/tQdbpy8p7ZXXYz6ciR4/06tXLzNSoMelaNSyLveAAQPEmV24cEF2794de33fvn1mPWtwQrly5cTZD+ycNWuW/PDDD+ZcTI4RPm9vb8mdO7c4s5EjR5qpYT4+PnL+/Hkz71qLviVLloiz03Wd8DgxHaXR439c4fix4cOHm3OP6fdbR8r0GKbIyEgzJRJwp7bJVdslRdtkDdom6wx3t7bJ6pg+T9K7d28TJ5rwsnz5cpszGjdunK18+fK2HDly2Bo1auQS8da6LhNbx7runV1iy60Xjfp1dn369In9rhQrVsx233332ZYuXWpzVa4UK961a1dbqVKlbNmzZ7eVLl3a1rFjR9uWLVusXiy4EFdqm1yxXVK0TdagbbJOVzdrm7Lof6wu2gAAAADAGZGSBwAAAABJoGACAAAAgCRQMAEAAABAEiiYAAAAACAJFEwAAAAAkAQKJgAAAABIAgUTAAAAACSBggkAAAAAkkDBBAAAAABJoGACnNDo0aOlQYMGVi8GAAAG7RI8GQUT4MKuXbtm9SIAABCLdgnuiIIJyCDTpk2TIkWKSFRUVLzbO3XqJE8++WSSjwsKCpI333xTNm7cKFmyZDEXvU3p/3/99dfSrl07yZs3r7zzzjvmbwULFoz3HN9//725b1w//vijNG7cWHLlyiWVKlUyr3H9+vV0fc8AAOdFuwSkDQUTkEE6d+4s0dHRsnjx4tjbTp06JT/99JMEBAQk+biuXbvKCy+8ILVr15Zjx46Zi97mMGrUKFMwbdq0Sfr06ZOiZfntt9+kZ8+e8uyzz8rWrVtl4sSJptB69913b/NdAgBcBe0SkDYUTEAGyZ07tzzxxBMSGBgYe9vMmTOlbNmy0qpVq1s+Ll++fJItWzYpWbKkuehtDvqcWijpKFH58uVTtCxaGI0YMUJ69+5tHvfAAw/I22+/bQonAIBnoF0C0iZbGh8HIAWeeuop8fPzkyNHjkiZMmVM8eTv73/TdLnU8PX1TfVjQkNDJTg4ON6Iko5+XblyRS5duiR58uRJ8/IAAFwH7RKQehRMQAZq2LCh1K9f38wbf/DBB800Oj2W6HbosUtxZc2aVWw22y0Pur1x44Y5Zqljx443PZ8e0wQA8Ay0S0DqUTABGaxfv37y2WefmVGm+++/X3x8fJJ9TI4cOcwIUEoUK1ZMzp8/LxcvXowtpjZs2BDvPo0aNZIdO3ZIlSpV0vguAADugnYJSB2OYQIyWI8ePUyx9M0336Q4pKFChQqyb98+U/hoUETCpL24mjZtaqbUjRw5Unbv3i2zZs2KTdVzeOONN8wol55HY8uWLbJt2zaZO3euvPbaa7f9/gAAroV2CUgdCiYggxUoUMBEiWuQQ/v27VP0GL3/Qw89JPfcc48ZQZo9e3aS9y1cuLDMmDFDfvnlF6lbt665rxZGcel0QE3nW7ZsmTmmqlmzZvLpp5+mODQCAOA+aJeA1MliS3jwA4B0p6l0NWvWlLFjx7J2AQCWo10CUo6CCchAZ86ckaVLl5rpD3r+o+rVq7O+AQCWoV0CUo/QByADadjC2bNn5YMPPohXLOlJaQ8cOJDoY/TcSFpgAQBAuwRYjxEmwAJaLCWM/nYoUaKE5M+fP9OXCQDguWiXgKRRMAEAAABAEkjJAwAAAIAkUDABAAAAQBIomAAAAAAgCRRMAAAAAJAECiYAAAAASAIFEwAAAAAkgYIJAAAAACRx/wPSuUFJfDBgMwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import sys\n",
    "import random\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Assumes you already have these defined (from the updated notebook):\n",
    "#   - DFN              (no alpha; outputs obj + beta)\n",
    "#   - make_int_param   (now continuous; softplus for nonneg)\n",
    "#   - min_cost_flow_value / MinCostFlowValue uses GUROBI inside\n",
    "\n",
    "# -------------------- Integer X + convex quadratic targets --------------------\n",
    "def make_int_quadratic_data(N=64, d=5, x_min=-2, x_max=3, noise=0.0, seed=0, device=\"cpu\"):\n",
    "    g = torch.Generator(device=\"cpu\").manual_seed(seed)\n",
    "\n",
    "    X_int = torch.randint(x_min, x_max, (N, d), generator=g)\n",
    "    X = X_int.to(device=device, dtype=torch.float32)\n",
    "\n",
    "    R = torch.randn(d, d, generator=g).to(device)\n",
    "    Q = R.T @ R + 0.2 * torch.eye(d, device=device)\n",
    "    lin = torch.randn(d, generator=g).to(device)\n",
    "\n",
    "    y = (X @ Q * X).sum(dim=1) + X @ lin\n",
    "    if noise > 0:\n",
    "        y = y + noise * torch.randn(y.shape, generator=g, device=device)\n",
    "\n",
    "    # normalize\n",
    "    y = (y - y.mean()) / (y.std() + 1e-8)\n",
    "\n",
    "    eigs = torch.linalg.eigvalsh(Q).detach().cpu()\n",
    "    print(f\"[data] X integer? {bool(torch.all(X == X.round()).item())}  X min/max: {int(X.min())}/{int(X.max())}\")\n",
    "    print(f\"[data] Q eig min/max: {eigs.min():.3f}/{eigs.max():.3f}  y mean/std: {y.mean().item():+.3f}/{y.std().item():.3f}\")\n",
    "    return X, y\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def eval_mse(model, X, y):\n",
    "    model.eval()\n",
    "    pred = model(X)\n",
    "    return F.mse_loss(pred, y).item()\n",
    "\n",
    "\n",
    "# -------------------- Baseline neural net --------------------\n",
    "class MLPRegressor(nn.Module):\n",
    "    def __init__(self, input_dim: int, hidden=(128, 128), dropout=0.0):\n",
    "        super().__init__()\n",
    "        layers = []\n",
    "        d = input_dim\n",
    "        for h in hidden:\n",
    "            layers += [nn.Linear(d, h), nn.ReLU()]\n",
    "            if dropout > 0:\n",
    "                layers += [nn.Dropout(dropout)]\n",
    "            d = h\n",
    "        layers += [nn.Linear(d, 1)]\n",
    "        self.net = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.net(x).squeeze(-1)  # [B]\n",
    "\n",
    "\n",
    "# -------------------- Live plot helpers (LOSS + GAP) --------------------\n",
    "def setup_live_plot():\n",
    "    fig, (ax_loss, ax_gap) = plt.subplots(2, 1, sharex=True, figsize=(7, 6))\n",
    "\n",
    "    lines = {}\n",
    "    # top: losses\n",
    "    (lines[\"DFN_train\"],) = ax_loss.plot([], [], label=\"DFN train\")\n",
    "    (lines[\"DFN_val\"],)   = ax_loss.plot([], [], label=\"DFN val\")\n",
    "    (lines[\"MLP_train\"],) = ax_loss.plot([], [], label=\"MLP train\")\n",
    "    (lines[\"MLP_val\"],)   = ax_loss.plot([], [], label=\"MLP val\")\n",
    "\n",
    "    ax_loss.set_title(\"Live MSE curves (DFN(Gurobi-LP) vs MLP)\")\n",
    "    ax_loss.set_ylabel(\"loss\")\n",
    "    ax_loss.legend()\n",
    "\n",
    "    # bottom: gap\n",
    "    (lines[\"gap_train\"],) = ax_gap.plot([], [], label=\"gap train (DFN - MLP)\")\n",
    "    (lines[\"gap_val\"],)   = ax_gap.plot([], [], label=\"gap val (DFN - MLP)\")\n",
    "    ax_gap.axhline(0.0, linewidth=1)\n",
    "\n",
    "    ax_gap.set_yscale(\"symlog\", linthresh=1e-6, linscale=1.0)\n",
    "\n",
    "    ax_gap.set_xlabel(\"step\")\n",
    "    ax_gap.set_ylabel(\"DFN - MLP\")\n",
    "    ax_gap.legend()\n",
    "\n",
    "    disp = None\n",
    "    if \"ipykernel\" in sys.modules:\n",
    "        from IPython.display import display\n",
    "        disp = display(fig, display_id=True)\n",
    "    else:\n",
    "        plt.ion()\n",
    "        plt.show(block=False)\n",
    "\n",
    "    return fig, (ax_loss, ax_gap), lines, disp\n",
    "\n",
    "\n",
    "def update_live_plot(fig, axes, lines, disp, steps_x, dfn_tr, dfn_va, mlp_tr, mlp_va):\n",
    "    ax_loss, ax_gap = axes\n",
    "\n",
    "    # losses\n",
    "    lines[\"DFN_train\"].set_data(steps_x, dfn_tr)\n",
    "    lines[\"DFN_val\"].set_data(steps_x, dfn_va)\n",
    "    lines[\"MLP_train\"].set_data(steps_x, mlp_tr)\n",
    "    lines[\"MLP_val\"].set_data(steps_x, mlp_va)\n",
    "\n",
    "    # gaps\n",
    "    gap_tr = [a - b for a, b in zip(dfn_tr, mlp_tr)]\n",
    "    gap_va = [a - b for a, b in zip(dfn_va, mlp_va)]\n",
    "    lines[\"gap_train\"].set_data(steps_x, gap_tr)\n",
    "    lines[\"gap_val\"].set_data(steps_x, gap_va)\n",
    "\n",
    "    ax_loss.relim(); ax_loss.autoscale_view()\n",
    "    ax_gap.relim();  ax_gap.autoscale_view()\n",
    "\n",
    "    fig.canvas.draw()\n",
    "    if disp is not None:\n",
    "        disp.update(fig)\n",
    "    else:\n",
    "        fig.canvas.flush_events()\n",
    "        plt.pause(0.001)\n",
    "\n",
    "\n",
    "# -------------------- Live scatter helpers (y_pred vs y_true) --------------------\n",
    "def setup_live_scatter(y_true_cpu, mn, mx):\n",
    "    fig, (ax_dfn, ax_mlp) = plt.subplots(1, 2, figsize=(10, 4), sharex=True, sharey=True)\n",
    "\n",
    "    sc_dfn = ax_dfn.scatter([], [], s=12)\n",
    "    sc_mlp = ax_mlp.scatter([], [], s=12)\n",
    "\n",
    "    ax_dfn.plot([mn, mx], [mn, mx])\n",
    "    ax_mlp.plot([mn, mx], [mn, mx])\n",
    "\n",
    "    ax_dfn.set_title(\"DFN(LP): y_pred vs y_true (val sample)\")\n",
    "    ax_mlp.set_title(\"MLP: y_pred vs y_true (val sample)\")\n",
    "\n",
    "    for ax in (ax_dfn, ax_mlp):\n",
    "        ax.set_xlabel(\"y_true\")\n",
    "        ax.set_xlim(mn, mx)\n",
    "        ax.set_ylim(mn, mx)\n",
    "\n",
    "    ax_dfn.set_ylabel(\"y_pred\")\n",
    "\n",
    "    disp = None\n",
    "    if \"ipykernel\" in sys.modules:\n",
    "        from IPython.display import display\n",
    "        disp = display(fig, display_id=True)\n",
    "    else:\n",
    "        plt.ion()\n",
    "        plt.show(block=False)\n",
    "\n",
    "    return fig, (ax_dfn, ax_mlp), (sc_dfn, sc_mlp), disp\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def update_live_scatter(fig, axes, scs, disp, y_true_cpu, ypred_dfn_cpu, ypred_mlp_cpu, dfn_mse=None, mlp_mse=None):\n",
    "    ax_dfn, ax_mlp = axes\n",
    "    sc_dfn, sc_mlp = scs\n",
    "\n",
    "    sc_dfn.set_offsets(np.column_stack([y_true_cpu, ypred_dfn_cpu]))\n",
    "    sc_mlp.set_offsets(np.column_stack([y_true_cpu, ypred_mlp_cpu]))\n",
    "\n",
    "    if dfn_mse is not None:\n",
    "        ax_dfn.set_title(f\"DFN(LP): y_pred vs y_true (val sample)  MSE={dfn_mse:.3g}\")\n",
    "    if mlp_mse is not None:\n",
    "        ax_mlp.set_title(f\"MLP: y_pred vs y_true (val sample)  MSE={mlp_mse:.3g}\")\n",
    "\n",
    "    fig.canvas.draw()\n",
    "    if disp is not None:\n",
    "        disp.update(fig)\n",
    "    else:\n",
    "        fig.canvas.flush_events()\n",
    "        plt.pause(0.001)\n",
    "\n",
    "\n",
    "# -------------------- Optional: parameter sanity checks --------------------\n",
    "@torch.no_grad()\n",
    "def print_dfn_param_summary(dfn: nn.Module, name=\"DFN\"):\n",
    "    dfn.eval()\n",
    "    cap0  = make_int_param(dfn.cap_raw, nonneg=True)\n",
    "    cost0 = make_int_param(dfn.cost_raw, nonneg=True)\n",
    "    A     = make_int_param(dfn.A_raw, nonneg=False)\n",
    "    b     = make_int_param(dfn.b_raw, nonneg=False)\n",
    "    print(f\"\\n[{name}] cap0 min/mean/max: {cap0.min().item():.3g}/{cap0.mean().item():.3g}/{cap0.max().item():.3g}\")\n",
    "    print(f\"[{name}] cost0 min/mean/max: {cost0.min().item():.3g}/{cost0.mean().item():.3g}/{cost0.max().item():.3g}\")\n",
    "    print(f\"[{name}] A   min/mean/max: {A.min().item():.3g}/{A.mean().item():.3g}/{A.max().item():.3g}\")\n",
    "    print(f\"[{name}] b   min/mean/max: {b.min().item():.3g}/{b.mean().item():.3g}/{b.max().item():.3g}\")\n",
    "\n",
    "\n",
    "# ===================== MAIN =====================\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "torch.manual_seed(0)\n",
    "random.seed(0)\n",
    "\n",
    "# NOTE: DFN(Gurobi) solves an LP per forward pass.\n",
    "# For speed, keep steps small or num_nodes smaller during quick experiments.\n",
    "N, d = 8000, 10\n",
    "X, y = make_int_quadratic_data(N=N, d=d, x_min=-50, x_max=50, noise=0.0, seed=0, device=device)\n",
    "\n",
    "# train/val split\n",
    "perm = torch.randperm(N, device=device)\n",
    "ntr = int(0.8 * N)\n",
    "tr, va = perm[:ntr], perm[ntr:]\n",
    "Xtr, ytr = X[tr], y[tr]\n",
    "Xva, yva = X[va], y[va]\n",
    "\n",
    "# -------------------- DFN (continuous params + Gurobi LP) --------------------\n",
    "dfn = DFN(input_dim=d, num_nodes=64, n_var=16, n_cst=16).to(device)\n",
    "\n",
    "# initialization (continuous)\n",
    "with torch.no_grad():\n",
    "    # A,b can be any real; costs/caps will be softplus()'d to be nonnegative\n",
    "    dfn.A_raw.normal_(0.0, 0.2)\n",
    "    dfn.b_raw.normal_(0.0, 0.2)\n",
    "    dfn.cost_raw.normal_(0.0, 0.2)\n",
    "    dfn.cap_raw.normal_(1.0, 0.2)\n",
    "\n",
    "# choose what you want to do with beta:\n",
    "# Option A (learn beta): keep trainable (default)\n",
    "# Option B (fix beta): uncomment below to freeze to min train label\n",
    "beta0 = ytr.min().detach()\n",
    "with torch.no_grad():\n",
    "    dfn.beta.fill_(beta0.item())\n",
    "dfn.beta.requires_grad_(False)\n",
    "\n",
    "print_dfn_param_summary(dfn)\n",
    "\n",
    "# DFN optimizer\n",
    "dfn_lrs = dict(cost=5e-2, cap=5e-2, A=5e-2, b=5e-2, beta=1e-2)\n",
    "opt_dfn = torch.optim.Adam([\n",
    "    {\"params\": [dfn.cost_raw], \"lr\": dfn_lrs[\"cost\"]},\n",
    "    {\"params\": [dfn.cap_raw],  \"lr\": dfn_lrs[\"cap\"]},\n",
    "    {\"params\": [dfn.A_raw],    \"lr\": dfn_lrs[\"A\"]},\n",
    "    {\"params\": [dfn.b_raw],    \"lr\": dfn_lrs[\"b\"]},\n",
    "    # {\"params\": [dfn.beta],     \"lr\": dfn_lrs[\"beta\"]},\n",
    "])\n",
    "\n",
    "# -------------------- Baseline model (MLP) --------------------\n",
    "mlp = MLPRegressor(input_dim=d, hidden=(128, 128), dropout=0.0).to(device)\n",
    "opt_mlp = torch.optim.Adam(mlp.parameters(), lr=1e-3)\n",
    "\n",
    "# static histogram\n",
    "plt.figure()\n",
    "plt.hist(y.detach().cpu().numpy(), bins=20)\n",
    "plt.title(\"Target y histogram\")\n",
    "plt.show()\n",
    "\n",
    "# -------------------- Live plots --------------------\n",
    "fig_lg, axes_lg, lines_lg, disp_lg = setup_live_plot()\n",
    "\n",
    "n_scatter = min(1024, Xva.shape[0])\n",
    "sc_idx = torch.randperm(Xva.shape[0], device=device)[:n_scatter]\n",
    "Xsc = Xva[sc_idx]\n",
    "ysc = yva[sc_idx]\n",
    "y_true_cpu = ysc.detach().cpu().numpy()\n",
    "mn, mx = float(ysc.min().item()), float(ysc.max().item())\n",
    "fig_sc, axes_sc, scs, disp_sc = setup_live_scatter(y_true_cpu, mn, mx)\n",
    "\n",
    "# -------------------- Training loop --------------------\n",
    "# Because DFN uses Gurobi per forward pass, keep steps modest.\n",
    "B, steps, print_every = 64, 40, 1\n",
    "\n",
    "dfn_train_hist, dfn_val_hist = [], []\n",
    "mlp_train_hist, mlp_val_hist = [], []\n",
    "hist_steps = []\n",
    "\n",
    "for t in range(steps):\n",
    "    idx = torch.randint(0, Xtr.shape[0], (B,), device=device)\n",
    "    Xb, yb = Xtr[idx], ytr[idx]\n",
    "\n",
    "    # ---- DFN step ----\n",
    "    dfn.train()\n",
    "    pred_dfn = dfn(Xb)\n",
    "    loss_dfn = F.mse_loss(pred_dfn, yb)\n",
    "    opt_dfn.zero_grad(set_to_none=True)\n",
    "    loss_dfn.backward()\n",
    "    opt_dfn.step()\n",
    "\n",
    "    # ---- MLP step ----\n",
    "    mlp.train()\n",
    "    pred_mlp = mlp(Xb)\n",
    "    loss_mlp = F.mse_loss(pred_mlp, yb)\n",
    "    opt_mlp.zero_grad(set_to_none=True)\n",
    "    loss_mlp.backward()\n",
    "    opt_mlp.step()\n",
    "\n",
    "    if t % print_every == 0:\n",
    "        hist_steps.append(t)\n",
    "        dfn_train_hist.append(loss_dfn.item())\n",
    "        mlp_train_hist.append(loss_mlp.item())\n",
    "\n",
    "        dfn_val = eval_mse(dfn, Xva, yva)\n",
    "        mlp_val = eval_mse(mlp, Xva, yva)\n",
    "        dfn_val_hist.append(dfn_val)\n",
    "        mlp_val_hist.append(mlp_val)\n",
    "\n",
    "        update_live_plot(\n",
    "            fig_lg, axes_lg, lines_lg, disp_lg,\n",
    "            hist_steps,\n",
    "            dfn_train_hist, dfn_val_hist,\n",
    "            mlp_train_hist, mlp_val_hist\n",
    "        )\n",
    "\n",
    "        # scatter update on fixed subset\n",
    "        dfn.eval()\n",
    "        mlp.eval()\n",
    "        with torch.no_grad():\n",
    "            yp_dfn_sc = dfn(Xsc).detach().cpu().numpy()\n",
    "            yp_mlp_sc = mlp(Xsc).detach().cpu().numpy()\n",
    "\n",
    "        update_live_scatter(\n",
    "            fig_sc, axes_sc, scs, disp_sc,\n",
    "            y_true_cpu,\n",
    "            yp_dfn_sc, yp_mlp_sc,\n",
    "            dfn_mse=dfn_val, mlp_mse=mlp_val\n",
    "        )\n",
    "\n",
    "# -------------------- Final comparison --------------------\n",
    "dfn_val_final = eval_mse(dfn, Xva, yva)\n",
    "mlp_val_final = eval_mse(mlp, Xva, yva)\n",
    "\n",
    "print(\"\\n=== FINAL COMPARISON (validation MSE) ===\")\n",
    "print(f\"DFN(LP+Gurobi): {dfn_val_final:.6f}\")\n",
    "print(f\"MLP          : {mlp_val_final:.6f}\")\n",
    "\n",
    "# static curves (end)\n",
    "plt.figure()\n",
    "plt.plot(hist_steps, dfn_train_hist, label=\"DFN train\")\n",
    "plt.plot(hist_steps, dfn_val_hist,   label=\"DFN val\")\n",
    "plt.plot(hist_steps, mlp_train_hist, label=\"MLP train\")\n",
    "plt.plot(hist_steps, mlp_val_hist,   label=\"MLP val\")\n",
    "plt.legend()\n",
    "plt.title(\"MSE curves (DFN(LP) vs MLP)\")\n",
    "plt.xlabel(\"step\")\n",
    "plt.ylabel(\"loss\")\n",
    "plt.show()\n",
    "\n",
    "# static gap plot at end\n",
    "plt.figure()\n",
    "gap_train = [a - b for a, b in zip(dfn_train_hist, mlp_train_hist)]\n",
    "gap_val   = [a - b for a, b in zip(dfn_val_hist, mlp_val_hist)]\n",
    "plt.plot(hist_steps, gap_train, label=\"gap train (DFN - MLP)\")\n",
    "plt.plot(hist_steps, gap_val,   label=\"gap val (DFN - MLP)\")\n",
    "plt.axhline(0.0)\n",
    "plt.yscale(\"symlog\", linthresh=1e-6, linscale=1.0)\n",
    "plt.legend()\n",
    "plt.title(\"Gap curves (DFN - MLP) [symlog]\")\n",
    "plt.xlabel(\"step\")\n",
    "plt.ylabel(\"loss difference\")\n",
    "plt.show()\n",
    "\n",
    "# final scatters (full val)\n",
    "dfn.eval()\n",
    "mlp.eval()\n",
    "with torch.no_grad():\n",
    "    yp_dfn = dfn(Xva).detach().cpu()\n",
    "    yp_mlp = mlp(Xva).detach().cpu()\n",
    "    yt = yva.detach().cpu()\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(yt.numpy(), yp_dfn.numpy(), s=12)\n",
    "mn2, mx2 = yt.min().item(), yt.max().item()\n",
    "plt.plot([mn2, mx2], [mn2, mx2])\n",
    "plt.title(\"DFN(LP): y_true vs y_pred (val)\")\n",
    "plt.xlabel(\"y_true\")\n",
    "plt.ylabel(\"y_pred\")\n",
    "plt.show()\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(yt.numpy(), yp_mlp.numpy(), s=12)\n",
    "plt.plot([mn2, mx2], [mn2, mx2])\n",
    "plt.title(\"MLP: y_true vs y_pred (val)\")\n",
    "plt.xlabel(\"y_true\")\n",
    "plt.ylabel(\"y_pred\")\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "0cf48dbf-df84-4912-9377-faf0b5ae73a0",
   "metadata": {},
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'DFN' object has no attribute '_build_cap'",
     "output_type": "error",
     "traceback": [
      "\u001b[31m---------------------------------------------------------------------------\u001b[39m",
      "\u001b[31mAttributeError\u001b[39m                            Traceback (most recent call last)",
      "\u001b[36mCell\u001b[39m\u001b[36m \u001b[39m\u001b[32mIn[18]\u001b[39m\u001b[32m, line 3\u001b[39m\n\u001b[32m      1\u001b[39m \u001b[38;5;28;01mwith\u001b[39;00m torch.no_grad():\n\u001b[32m      2\u001b[39m     m = model.n_aug * (model.n_aug - \u001b[32m1\u001b[39m)\n\u001b[32m----> \u001b[39m\u001b[32m3\u001b[39m     cap = \u001b[43mmodel\u001b[49m\u001b[43m.\u001b[49m\u001b[43m_build_cap\u001b[49m()  \u001b[38;5;66;03m# or whatever you name it; else print the cap tensor you pass\u001b[39;00m\n\u001b[32m      4\u001b[39m     \u001b[38;5;28mprint\u001b[39m(\u001b[33m\"\u001b[39m\u001b[33mcap shape:\u001b[39m\u001b[33m\"\u001b[39m, cap.shape, \u001b[33m\"\u001b[39m\u001b[33mexpected:\u001b[39m\u001b[33m\"\u001b[39m, (\u001b[32m2\u001b[39m*m,))\n\u001b[32m      5\u001b[39m     \u001b[38;5;28mprint\u001b[39m(\u001b[33m\"\u001b[39m\u001b[33mcap first half  min/max:\u001b[39m\u001b[33m\"\u001b[39m, \u001b[38;5;28mfloat\u001b[39m(cap[:m].min()), \u001b[38;5;28mfloat\u001b[39m(cap[:m].max()))\n",
      "\u001b[36mFile \u001b[39m\u001b[32m/opt/anaconda3/envs/dfn/lib/python3.11/site-packages/torch/nn/modules/module.py:1931\u001b[39m, in \u001b[36mModule.__getattr__\u001b[39m\u001b[34m(self, name)\u001b[39m\n\u001b[32m   1929\u001b[39m     \u001b[38;5;28;01mif\u001b[39;00m name \u001b[38;5;129;01min\u001b[39;00m modules:\n\u001b[32m   1930\u001b[39m         \u001b[38;5;28;01mreturn\u001b[39;00m modules[name]\n\u001b[32m-> \u001b[39m\u001b[32m1931\u001b[39m \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mAttributeError\u001b[39;00m(\n\u001b[32m   1932\u001b[39m     \u001b[33mf\u001b[39m\u001b[33m\"\u001b[39m\u001b[33m'\u001b[39m\u001b[38;5;132;01m{\u001b[39;00m\u001b[38;5;28mtype\u001b[39m(\u001b[38;5;28mself\u001b[39m).\u001b[34m__name__\u001b[39m\u001b[38;5;132;01m}\u001b[39;00m\u001b[33m'\u001b[39m\u001b[33m object has no attribute \u001b[39m\u001b[33m'\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mname\u001b[38;5;132;01m}\u001b[39;00m\u001b[33m'\u001b[39m\u001b[33m\"\u001b[39m\n\u001b[32m   1933\u001b[39m )\n",
      "\u001b[31mAttributeError\u001b[39m: 'DFN' object has no attribute '_build_cap'"
     ]
    }
   ],
   "source": [
    "with torch.no_grad():\n",
    "    m = model.n_aug * (model.n_aug - 1)\n",
    "    cap = model._build_cap()  # or whatever you name it; else print the cap tensor you pass\n",
    "    print(\"cap shape:\", cap.shape, \"expected:\", (2*m,))\n",
    "    print(\"cap first half  min/max:\", float(cap[:m].min()), float(cap[:m].max()))\n",
    "    print(\"cap second half min/max:\", float(cap[m:].min()), float(cap[m:].max()))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9b0b3228-f445-4bf1-848b-8d1b2e18557d",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Tests + visualizations (DFN with continuous params + Gurobi LP)\n",
    "\n",
    "This section reproduces the training / live plots / scatter workflow from your script, but using the new DFN:\n",
    "- No `alpha` (output is `LP_obj + beta`)\n",
    "- `cap` and `cost` are continuous nonnegative via `softplus`\n",
    "- LP solve is done with Gurobi inside `MinCostFlowValue`\n",
    "\n",
    "Note: DFN now solves an LP during each forward pass, so start with smaller `num_nodes` / fewer `steps` and scale up as needed.\n"
   ]
  },
  {
   "cell_type": "code",
   "metadata": {},
   "execution_count": null,
   "outputs": [],
   "source": [
    "\n",
    "import sys\n",
    "import random\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Assumes (from earlier cells) you have:\n",
    "#   - DFN\n",
    "#   - make_int_param (continuous; softplus for nonneg)\n",
    "#   - Gurobi-based min_cost_flow_value via MinCostFlowValue\n",
    "\n",
    "\n",
    "# -------------------- Integer X + convex quadratic targets --------------------\n",
    "def make_int_quadratic_data(N=64, d=5, x_min=-2, x_max=3, noise=0.0, seed=0, device=\"cpu\"):\n",
    "    g = torch.Generator(device=\"cpu\").manual_seed(seed)\n",
    "\n",
    "    X_int = torch.randint(x_min, x_max, (N, d), generator=g)\n",
    "    X = X_int.to(device=device, dtype=torch.float32)\n",
    "\n",
    "    R = torch.randn(d, d, generator=g).to(device)\n",
    "    Q = R.T @ R + 0.2 * torch.eye(d, device=device)\n",
    "    lin = torch.randn(d, generator=g).to(device)\n",
    "\n",
    "    y = (X @ Q * X).sum(dim=1) + X @ lin\n",
    "    if noise > 0:\n",
    "        y = y + noise * torch.randn(y.shape, generator=g, device=device)\n",
    "\n",
    "    # normalize\n",
    "    y = (y - y.mean()) / (y.std() + 1e-8)\n",
    "\n",
    "    eigs = torch.linalg.eigvalsh(Q).detach().cpu()\n",
    "    print(f\"[data] X integer? {bool(torch.all(X == X.round()).item())}  X min/max: {int(X.min())}/{int(X.max())}\")\n",
    "    print(f\"[data] Q eig min/max: {eigs.min():.3f}/{eigs.max():.3f}  y mean/std: {y.mean().item():+.3f}/{y.std().item():.3f}\")\n",
    "    return X, y\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def eval_mse(model, X, y):\n",
    "    model.eval()\n",
    "    pred = model(X)\n",
    "    return F.mse_loss(pred, y).item()\n",
    "\n",
    "\n",
    "# -------------------- Baseline neural net --------------------\n",
    "class MLPRegressor(nn.Module):\n",
    "    def __init__(self, input_dim: int, hidden=(128, 128), dropout=0.0):\n",
    "        super().__init__()\n",
    "        layers = []\n",
    "        d = input_dim\n",
    "        for h in hidden:\n",
    "            layers += [nn.Linear(d, h), nn.ReLU()]\n",
    "            if dropout > 0:\n",
    "                layers += [nn.Dropout(dropout)]\n",
    "            d = h\n",
    "        layers += [nn.Linear(d, 1)]\n",
    "        self.net = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.net(x).squeeze(-1)  # [B]\n",
    "\n",
    "\n",
    "# -------------------- Live plot helpers (LOSS + GAP) --------------------\n",
    "def setup_live_plot():\n",
    "    fig, (ax_loss, ax_gap) = plt.subplots(2, 1, sharex=True, figsize=(7, 6))\n",
    "\n",
    "    lines = {}\n",
    "    (lines[\"DFN_train\"],) = ax_loss.plot([], [], label=\"DFN train\")\n",
    "    (lines[\"DFN_val\"],)   = ax_loss.plot([], [], label=\"DFN val\")\n",
    "    (lines[\"MLP_train\"],) = ax_loss.plot([], [], label=\"MLP train\")\n",
    "    (lines[\"MLP_val\"],)   = ax_loss.plot([], [], label=\"MLP val\")\n",
    "\n",
    "    ax_loss.set_title(\"Live MSE curves (DFN(LP/Gurobi) vs MLP)\")\n",
    "    ax_loss.set_ylabel(\"loss\")\n",
    "    ax_loss.legend()\n",
    "\n",
    "    (lines[\"gap_train\"],) = ax_gap.plot([], [], label=\"gap train (DFN - MLP)\")\n",
    "    (lines[\"gap_val\"],)   = ax_gap.plot([], [], label=\"gap val (DFN - MLP)\")\n",
    "    ax_gap.axhline(0.0, linewidth=1)\n",
    "    ax_gap.set_yscale(\"symlog\", linthresh=1e-6, linscale=1.0)\n",
    "    ax_gap.set_xlabel(\"step\")\n",
    "    ax_gap.set_ylabel(\"DFN - MLP\")\n",
    "    ax_gap.legend()\n",
    "\n",
    "    disp = None\n",
    "    if \"ipykernel\" in sys.modules:\n",
    "        from IPython.display import display\n",
    "        disp = display(fig, display_id=True)\n",
    "    else:\n",
    "        plt.ion()\n",
    "        plt.show(block=False)\n",
    "\n",
    "    return fig, (ax_loss, ax_gap), lines, disp\n",
    "\n",
    "\n",
    "def update_live_plot(fig, axes, lines, disp, steps_x, dfn_tr, dfn_va, mlp_tr, mlp_va):\n",
    "    ax_loss, ax_gap = axes\n",
    "\n",
    "    lines[\"DFN_train\"].set_data(steps_x, dfn_tr)\n",
    "    lines[\"DFN_val\"].set_data(steps_x, dfn_va)\n",
    "    lines[\"MLP_train\"].set_data(steps_x, mlp_tr)\n",
    "    lines[\"MLP_val\"].set_data(steps_x, mlp_va)\n",
    "\n",
    "    gap_tr = [a - b for a, b in zip(dfn_tr, mlp_tr)]\n",
    "    gap_va = [a - b for a, b in zip(dfn_va, mlp_va)]\n",
    "    lines[\"gap_train\"].set_data(steps_x, gap_tr)\n",
    "    lines[\"gap_val\"].set_data(steps_x, gap_va)\n",
    "\n",
    "    ax_loss.relim(); ax_loss.autoscale_view()\n",
    "    ax_gap.relim();  ax_gap.autoscale_view()\n",
    "\n",
    "    fig.canvas.draw()\n",
    "    if disp is not None:\n",
    "        disp.update(fig)\n",
    "    else:\n",
    "        fig.canvas.flush_events()\n",
    "        plt.pause(0.001)\n",
    "\n",
    "\n",
    "# -------------------- Live scatter helpers (y_pred vs y_true) --------------------\n",
    "def setup_live_scatter(y_true_cpu, mn, mx):\n",
    "    fig, (ax_dfn, ax_mlp) = plt.subplots(1, 2, figsize=(10, 4), sharex=True, sharey=True)\n",
    "\n",
    "    sc_dfn = ax_dfn.scatter([], [], s=12)\n",
    "    sc_mlp = ax_mlp.scatter([], [], s=12)\n",
    "\n",
    "    ax_dfn.plot([mn, mx], [mn, mx])\n",
    "    ax_mlp.plot([mn, mx], [mn, mx])\n",
    "\n",
    "    ax_dfn.set_title(\"DFN(LP): y_pred vs y_true (val sample)\")\n",
    "    ax_mlp.set_title(\"MLP: y_pred vs y_true (val sample)\")\n",
    "\n",
    "    for ax in (ax_dfn, ax_mlp):\n",
    "        ax.set_xlabel(\"y_true\")\n",
    "        ax.set_xlim(mn, mx)\n",
    "        ax.set_ylim(mn, mx)\n",
    "\n",
    "    ax_dfn.set_ylabel(\"y_pred\")\n",
    "\n",
    "    disp = None\n",
    "    if \"ipykernel\" in sys.modules:\n",
    "        from IPython.display import display\n",
    "        disp = display(fig, display_id=True)\n",
    "    else:\n",
    "        plt.ion()\n",
    "        plt.show(block=False)\n",
    "\n",
    "    return fig, (ax_dfn, ax_mlp), (sc_dfn, sc_mlp), disp\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def update_live_scatter(fig, axes, scs, disp, y_true_cpu, ypred_dfn_cpu, ypred_mlp_cpu, dfn_mse=None, mlp_mse=None):\n",
    "    ax_dfn, ax_mlp = axes\n",
    "    sc_dfn, sc_mlp = scs\n",
    "\n",
    "    sc_dfn.set_offsets(np.column_stack([y_true_cpu, ypred_dfn_cpu]))\n",
    "    sc_mlp.set_offsets(np.column_stack([y_true_cpu, ypred_mlp_cpu]))\n",
    "\n",
    "    if dfn_mse is not None:\n",
    "        ax_dfn.set_title(f\"DFN(LP): y_pred vs y_true (val sample)  MSE={dfn_mse:.3g}\")\n",
    "    if mlp_mse is not None:\n",
    "        ax_mlp.set_title(f\"MLP: y_pred vs y_true (val sample)  MSE={mlp_mse:.3g}\")\n",
    "\n",
    "    fig.canvas.draw()\n",
    "    if disp is not None:\n",
    "        disp.update(fig)\n",
    "    else:\n",
    "        fig.canvas.flush_events()\n",
    "        plt.pause(0.001)\n",
    "\n",
    "\n",
    "# -------------------- Optional: parameter sanity checks --------------------\n",
    "@torch.no_grad()\n",
    "def print_dfn_param_summary(dfn: nn.Module, name=\"DFN\"):\n",
    "    dfn.eval()\n",
    "    cap0  = make_int_param(dfn.cap_raw, nonneg=True)\n",
    "    cost0 = make_int_param(dfn.cost_raw, nonneg=True)\n",
    "    A     = make_int_param(dfn.A_raw, nonneg=False)\n",
    "    b     = make_int_param(dfn.b_raw, nonneg=False)\n",
    "    print(f\"\\n[{name}] cap0  min/mean/max: {cap0.min().item():.3g}/{cap0.mean().item():.3g}/{cap0.max().item():.3g}\")\n",
    "    print(f\"[{name}] cost0 min/mean/max: {cost0.min().item():.3g}/{cost0.mean().item():.3g}/{cost0.max().item():.3g}\")\n",
    "    print(f\"[{name}] A     min/mean/max: {A.min().item():.3g}/{A.mean().item():.3g}/{A.max().item():.3g}\")\n",
    "    print(f\"[{name}] b     min/mean/max: {b.min().item():.3g}/{b.mean().item():.3g}/{b.max().item():.3g}\")\n",
    "\n",
    "\n",
    "# ===================== MAIN =====================\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "torch.manual_seed(0)\n",
    "random.seed(0)\n",
    "\n",
    "# DFN solves an LP per forward pass -> start small, then scale.\n",
    "N, d = 8000, 10\n",
    "X, y = make_int_quadratic_data(N=N, d=d, x_min=-50, x_max=50, noise=0.0, seed=0, device=device)\n",
    "\n",
    "# train/val split\n",
    "perm = torch.randperm(N, device=device)\n",
    "ntr = int(0.8 * N)\n",
    "tr, va = perm[:ntr], perm[ntr:]\n",
    "Xtr, ytr = X[tr], y[tr]\n",
    "Xva, yva = X[va], y[va]\n",
    "\n",
    "# -------------------- DFN (continuous params + Gurobi LP) --------------------\n",
    "dfn = DFN(input_dim=d, num_nodes=64, n_var=16, n_cst=16).to(device)\n",
    "\n",
    "with torch.no_grad():\n",
    "    # A,b can be any real; costs/caps become nonnegative via softplus()\n",
    "    dfn.A_raw.normal_(0.0, 0.2)\n",
    "    dfn.b_raw.normal_(0.0, 0.2)\n",
    "    dfn.cost_raw.normal_(0.0, 0.2)\n",
    "    dfn.cap_raw.normal_(1.0, 0.2)\n",
    "\n",
    "# (optional) Fix beta to min TRAIN label (frozen), like your earlier script:\n",
    "# beta0 = ytr.min().detach()\n",
    "# with torch.no_grad():\n",
    "#     dfn.beta.fill_(beta0.item())\n",
    "# dfn.beta.requires_grad_(False)\n",
    "\n",
    "print_dfn_param_summary(dfn)\n",
    "\n",
    "dfn_lrs = dict(cost=5e-2, cap=5e-2, A=5e-2, b=5e-2, beta=1e-2)\n",
    "opt_dfn = torch.optim.Adam([\n",
    "    {\"params\": [dfn.cost_raw], \"lr\": dfn_lrs[\"cost\"]},\n",
    "    {\"params\": [dfn.cap_raw],  \"lr\": dfn_lrs[\"cap\"]},\n",
    "    {\"params\": [dfn.A_raw],    \"lr\": dfn_lrs[\"A\"]},\n",
    "    {\"params\": [dfn.b_raw],    \"lr\": dfn_lrs[\"b\"]},\n",
    "    {\"params\": [dfn.beta],     \"lr\": dfn_lrs[\"beta\"]},  # remove if you freeze beta\n",
    "])\n",
    "\n",
    "# -------------------- Baseline model (MLP) --------------------\n",
    "mlp = MLPRegressor(input_dim=d, hidden=(128, 128), dropout=0.0).to(device)\n",
    "opt_mlp = torch.optim.Adam(mlp.parameters(), lr=1e-3)\n",
    "\n",
    "# histogram\n",
    "plt.figure()\n",
    "plt.hist(y.detach().cpu().numpy(), bins=20)\n",
    "plt.title(\"Target y histogram\")\n",
    "plt.show()\n",
    "\n",
    "# -------------------- Live plots --------------------\n",
    "fig_lg, axes_lg, lines_lg, disp_lg = setup_live_plot()\n",
    "\n",
    "n_scatter = min(1024, Xva.shape[0])\n",
    "sc_idx = torch.randperm(Xva.shape[0], device=device)[:n_scatter]\n",
    "Xsc = Xva[sc_idx]\n",
    "ysc = yva[sc_idx]\n",
    "y_true_cpu = ysc.detach().cpu().numpy()\n",
    "mn, mx = float(ysc.min().item()), float(ysc.max().item())\n",
    "fig_sc, axes_sc, scs, disp_sc = setup_live_scatter(y_true_cpu, mn, mx)\n",
    "\n",
    "# -------------------- Training loop --------------------\n",
    "B, steps, print_every = 64, 40, 1\n",
    "\n",
    "dfn_train_hist, dfn_val_hist = [], []\n",
    "mlp_train_hist, mlp_val_hist = [], []\n",
    "hist_steps = []\n",
    "\n",
    "for t in range(steps):\n",
    "    idx = torch.randint(0, Xtr.shape[0], (B,), device=device)\n",
    "    Xb, yb = Xtr[idx], ytr[idx]\n",
    "\n",
    "    # ---- DFN step ----\n",
    "    dfn.train()\n",
    "    pred_dfn = dfn(Xb)\n",
    "    loss_dfn = F.mse_loss(pred_dfn, yb)\n",
    "    opt_dfn.zero_grad(set_to_none=True)\n",
    "    loss_dfn.backward()\n",
    "    opt_dfn.step()\n",
    "\n",
    "    # ---- MLP step ----\n",
    "    mlp.train()\n",
    "    pred_mlp = mlp(Xb)\n",
    "    loss_mlp = F.mse_loss(pred_mlp, yb)\n",
    "    opt_mlp.zero_grad(set_to_none=True)\n",
    "    loss_mlp.backward()\n",
    "    opt_mlp.step()\n",
    "\n",
    "    if t % print_every == 0:\n",
    "        hist_steps.append(t)\n",
    "        dfn_train_hist.append(loss_dfn.item())\n",
    "        mlp_train_hist.append(loss_mlp.item())\n",
    "\n",
    "        dfn_val = eval_mse(dfn, Xva, yva)\n",
    "        mlp_val = eval_mse(mlp, Xva, yva)\n",
    "        dfn_val_hist.append(dfn_val)\n",
    "        mlp_val_hist.append(mlp_val)\n",
    "\n",
    "        update_live_plot(\n",
    "            fig_lg, axes_lg, lines_lg, disp_lg,\n",
    "            hist_steps,\n",
    "            dfn_train_hist, dfn_val_hist,\n",
    "            mlp_train_hist, mlp_val_hist\n",
    "        )\n",
    "\n",
    "        dfn.eval()\n",
    "        mlp.eval()\n",
    "        with torch.no_grad():\n",
    "            yp_dfn_sc = dfn(Xsc).detach().cpu().numpy()\n",
    "            yp_mlp_sc = mlp(Xsc).detach().cpu().numpy()\n",
    "\n",
    "        update_live_scatter(\n",
    "            fig_sc, axes_sc, scs, disp_sc,\n",
    "            y_true_cpu,\n",
    "            yp_dfn_sc, yp_mlp_sc,\n",
    "            dfn_mse=dfn_val, mlp_mse=mlp_val\n",
    "        )\n",
    "\n",
    "# -------------------- Final comparison --------------------\n",
    "dfn_val_final = eval_mse(dfn, Xva, yva)\n",
    "mlp_val_final = eval_mse(mlp, Xva, yva)\n",
    "\n",
    "print(\"\\n=== FINAL COMPARISON (validation MSE) ===\")\n",
    "print(f\"DFN(LP+Gurobi): {dfn_val_final:.6f}\")\n",
    "print(f\"MLP          : {mlp_val_final:.6f}\")\n",
    "\n",
    "# Static curves\n",
    "plt.figure()\n",
    "plt.plot(hist_steps, dfn_train_hist, label=\"DFN train\")\n",
    "plt.plot(hist_steps, dfn_val_hist,   label=\"DFN val\")\n",
    "plt.plot(hist_steps, mlp_train_hist, label=\"MLP train\")\n",
    "plt.plot(hist_steps, mlp_val_hist,   label=\"MLP val\")\n",
    "plt.legend()\n",
    "plt.title(\"MSE curves (DFN(LP) vs MLP)\")\n",
    "plt.xlabel(\"step\")\n",
    "plt.ylabel(\"loss\")\n",
    "plt.show()\n",
    "\n",
    "# Gap plot\n",
    "plt.figure()\n",
    "gap_train = [a - b for a, b in zip(dfn_train_hist, mlp_train_hist)]\n",
    "gap_val   = [a - b for a, b in zip(dfn_val_hist, mlp_val_hist)]\n",
    "plt.plot(hist_steps, gap_train, label=\"gap train (DFN - MLP)\")\n",
    "plt.plot(hist_steps, gap_val,   label=\"gap val (DFN - MLP)\")\n",
    "plt.axhline(0.0)\n",
    "plt.yscale(\"symlog\", linthresh=1e-6, linscale=1.0)\n",
    "plt.legend()\n",
    "plt.title(\"Gap curves (DFN - MLP) [symlog]\")\n",
    "plt.xlabel(\"step\")\n",
    "plt.ylabel(\"loss difference\")\n",
    "plt.show()\n",
    "\n",
    "# Final scatters (full val)\n",
    "dfn.eval()\n",
    "mlp.eval()\n",
    "with torch.no_grad():\n",
    "    yp_dfn = dfn(Xva).detach().cpu()\n",
    "    yp_mlp = mlp(Xva).detach().cpu()\n",
    "    yt = yva.detach().cpu()\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(yt.numpy(), yp_dfn.numpy(), s=12)\n",
    "mn2, mx2 = yt.min().item(), yt.max().item()\n",
    "plt.plot([mn2, mx2], [mn2, mx2])\n",
    "plt.title(\"DFN(LP): y_true vs y_pred (val)\")\n",
    "plt.xlabel(\"y_true\")\n",
    "plt.ylabel(\"y_pred\")\n",
    "plt.show()\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(yt.numpy(), yp_mlp.numpy(), s=12)\n",
    "plt.plot([mn2, mx2], [mn2, mx2])\n",
    "plt.title(\"MLP: y_true vs y_pred (val)\")\n",
    "plt.xlabel(\"y_true\")\n",
    "plt.ylabel(\"y_pred\")\n",
    "plt.show()\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "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"
  },
  "generated_by": {
   "continuous_gurobi_tests_added": "2026-01-15T23:41:53.997804Z"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}