{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "dd25a7af",
   "metadata": {},
   "source": [
    "# Fast Extragradient for Lipschitz Monotone Operators\n",
    "\n",
    "We consider a monotone and L-Lipschitz operator A with a zero x_star satisfying A(x_star)=0. The initial condition is ||x_0 - x_star||^2 <= R^2, and the performance metric is ||A(x_N)||^2.\n",
    "\n",
    "For k = 0, the half iterate is the initial point itself: x_{0.5} = x_0. For k >= 0, the method is\n",
    "\n",
    "$$\n",
    "x_{k+1/2} = x_k + \\frac{1}{k+1}(x_0 - x_k) - \\frac{k}{k+1}\\frac{1}{L} A(x_k),\n",
    "$$\n",
    "\n",
    "$$\n",
    "x_{k+1} = x_k + \\frac{1}{k+1}(x_0 - x_k) - \\frac{1}{L} A(x_{k+1/2}).\n",
    "$$\n",
    "\n",
    "Block 1 numerical evidence suggests the candidate rate ||A(x_N)||^2 <= 4 L^2 R^2 / N^2 for N >= 2, with the separate value 2 L^2 R^2 at N = 1."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dc192848",
   "metadata": {},
   "source": [
    "## Proof Statement"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3db493f9",
   "metadata": {},
   "source": [
    "### Theorem\n",
    "\n",
    "Assume that $A$ is monotone and $L$-Lipschitz, and let $x_{\\star}$ satisfy $A(x_{\\star})=0$. Let the Fast Extragradient method be\n",
    "\n",
    "$$\n",
    "x_{k+1/2}=x_k+\\frac{1}{k+1}(x_0-x_k)-\\frac{k}{k+1}\\frac{1}{L}A(x_k),\n",
    "$$\n",
    "\n",
    "$$\n",
    "x_{k+1}=x_k+\\frac{1}{k+1}(x_0-x_k)-\\frac{1}{L}A(x_{k+1/2}),\n",
    "$$\n",
    "\n",
    "with $x_{0+1/2}=x_0$. If $\\|x_0-x_{\\star}\\|^2\\le R^2$, then for every $N\\ge 2$,\n",
    "\n",
    "$$\n",
    "\\|A(x_N)\\|^2\\le \\frac{4L^2R^2}{N^2}.\n",
    "$$\n",
    "\n",
    "In normalized variables with $L=1$, the Lyapunov certificate is\n",
    "\n",
    "$$\n",
    "V_0=0,\n",
    "$$\n",
    "\n",
    "$$\n",
    "V_1=\\frac{2}{N^2}\\|x_0-x_1\\|^2-\\frac{2}{N^2}\\|A(x_0)-A(x_1)\\|^2,\n",
    "$$\n",
    "\n",
    "$$\n",
    "V_k=\\frac{2}{N^2}\\|x_0-x_k\\|^2-\\frac{2(k+1)^2}{N^2}\\|x_k-x_{k+1/2}\\|^2,\n",
    "\\qquad 2\\le k\\le N-1,\n",
    "$$\n",
    "\n",
    "and\n",
    "\n",
    "$$\n",
    "V_N=\\frac{4}{N^2}\\|x_0-x_{\\star}\\|^2.\n",
    "$$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3dcb76ad",
   "metadata": {},
   "source": [
    "### Proof outline\n",
    "\n",
    "Use the residuals\n",
    "\n",
    "$$\n",
    "M(u,v)=-\\langle u-v,A(u)-A(v)\\rangle\\le 0,\n",
    "$$\n",
    "\n",
    "and, in normalized units $L=1$,\n",
    "\n",
    "$$\n",
    "G(u,v)=\\|A(u)-A(v)\\|^2-\\|u-v\\|^2\\le 0.\n",
    "$$\n",
    "\n",
    "The base identity is\n",
    "\n",
    "$$\n",
    "V_1-V_0=-\\frac{2}{N^2}G(x_0,x_1).\n",
    "$$\n",
    "\n",
    "For $1\\le k\\le N-2$, the symbolic step identity is\n",
    "\n",
    "$$\n",
    "V_{k+1}-V_k\n",
    "=-\\frac{2(k+1)^2}{N^2}G(x_{k+1/2},x_{k+1})\n",
    "-\\frac{4k(k+1)}{N^2}M(x_k,x_{k+1}).\n",
    "$$\n",
    "\n",
    "The terminal transition is\n",
    "\n",
    "$$\n",
    "\\begin{aligned}\n",
    "V_N-V_{N-1}={}&-2G(x_{N-1/2},x_N)\n",
    "-\\frac{4(N-1)}{N}M(x_{N-1},x_N)\n",
    "-\\frac{4}{N}M(x_N,x_{\\star}) \\\\\n",
    "&+\\left\\|\\frac{2}{N}(x_0-x_{\\star})-A(x_N)\\right\\|^2\n",
    "+\\|A(x_N)\\|^2.\n",
    "\\end{aligned}\n",
    "$$\n",
    "\n",
    "Since $M\\le 0$, $G\\le 0$, and the square is nonnegative, telescoping gives\n",
    "\n",
    "$$\n",
    "\\frac{4}{N^2}\\|x_0-x_{\\star}\\|^2=V_N\\ge \\|A(x_N)\\|^2.\n",
    "$$\n",
    "\n",
    "Rescaling the normalized operator by $L$ yields $\\|A(x_N)\\|^2\\le 4L^2R^2/N^2$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "a96d0735",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:50.267351Z",
     "iopub.status.busy": "2026-05-13T16:57:50.267138Z",
     "iopub.status.idle": "2026-05-13T16:57:55.302615Z",
     "shell.execute_reply": "2026-05-13T16:57:55.301617Z"
    }
   },
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "import json\n",
    "import sys\n",
    "\n",
    "ROOT = Path.cwd()\n",
    "while not (ROOT / \"pyproject.toml\").exists() and ROOT != ROOT.parent:\n",
    "    ROOT = ROOT.parent\n",
    "if str(ROOT) not in sys.path:\n",
    "    sys.path.insert(0, str(ROOT))\n",
    "\n",
    "import matplotlib.pyplot as plt  # noqa: E402\n",
    "import pepflow as pf  # noqa: E402\n",
    "\n",
    "from examples.fast_extragradient.fast_extragradient_setup import (  # noqa: E402\n",
    "    A,\n",
    "    L,\n",
    "    alpha,\n",
    "    get_pep_setup,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "44a349e2",
   "metadata": {},
   "source": [
    "## Operator And Parameters\n",
    "\n",
    "The setup module uses a monotone operator with pairwise L-Lipschitz interpolation constraints."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "89704061",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.305523Z",
     "iopub.status.busy": "2026-05-13T16:57:55.304960Z",
     "iopub.status.idle": "2026-05-13T16:57:55.315079Z",
     "shell.execute_reply": "2026-05-13T16:57:55.313864Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(L, 1/L, A)"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "L, alpha, A"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a2062adb",
   "metadata": {},
   "source": [
    "## PEP Setup\n",
    "\n",
    "The code below mirrors the setup module so the iterate recursion, initial condition, and performance metric are visible in the notebook."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "f158d44c",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.318086Z",
     "iopub.status.busy": "2026-05-13T16:57:55.317817Z",
     "iopub.status.idle": "2026-05-13T16:57:55.324436Z",
     "shell.execute_reply": "2026-05-13T16:57:55.323669Z"
    }
   },
   "outputs": [],
   "source": [
    "def notebook_make_ctx_fast_extragradient(ctx_name: str, N, params) -> pf.PEPContext:\n",
    "    del params\n",
    "    ctx = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x0 = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    x0.add_tag(\"x_0_half\")\n",
    "    x = x0\n",
    "    A.set_zero_point(\"x_star\")\n",
    "\n",
    "    for k in range(int(N)):\n",
    "        Ax = A(x)\n",
    "        if k == 0:\n",
    "            x_half = x0\n",
    "        else:\n",
    "            x_half = x + (1 / (k + 1)) * (x0 - x) - (k / (k + 1)) * alpha * Ax\n",
    "            x_half.add_tag(f\"x_{k}_half\")\n",
    "\n",
    "        x = x + (1 / (k + 1)) * (x0 - x) - alpha * A(x_half)\n",
    "        x.add_tag(f\"x_{k + 1}\")\n",
    "\n",
    "    return ctx\n",
    "\n",
    "\n",
    "def notebook_get_pep_setup(N, params):\n",
    "    R = pf.Parameter(\"R\")\n",
    "    ctx = notebook_make_ctx_fast_extragradient(f\"notebook_ctx_{N}\", N, params)\n",
    "    pb = pf.PEPBuilder(ctx)\n",
    "    pb.add_initial_constraint(\n",
    "        ((ctx[\"x_0\"] - ctx[\"x_star\"]) ** 2).le(R**2, name=\"initial_condition\")\n",
    "    )\n",
    "    pb.set_performance_metric(A(ctx[f\"x_{N}\"]) ** 2)\n",
    "    return ctx, pb, A"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4eee0ac9",
   "metadata": {},
   "source": [
    "## Numerical Evidence\n",
    "\n",
    "The Block 1 sweep uses L=1 and R=1. The plotted candidate is 4/N^2, with the N=1 point left as its observed special case."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "433e54ac",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.327968Z",
     "iopub.status.busy": "2026-05-13T16:57:55.327542Z",
     "iopub.status.idle": "2026-05-13T16:57:55.512110Z",
     "shell.execute_reply": "2026-05-13T16:57:55.510585Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N=1: PEP=2.0000195262, candidate=2.0000000000, N^2*PEP=2.000020\n",
      "N=2: PEP=0.9999995446, candidate=1.0000000000, N^2*PEP=3.999998\n",
      "N=3: PEP=0.4444401868, candidate=0.4444444444, N^2*PEP=3.999962\n",
      "N=4: PEP=0.2499991745, candidate=0.2500000000, N^2*PEP=3.999987\n",
      "N=5: PEP=0.1600145857, candidate=0.1600000000, N^2*PEP=4.000365\n",
      "N=6: PEP=0.1111131266, candidate=0.1111111111, N^2*PEP=4.000073\n",
      "N=7: PEP=0.0816473050, candidate=0.0816326531, N^2*PEP=4.000718\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "state_path = (\n",
    "    ROOT / \"examples\" / \"fast_extragradient\" / \"state\" / \"fast_extragradient_b1.json\"\n",
    ")\n",
    "state = json.loads(state_path.read_text())\n",
    "results = state[\"sweep_results\"]\n",
    "Ns = [r[\"N\"] for r in results]\n",
    "values = [float(r[\"opt_value\"]) for r in results]\n",
    "candidate = [2.0 if N == 1 else 4.0 / (N * N) for N in Ns]\n",
    "\n",
    "for N, value, guess in zip(Ns, values, candidate):\n",
    "    print(\n",
    "        f\"N={N}: PEP={value:.10f}, candidate={guess:.10f}, N^2*PEP={N * N * value:.6f}\"\n",
    "    )\n",
    "\n",
    "plt.figure(figsize=(6, 4))\n",
    "plt.plot(Ns, values, \"o-\", label=\"PEP value\")\n",
    "plt.plot(Ns, candidate, \"--\", label=\"2 at N=1, 4/N^2 for N>=2\")\n",
    "plt.xlabel(\"N\")\n",
    "plt.ylabel(\"worst-case ||A(x_N)||^2\")\n",
    "plt.title(\"Fast Extragradient numerical PEP evidence\")\n",
    "plt.grid(True, alpha=0.3)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6b626fd3",
   "metadata": {},
   "source": [
    "## Dense And Relaxed Proof Solves\n",
    "\n",
    "At `N_verify = 4`, the dense solve matches the candidate value `4/N^2 = 1/4`. The dense dual certificate is already sparse, so the relaxed solve keeps only the structurally active monotone and Lipschitz constraints and drops the rest."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "fae3a561",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.514529Z",
     "iopub.status.busy": "2026-05-13T16:57:55.514279Z",
     "iopub.status.idle": "2026-05-13T16:57:55.523444Z",
     "shell.execute_reply": "2026-05-13T16:57:55.522120Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N_verify = 4\n",
      "dense objective  = 0.2499991745\n",
      "relaxed objective = 0.2500006838\n",
      "candidate 4/N^2  = 0.2500000000\n",
      "preserved optimum: True\n",
      "basis vectors:\n",
      "  x_0_half\n",
      "  x_star\n",
      "  A(x_0_half)\n",
      "  A(x_1)\n",
      "  A(x_1_half)\n",
      "  A(x_2)\n",
      "  A(x_2_half)\n",
      "  A(x_3)\n",
      "  A(x_3_half)\n",
      "  A(x_4)\n"
     ]
    }
   ],
   "source": [
    "b2_path = (\n",
    "    ROOT / \"examples\" / \"fast_extragradient\" / \"state\" / \"fast_extragradient_b2.json\"\n",
    ")\n",
    "dense_path = (\n",
    "    ROOT / \"examples\" / \"fast_extragradient\" / \"state\" / \"fast_extragradient_dense.json\"\n",
    ")\n",
    "relaxed_path = (\n",
    "    ROOT\n",
    "    / \"examples\"\n",
    "    / \"fast_extragradient\"\n",
    "    / \"state\"\n",
    "    / \"fast_extragradient_relaxed.json\"\n",
    ")\n",
    "\n",
    "b2 = json.loads(b2_path.read_text())\n",
    "dense = json.loads(dense_path.read_text())\n",
    "relaxed = json.loads(relaxed_path.read_text())\n",
    "\n",
    "N_int = b2[\"N_verify\"]\n",
    "print(f\"N_verify = {N_int}\")\n",
    "print(f\"dense objective  = {dense['opt_value']:.10f}\")\n",
    "print(f\"relaxed objective = {relaxed['opt_value']:.10f}\")\n",
    "print(f\"candidate 4/N^2  = {4 / N_int**2:.10f}\")\n",
    "print(f\"preserved optimum: {abs(dense['opt_value'] - relaxed['opt_value']) < 1e-5}\")\n",
    "print(\"basis vectors:\")\n",
    "for v in relaxed[\"basis_vectors\"]:\n",
    "    print(\" \", v)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "abcf84ed",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.526094Z",
     "iopub.status.busy": "2026-05-13T16:57:55.525709Z",
     "iopub.status.idle": "2026-05-13T16:57:55.532592Z",
     "shell.execute_reply": "2026-05-13T16:57:55.531023Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Monotone Operator Inequality\n",
      "  (x_1, x_2) -> 0.499999468\n",
      "  (x_2, x_3) -> 1.499998356\n",
      "  (x_3, x_4) -> 2.999996651\n",
      "  (x_4, x_star) -> 0.999999013\n",
      "Lipschitz Operator Inequality\n",
      "  (x_0_half, x_1) -> 0.124999868\n",
      "  (x_1_half, x_2) -> 0.499999461\n",
      "  (x_2_half, x_3) -> 1.124998762\n",
      "  (x_3_half, x_4) -> 1.999997798\n"
     ]
    }
   ],
   "source": [
    "def active_entries(group, tol=1e-7):\n",
    "    matrix = group[\"matrix\"]\n",
    "    rows = group[\"row_names\"]\n",
    "    cols = group[\"col_names\"]\n",
    "    out = []\n",
    "    for i, ri in enumerate(rows):\n",
    "        for j, ci in enumerate(cols):\n",
    "            value = float(matrix[i][j])\n",
    "            if abs(value) > tol:\n",
    "                out.append((ri, ci, value))\n",
    "    return out\n",
    "\n",
    "\n",
    "for group_name, group in relaxed[\"lambda_groups\"].items():\n",
    "    print(group_name)\n",
    "    for ri, ci, value in active_entries(group):\n",
    "        print(f\"  ({ri}, {ci}) -> {value:.9f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "26acaf0d",
   "metadata": {},
   "source": [
    "## Relaxed Lambda Formulas\n",
    "\n",
    "The relaxed certificate keeps the following closed-form coefficients:\n",
    "\n",
    "- Monotone constraints `(x_i, x_{i+1})` for `1 <= i <= N-1` with coefficient `4 i(i+1) / N^2`.\n",
    "- Monotone constraint `(x_N, x_star)` with coefficient `4 / N`.\n",
    "- Lipschitz constraints `(x_{i+1/2}, x_{i+1})` for `0 <= i <= N-1`, using `x_{0+1/2}=x_0`, with coefficient `2(i+1)^2 / N^2`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "63aeeea4",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.534903Z",
     "iopub.status.busy": "2026-05-13T16:57:55.534677Z",
     "iopub.status.idle": "2026-05-13T16:57:55.547717Z",
     "shell.execute_reply": "2026-05-13T16:57:55.546204Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Monotone lambda candidate\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_1 & x_1_half & x_2 & x_2_half & x_3 & x_3_half & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.5 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 & 1.5 & 0.0 & 0.0 & 0.0 \\\\x_2_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 3.0 & 0.0 \\\\x_3_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 1.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Lipschitz lambda candidate\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_1 & x_1_half & x_2 & x_2_half & x_3 & x_3_half & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0_half & 0.125 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1_half & 0.0 & 0.0 & 0.5 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_2_half & 0.0 & 0.0 & 0.0 & 0.0 & 1.125 & 0.0 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_3_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 2.0 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "monotone max residual: 3.3490080397768907e-06\n",
      "lipschitz max residual: 2.201866910356287e-06\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import sympy as sp\n",
    "\n",
    "exec(b2[\"lamb_code\"], globals())\n",
    "monotone_lamb = globals()[\"monotone_lamb\"]\n",
    "lipschitz_lamb = globals()[\"lipschitz_lamb\"]\n",
    "\n",
    "mono_rows = b2[\"monotone_lambda_row_names\"]\n",
    "mono_cols = b2[\"monotone_lambda_col_names\"]\n",
    "lip_rows = b2[\"lipschitz_lambda_row_names\"]\n",
    "lip_cols = b2[\"lipschitz_lambda_col_names\"]\n",
    "\n",
    "mono_candidate = np.array(\n",
    "    [[float(monotone_lamb(ri, ci)) for ci in mono_cols] for ri in mono_rows]\n",
    ")\n",
    "lip_candidate = np.array(\n",
    "    [[float(lipschitz_lamb(ri, ci)) for ci in lip_cols] for ri in lip_rows]\n",
    ")\n",
    "\n",
    "print(\"Monotone lambda candidate\")\n",
    "pf.pprint_labeled_matrix(mono_candidate, mono_rows, mono_cols, precision=3)\n",
    "print(\"Lipschitz lambda candidate\")\n",
    "pf.pprint_labeled_matrix(lip_candidate, lip_rows, lip_cols, precision=3)\n",
    "\n",
    "mono_relaxed = np.array(\n",
    "    relaxed[\"lambda_groups\"][\"Monotone Operator Inequality\"][\"matrix\"], dtype=float\n",
    ")\n",
    "lip_relaxed = np.array(\n",
    "    relaxed[\"lambda_groups\"][\"Lipschitz Operator Inequality\"][\"matrix\"], dtype=float\n",
    ")\n",
    "print(\"monotone max residual:\", np.max(np.abs(mono_candidate - mono_relaxed)))\n",
    "print(\"lipschitz max residual:\", np.max(np.abs(lip_candidate - lip_relaxed)))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7455c58c",
   "metadata": {},
   "source": [
    "## Closed-Form S Verification\n",
    "\n",
    "The Gram dual matrix is rank one. The direct decomposition is\n",
    "\n",
    "$$\n",
    "S = \\left(\\frac{2}{N}(x_0-x_\\star)-A(x_N)\\right)^2.\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a266c021",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.550293Z",
     "iopub.status.busy": "2026-05-13T16:57:55.550014Z",
     "iopub.status.idle": "2026-05-13T16:57:55.562456Z",
     "shell.execute_reply": "2026-05-13T16:57:55.561044Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "S rank: 1\n",
      "S max residual: 2.206142155847246e-06\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccc}\n",
       "         & x_0_half & x_\\star & A(x_0_half) & A(x_1) & A(x_1_half) & A(x_2) & A(x_2_half) & A(x_3) & A(x_3_half) & A(x_4) \\\\\n",
       "        \\hline\n",
       "        x_0_half & 0.25 & -0.25 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.5 \\\\x_\\star & -0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.5 \\\\A(x_0_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_1) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_1_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_2) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_2_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4) & -0.5 & 0.5 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 1.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "params_sp = {\"L\": sp.S(1), \"R\": sp.S(1)}\n",
    "ctx, pb, obj = get_pep_setup(sp.S(N_int), params_sp)\n",
    "pm = pf.ExpressionManager(ctx, resolve_parameters=params_sp)\n",
    "exec(b2[\"S_code\"], globals())\n",
    "S_guess = globals()[\"S_guess\"]\n",
    "\n",
    "S_candidate = np.array(pm.eval_scalar(S_guess).inner_prod_coords, dtype=float)\n",
    "S_relaxed = np.array(relaxed[\"S_matrix\"], dtype=float)\n",
    "print(\"S rank:\", np.linalg.matrix_rank(S_relaxed, tol=1e-7))\n",
    "print(\"S max residual:\", np.max(np.abs(S_candidate - S_relaxed)))\n",
    "pf.pprint_labeled_matrix(\n",
    "    S_candidate, relaxed[\"S_row_names\"], relaxed[\"S_col_names\"], precision=3\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2f357ab2",
   "metadata": {},
   "source": [
    "## Fixed-N Full Proof Identity\n",
    "\n",
    "For `N = N_verify`, the certificate verifies\n",
    "\n",
    "$$\n",
    "\\|A(x_N)\\|^2 - \\frac{4}{N^2}\\|x_0-x_\\star\\|^2 - \\sum \\lambda_{mon} g_{mon} - \\sum \\lambda_L g_L + S = 0,\n",
    "$$\n",
    "\n",
    "where each `g` is the corresponding interpolation inequality in `<= 0` form."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "c703ecfb",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.565130Z",
     "iopub.status.busy": "2026-05-13T16:57:55.564868Z",
     "iopub.status.idle": "2026-05-13T16:57:55.659169Z",
     "shell.execute_reply": "2026-05-13T16:57:55.657745Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "proof valid: True\n",
      "proof residual max abs: 1.6653345369377348e-16\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccc}\n",
       "         & x_0_half & x_\\star & A(x_0_half) & A(x_1) & A(x_1_half) & A(x_2) & A(x_2_half) & A(x_3) & A(x_3_half) & A(x_4) \\\\\n",
       "        \\hline\n",
       "        x_0_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_0_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & 0.0 & 0.0 \\\\A(x_1) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_1_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & 0.0 & 0.0 \\\\A(x_2) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_2_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3) & 0.0 & 0.0 & -0.0 & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import itertools\n",
    "\n",
    "\n",
    "def lipschitz_ineq(ri, ci):\n",
    "    d1 = ctx.get_duplet_by_point_tag(ri, obj)\n",
    "    d2 = ctx.get_duplet_by_point_tag(ci, obj)\n",
    "    return (d1.output - d2.output) ** 2 - (L**2) * (d1.point - d2.point) ** 2\n",
    "\n",
    "\n",
    "interp_sum = pf.Scalar.zero()\n",
    "for ri, ci in itertools.product(mono_rows, mono_cols):\n",
    "    c = monotone_lamb(ri, ci)\n",
    "    if c != 0:\n",
    "        interp_sum += c * obj.interp_ineq(ri, ci)\n",
    "for ri, ci in itertools.product(lip_rows, lip_cols):\n",
    "    c = lipschitz_lamb(ri, ci)\n",
    "    if c != 0:\n",
    "        interp_sum += c * lipschitz_ineq(ri, ci)\n",
    "\n",
    "x_N = ctx[f\"x_{N_int}\"]\n",
    "x_0 = ctx[\"x_0\"]\n",
    "x_star = ctx[\"x_star\"]\n",
    "tau = sp.Rational(4, N_int**2)\n",
    "residual = obj(x_N) ** 2 - tau * (x_0 - x_star) ** 2 - interp_sum + S_guess\n",
    "residual_matrix = np.array(pm.eval_scalar(residual).inner_prod_coords, dtype=float)\n",
    "\n",
    "print(\"proof valid:\", np.allclose(residual_matrix, 0, atol=1e-8))\n",
    "print(\"proof residual max abs:\", np.max(np.abs(residual_matrix)))\n",
    "pf.pprint_labeled_matrix(\n",
    "    residual_matrix, relaxed[\"S_row_names\"], relaxed[\"S_col_names\"], precision=3\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d92529f6",
   "metadata": {},
   "source": [
    "## Partial-Sum Lyapunov Construction And Rank Profile\n",
    "\n",
    "The full proof identity is converted into partial sums by adding the active inequalities in chronological order. For `0 <= k <= N-1`, the increment includes the Lipschitz inequality from the half iterate to the next full iterate. For `k >= 1`, it also includes the monotonicity inequality between consecutive full iterates:\n",
    "\n",
    "$$\n",
    "V_{k+1}-V_k = -\\lambda_L(x_{k+1/2},x_{k+1})\\,g_L(x_{k+1/2},x_{k+1})\n",
    "-\\mathbf{1}_{k\\ge 1}\\lambda_M(x_k,x_{k+1})\\,g_M(x_k,x_{k+1}).\n",
    "$$\n",
    "\n",
    "At the terminal step, the boundary monotonicity term `(x_N,x_star)` and the rank-one term `S + ||A(x_N)||^2` are added. With this grouping, the interior Lyapunov matrices have constant rank 2, while the final term is the rank-one initial-distance boundary."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "219d9bfb",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.661904Z",
     "iopub.status.busy": "2026-05-13T16:57:55.661556Z",
     "iopub.status.idle": "2026-05-13T16:57:55.670987Z",
     "shell.execute_reply": "2026-05-13T16:57:55.670065Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Stored active terms by step:\n",
      "  step 0: - 1/8 * lipschitz('x_0_half', 'x_1')\n",
      "  step 1: - 1/2 * lipschitz('x_1_half', 'x_2')\n",
      "  step 1: - 1/2 * monotone('x_1', 'x_2')\n",
      "  step 2: - 9/8 * lipschitz('x_2_half', 'x_3')\n",
      "  step 2: - 3/2 * monotone('x_2', 'x_3')\n",
      "  step 3: - 2 * lipschitz('x_3_half', 'x_4')\n",
      "  step 3: - 3 * monotone('x_3', 'x_4')\n",
      "  step 3: - 1 * monotone_boundary('x_4', 'x_star')\n",
      "  step 3: + S_guess + ||A(x_N)||^2\n"
     ]
    }
   ],
   "source": [
    "b3_path = (\n",
    "    ROOT / \"examples\" / \"fast_extragradient\" / \"state\" / \"fast_extragradient_b3.json\"\n",
    ")\n",
    "b3 = json.loads(b3_path.read_text())\n",
    "\n",
    "N_int = b3[\"N_verify\"]\n",
    "params_sp = {\"L\": sp.S(1), \"R\": sp.S(1)}\n",
    "ctx, pb, obj = get_pep_setup(sp.S(N_int), params_sp)\n",
    "pm = pf.ExpressionManager(ctx, resolve_parameters=params_sp)\n",
    "\n",
    "# The formulas are stored in Block 2/3 state and were displayed above.\n",
    "exec(b3[\"lamb_code\"], globals())\n",
    "\n",
    "print(\"Stored active terms by step:\")\n",
    "for term in b3[\"grouping_terms\"]:\n",
    "    if term[\"kind\"] == \"rank_one_boundary\":\n",
    "        print(f\"  step {term['step']}: + {term['expr']}\")\n",
    "    else:\n",
    "        pair = tuple(term[\"pair\"])\n",
    "        print(\n",
    "            f\"  step {term['step']}: {term['sign']} {term['coeff']} * {term['kind']}{pair}\"\n",
    "        )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "c8b26f43",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.673494Z",
     "iopub.status.busy": "2026-05-13T16:57:55.673016Z",
     "iopub.status.idle": "2026-05-13T16:57:55.726669Z",
     "shell.execute_reply": "2026-05-13T16:57:55.725029Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rank V_0: 0\n",
      "\n",
      "rank V_1: 2\n",
      "rank V_2: 2\n",
      "rank V_3: 2\n",
      "rank V_4: 1\n",
      "Interior rank is constant: True\n"
     ]
    }
   ],
   "source": [
    "def lipschitz_ineq(ri, ci):\n",
    "    d1 = ctx.get_duplet_by_point_tag(ri, obj)\n",
    "    d2 = ctx.get_duplet_by_point_tag(ci, obj)\n",
    "    return (d1.output - d2.output) ** 2 - (L**2) * (d1.point - d2.point) ** 2\n",
    "\n",
    "\n",
    "def half_tag(step):\n",
    "    return \"x_0_half\" if step == 0 else f\"x_{step}_half\"\n",
    "\n",
    "\n",
    "lyap = [pf.Scalar.zero()]\n",
    "partial_sum = pf.Scalar.zero()\n",
    "S_guess = (\n",
    "    sp.Rational(2, N_int) * (ctx[\"x_0\"] - ctx[\"x_star\"]) - obj(ctx[f\"x_{N_int}\"])\n",
    ") ** 2\n",
    "perf = obj(ctx[f\"x_{N_int}\"]) ** 2\n",
    "\n",
    "for step in range(N_int):\n",
    "    lip_pair = (half_tag(step), f\"x_{step + 1}\")\n",
    "    partial_sum -= lipschitz_lamb(*lip_pair) * lipschitz_ineq(*lip_pair)\n",
    "\n",
    "    if 1 <= step <= N_int - 1:\n",
    "        mon_pair = (f\"x_{step}\", f\"x_{step + 1}\")\n",
    "        partial_sum -= monotone_lamb(*mon_pair) * obj.interp_ineq(*mon_pair)\n",
    "\n",
    "    if step == N_int - 1:\n",
    "        star_pair = (f\"x_{N_int}\", \"x_star\")\n",
    "        partial_sum -= monotone_lamb(*star_pair) * obj.interp_ineq(*star_pair)\n",
    "        partial_sum += S_guess + perf\n",
    "\n",
    "    lyap.append(partial_sum)\n",
    "\n",
    "rank_tolerance = b3[\"rank_tolerance\"]\n",
    "ranks = []\n",
    "for k, Vk in enumerate(lyap):\n",
    "    matrix = pm.eval_scalar(Vk).inner_prod_coords.astype(float)\n",
    "    rank = int(np.linalg.matrix_rank(matrix, tol=rank_tolerance))\n",
    "    ranks.append(rank)\n",
    "    print(f\"rank V_{k}: {rank}\")\n",
    "    if k == 0:\n",
    "        print()\n",
    "\n",
    "print(\"Interior rank is constant:\", len(set(ranks[1:N_int])) == 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "d4d13ce9",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.729686Z",
     "iopub.status.busy": "2026-05-13T16:57:55.729315Z",
     "iopub.status.idle": "2026-05-13T16:57:55.736046Z",
     "shell.execute_reply": "2026-05-13T16:57:55.734564Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lyap[4] rank: 1\n",
      "stored rank profile: [0, 2, 2, 2, 1]\n",
      "computed rank profile: [0, 2, 2, 2, 1]\n",
      "matches stored profile: True\n"
     ]
    }
   ],
   "source": [
    "# Coverage check: the terminal partial sum is the boundary initial-distance term.\n",
    "terminal_matrix = pm.eval_scalar(lyap[N_int]).inner_prod_coords.astype(float)\n",
    "print(\n",
    "    f\"lyap[{N_int}] rank:\",\n",
    "    int(np.linalg.matrix_rank(terminal_matrix, tol=rank_tolerance)),\n",
    ")\n",
    "print(\"stored rank profile:\", b3[\"rank_profile\"])\n",
    "print(\"computed rank profile:\", ranks)\n",
    "print(\"matches stored profile:\", ranks == b3[\"rank_profile\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5693852d",
   "metadata": {},
   "source": [
    "## Identify the vectors composing the Lyapunov function\n",
    "\n",
    "Block 4 starts from the Block 3 partial sums and searches for interpretable vectors spanning each low-rank quadratic form. The interior terms have rank 2; the terminal term is the rank-one initial-distance boundary."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "e7a6babf",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.738388Z",
     "iopub.status.busy": "2026-05-13T16:57:55.738070Z",
     "iopub.status.idle": "2026-05-13T16:57:55.747971Z",
     "shell.execute_reply": "2026-05-13T16:57:55.746301Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "stored rank profile:"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " [0, 2, 2, 2, 1]\n",
      "computed rank profile: [0, 2, 2, 2, 1]\n",
      "rank profile matches: True\n"
     ]
    }
   ],
   "source": [
    "b4_path = (\n",
    "    ROOT / \"examples\" / \"fast_extragradient\" / \"state\" / \"fast_extragradient_b4.json\"\n",
    ")\n",
    "b4 = json.loads(b4_path.read_text())\n",
    "\n",
    "# Sanity check against Block 3 reconstruction.\n",
    "print(\"stored rank profile:\", b4[\"rank_profile\"])\n",
    "print(\"computed rank profile:\", ranks)\n",
    "print(\"rank profile matches:\", ranks == b4[\"rank_profile\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "12a9d1f8",
   "metadata": {},
   "source": [
    "### Candidate-vector scan\n",
    "\n",
    "The candidate scan uses tagged iterates, operator values, point-to-solution gaps, Lipschitz gaps, monotonicity gaps, and the rank-one terminal vector suggested by the full proof certificate. Duplicate and zero candidates are removed by evaluated coordinates."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "aaef37c2",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.751150Z",
     "iopub.status.busy": "2026-05-13T16:57:55.750751Z",
     "iopub.status.idle": "2026-05-13T16:57:55.788606Z",
     "shell.execute_reply": "2026-05-13T16:57:55.787532Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "candidate count:"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " 30\n"
     ]
    }
   ],
   "source": [
    "from pepflow.lyapunov_utils import vectors_in_column_space, decompose_rankr_symmetric\n",
    "\n",
    "candidate_pairs = []\n",
    "\n",
    "\n",
    "def add_candidate(label, vector):\n",
    "    coords = np.asarray(pm.eval_vector(vector).coords, dtype=float)\n",
    "    if np.linalg.norm(coords) < 1e-9:\n",
    "        return\n",
    "    for _, existing in candidate_pairs:\n",
    "        old = np.asarray(pm.eval_vector(existing).coords, dtype=float)\n",
    "        if np.allclose(coords, old, atol=1e-8):\n",
    "            return\n",
    "    candidate_pairs.append((label, vector))\n",
    "\n",
    "\n",
    "for i in range(N_int + 1):\n",
    "    add_candidate(f\"x_{i} - x_star\", ctx[f\"x_{i}\"] - ctx[\"x_star\"])\n",
    "    add_candidate(f\"A(x_{i})\", obj(ctx[f\"x_{i}\"]))\n",
    "\n",
    "for i in range(N_int):\n",
    "    half = \"x_0_half\" if i == 0 else f\"x_{i}_half\"\n",
    "    add_candidate(f\"x_{i + 1} - {half}\", ctx[f\"x_{i + 1}\"] - ctx[half])\n",
    "    add_candidate(f\"{half} - x_{i + 1}\", ctx[half] - ctx[f\"x_{i + 1}\"])\n",
    "    add_candidate(f\"A({half}) - A(x_{i + 1})\", obj(ctx[half]) - obj(ctx[f\"x_{i + 1}\"]))\n",
    "\n",
    "for i in range(1, N_int):\n",
    "    add_candidate(f\"x_0 - x_{i}\", ctx[\"x_0\"] - ctx[f\"x_{i}\"])\n",
    "    add_candidate(f\"x_{i} - x_{i}_half\", ctx[f\"x_{i}\"] - ctx[f\"x_{i}_half\"])\n",
    "    add_candidate(\n",
    "        f\"A(x_{i}) - A(x_{i + 1})\", obj(ctx[f\"x_{i}\"]) - obj(ctx[f\"x_{i + 1}\"])\n",
    "    )\n",
    "\n",
    "add_candidate(\"x_0 - x_star\", ctx[\"x_0\"] - ctx[\"x_star\"])\n",
    "add_candidate(\n",
    "    \"terminal square vector\",\n",
    "    sp.Rational(2, N_int) * (ctx[\"x_0\"] - ctx[\"x_star\"]) - obj(ctx[f\"x_{N_int}\"]),\n",
    ")\n",
    "\n",
    "candidates = [v for _, v in candidate_pairs]\n",
    "label_by_id = {id(v): label for label, v in candidate_pairs}\n",
    "print(\"candidate count:\", len(candidates))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "32151d48",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.790705Z",
     "iopub.status.busy": "2026-05-13T16:57:55.790496Z",
     "iopub.status.idle": "2026-05-13T16:57:55.896343Z",
     "shell.execute_reply": "2026-05-13T16:57:55.894147Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_1 column-space candidates:\n",
      "   A(x_0)\n",
      "   A(x_1)\n",
      "   x_1 - x_0_half\n",
      "   A(x_0_half) - A(x_1)\n",
      "   x_1 - x_1_half\n",
      "V_2 column-space candidates:\n",
      "   A(x_2)\n",
      "   x_0 - x_2\n",
      "   x_2 - x_2_half\n",
      "V_3 column-space candidates:\n",
      "   A(x_3)\n",
      "   x_0 - x_3\n",
      "   x_3 - x_3_half\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_4 column-space candidates:\n",
      "   x_0 - x_star\n"
     ]
    }
   ],
   "source": [
    "for k in range(1, N_int + 1):\n",
    "    in_col = vectors_in_column_space(\n",
    "        lyap[k],\n",
    "        candidates,\n",
    "        pep_context=ctx,\n",
    "        resolve_parameters=params_sp,\n",
    "        rtol=1e-5,\n",
    "        atol=1e-5,\n",
    "    )\n",
    "    print(f\"V_{k} column-space candidates:\")\n",
    "    for v in in_col:\n",
    "        print(\"  \", label_by_id.get(id(v), str(v)))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "61eb1e06",
   "metadata": {},
   "source": [
    "### Selected basis pattern\n",
    "\n",
    "The selected rank-spanning basis is\n",
    "\n",
    "$$\n",
    "V_1: \\quad [x_0-x_1,\\; A(x_0)-A(x_1)],\n",
    "$$\n",
    "\n",
    "$$\n",
    "V_k: \\quad [x_0-x_k,\\; x_k-x_{k+1/2}] \\quad (2\\le k\\le N-1),\n",
    "$$\n",
    "\n",
    "and the terminal boundary is\n",
    "\n",
    "$$\n",
    "V_N: \\quad [x_0-x_\\star].\n",
    "$$\n",
    "\n",
    "The `k=1` case is special because `x_{0.5}=x_0`, so the generic half-step gap would be degenerate there."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "cbf8d624",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.899026Z",
     "iopub.status.busy": "2026-05-13T16:57:55.898787Z",
     "iopub.status.idle": "2026-05-13T16:57:55.906645Z",
     "shell.execute_reply": "2026-05-13T16:57:55.905342Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: rank 2 basis ['x_0 - x_1', 'A(x_0) - A(x_1)']\n",
      "k=2: rank 2 basis ['x_0 - x_2', 'x_2 - x_2_half']\n",
      "k=3: rank 2 basis ['x_0 - x_3', 'x_3 - x_3_half']\n",
      "k=4: rank 1 basis ['x_0 - x_star']\n"
     ]
    }
   ],
   "source": [
    "exec(b4[\"basis_code\"], globals())\n",
    "V_k_basis = globals()[\"V_k_basis\"]\n",
    "V_k_basis_labels = globals()[\"V_k_basis_labels\"]\n",
    "\n",
    "for k in range(1, N_int + 1):\n",
    "    basis = V_k_basis(k)\n",
    "    labels = V_k_basis_labels(k)\n",
    "    matrix = np.column_stack(\n",
    "        [np.asarray(pm.eval_vector(v).coords, dtype=float) for v in basis]\n",
    "    )\n",
    "    rank = int(np.linalg.matrix_rank(matrix, tol=1e-8))\n",
    "    print(f\"k={k}: rank {rank} basis {labels}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0ae4a08f",
   "metadata": {},
   "source": [
    "### Coefficient matrices\n",
    "\n",
    "For the selected basis order, the coefficient pattern is diagonal:\n",
    "\n",
    "$$\n",
    "C_1 = \\operatorname{diag}\\left(\\frac{2}{N^2},-\\frac{2}{N^2}\\right),\n",
    "$$\n",
    "\n",
    "$$\n",
    "C_k = \\operatorname{diag}\\left(\\frac{2}{N^2},-\\frac{2(k+1)^2}{N^2}\\right) \\quad (2\\le k\\le N-1),\n",
    "$$\n",
    "\n",
    "and\n",
    "\n",
    "$$\n",
    "C_N = \\left[\\frac{4}{N^2}\\right].\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "021971d4",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:55.911079Z",
     "iopub.status.busy": "2026-05-13T16:57:55.910560Z",
     "iopub.status.idle": "2026-05-13T16:57:56.008210Z",
     "shell.execute_reply": "2026-05-13T16:57:56.006844Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: formula residual 8.327e-17\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & x_0 - x_1 & A(x_0) - A(x_1) \\\\\n",
       "        \\hline\n",
       "        x_0 - x_1 & 0.125 & 0.0 \\\\A(x_0) - A(x_1) & 0.0 & -0.125 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=2: formula residual 2.220e-16\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & x_0 - x_2 & x_2 - x_2_half \\\\\n",
       "        \\hline\n",
       "        x_0 - x_2 & 0.125 & 0.0 \\\\x_2 - x_2_half & 0.0 & -1.125 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=3: formula residual 5.274e-16\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & x_0 - x_3 & x_3 - x_3_half \\\\\n",
       "        \\hline\n",
       "        x_0 - x_3 & 0.125 & 0.0 \\\\x_3 - x_3_half & 0.0 & -2.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=4: formula residual 0.000e+00"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|c}\n",
       "         & x_0 - x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 - x_\\star & 0.25 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "exec(b4[\"coeff_code\"], globals())\n",
    "coeff_pattern = globals()[\"coeff_pattern\"]\n",
    "\n",
    "for k in range(1, N_int + 1):\n",
    "    basis = V_k_basis(k)\n",
    "    labels = V_k_basis_labels(k)\n",
    "    C = decompose_rankr_symmetric(\n",
    "        lyap[k], basis, pep_context=ctx, resolve_parameters=params_sp\n",
    "    )\n",
    "    C_formula = np.array(coeff_pattern(k, N_int), dtype=float)\n",
    "    residual = np.max(np.abs(C - C_formula)) if C.size else 0.0\n",
    "    print(f\"k={k}: formula residual {residual:.3e}\")\n",
    "    pf.pprint_labeled_matrix(C_formula, labels, labels, precision=3)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "55486828",
   "metadata": {},
   "source": [
    "### Block 4 conclusion\n",
    "\n",
    "The current closed-form candidate is therefore a rank-2 interior Lyapunov sequence with a special base case at `k=1` and a rank-one terminal boundary. Block 5 will symbolically verify the step, base, and boundary identities for these formulas."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "70a194e6",
   "metadata": {},
   "source": [
    "## Symbolic Step Recursion Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1003064b",
   "metadata": {},
   "source": [
    "The identity verified below is\n",
    "\n",
    "$$\n",
    "V_{k+1}-V_k\n",
    "+\\frac{2(k+1)^2}{N^2}G(x_{k+1/2},x_{k+1})\n",
    "+\\frac{4k(k+1)}{N^2}M(x_k,x_{k+1})=0.\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-0$ should simplify to the zero matrix."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "67333efc",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:56.011472Z",
     "iopub.status.busy": "2026-05-13T16:57:56.011194Z",
     "iopub.status.idle": "2026-05-13T16:57:56.271620Z",
     "shell.execute_reply": "2026-05-13T16:57:56.270859Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Step identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_0 & x_k & A_k & A_{k+1/2} & A_{k+1} \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_k & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A_k & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A_{k+1/2} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A_{k+1} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "k_sym = sp.Symbol(\"k\", positive=True)\n",
    "N_sym = sp.Symbol(\"N\", positive=True)\n",
    "ctx_step = pf.PEPContext(\"symbolic_step\").set_as_current()\n",
    "x0_step = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "xk = pf.Vector(is_basis=True, tags=[\"x_k\"])\n",
    "Ak = pf.Vector(is_basis=True, tags=[\"A_k\"])\n",
    "Akh = pf.Vector(is_basis=True, tags=[\"A_{k+1/2}\"])\n",
    "Akp1 = pf.Vector(is_basis=True, tags=[\"A_{k+1}\"])\n",
    "\n",
    "xkh = xk + sp.Rational(1, 1) / (k_sym + 1) * (x0_step - xk) - k_sym / (k_sym + 1) * Ak\n",
    "xkp1 = xk + sp.Rational(1, 1) / (k_sym + 1) * (x0_step - xk) - Akh\n",
    "xkp1h = (\n",
    "    xkp1\n",
    "    + sp.Rational(1, 1) / (k_sym + 2) * (x0_step - xkp1)\n",
    "    - (k_sym + 1) / (k_sym + 2) * Akp1\n",
    ")\n",
    "\n",
    "Vk = (\n",
    "    sp.Rational(2, 1) / N_sym**2 * (x0_step - xk) ** 2\n",
    "    - (sp.Rational(2, 1) * (k_sym + 1) ** 2 / N_sym**2) * (xk - xkh) ** 2\n",
    ")\n",
    "Vkp1 = (\n",
    "    sp.Rational(2, 1) / N_sym**2 * (x0_step - xkp1) ** 2\n",
    "    - (sp.Rational(2, 1) * (k_sym + 2) ** 2 / N_sym**2) * (xkp1 - xkp1h) ** 2\n",
    ")\n",
    "G_step = (Akh - Akp1) ** 2 - (xkh - xkp1) ** 2\n",
    "M_step = -(xk - xkp1) * (Ak - Akp1)\n",
    "\n",
    "step_residual = (\n",
    "    Vkp1\n",
    "    - Vk\n",
    "    + sp.Rational(2, 1) * (k_sym + 1) ** 2 / N_sym**2 * G_step\n",
    "    + sp.Rational(4, 1) * k_sym * (k_sym + 1) / N_sym**2 * M_step\n",
    ")\n",
    "pm_step = pf.ExpressionManager(ctx_step, resolve_parameters={})\n",
    "step_matrix = sp.Matrix(pm_step.eval_scalar(step_residual).inner_prod_coords).applyfunc(\n",
    "    sp.simplify\n",
    ")\n",
    "print(\"Step identity zero:\", step_matrix == sp.zeros(*step_matrix.shape))\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(step_matrix.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_step.basis_vectors()],\n",
    "    precision=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b26e59f8",
   "metadata": {},
   "source": [
    "## Base Case and Boundary Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cc3f2a68",
   "metadata": {},
   "source": [
    "The base identity verified below is\n",
    "\n",
    "$$\n",
    "V_1-V_0+\\frac{2}{N^2}G(x_0,x_1)=0.\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-0$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "a34dc101",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:56.273957Z",
     "iopub.status.busy": "2026-05-13T16:57:56.273748Z",
     "iopub.status.idle": "2026-05-13T16:57:56.283691Z",
     "shell.execute_reply": "2026-05-13T16:57:56.282678Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Base identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccc}\n",
       "         & x_0 & A_0 & A_1 \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 0.0 \\\\A_0 & 0.0 & 0.0 & 0.0 \\\\A_1 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_sym = sp.Symbol(\"N\", positive=True)\n",
    "ctx_base = pf.PEPContext(\"symbolic_base\").set_as_current()\n",
    "x0_base = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "A0 = pf.Vector(is_basis=True, tags=[\"A_0\"])\n",
    "A1 = pf.Vector(is_basis=True, tags=[\"A_1\"])\n",
    "x1_base = x0_base - A0\n",
    "\n",
    "V1_base = (\n",
    "    sp.Rational(2, 1) / N_sym**2 * (x0_base - x1_base) ** 2\n",
    "    - sp.Rational(2, 1) / N_sym**2 * (A0 - A1) ** 2\n",
    ")\n",
    "G_base = (A0 - A1) ** 2 - (x0_base - x1_base) ** 2\n",
    "base_residual = V1_base + sp.Rational(2, 1) / N_sym**2 * G_base\n",
    "pm_base = pf.ExpressionManager(ctx_base, resolve_parameters={})\n",
    "base_matrix = sp.Matrix(pm_base.eval_scalar(base_residual).inner_prod_coords).applyfunc(\n",
    "    sp.simplify\n",
    ")\n",
    "print(\"Base identity zero:\", base_matrix == sp.zeros(*base_matrix.shape))\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(base_matrix.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_base.basis_vectors()],\n",
    "    precision=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "912a2970",
   "metadata": {},
   "source": [
    "### Boundary Identity Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9d5c4fbc",
   "metadata": {},
   "source": [
    "The terminal transition verified below is\n",
    "\n",
    "$$\n",
    "\\begin{aligned}\n",
    "0={}&V_N-V_{N-1}+2G(x_{N-1/2},x_N)\n",
    "+\\frac{4(N-1)}{N}M(x_{N-1},x_N)\n",
    "+\\frac{4}{N}M(x_N,x_{\\star}) \\\\\n",
    "&-\\left\\|\\frac{2}{N}(x_0-x_{\\star})-A(x_N)\\right\\|^2-\\|A(x_N)\\|^2.\n",
    "\\end{aligned}\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "7594fb20",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:56.286065Z",
     "iopub.status.busy": "2026-05-13T16:57:56.285855Z",
     "iopub.status.idle": "2026-05-13T16:57:56.382692Z",
     "shell.execute_reply": "2026-05-13T16:57:56.381667Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Boundary transition identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & x_0 & x_\\star & x_{N-1} & A_{N-1} & A_{N-1/2} & A_N \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{N-1} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A_{N-1} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A_{N-1/2} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A_N & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_sym = sp.Symbol(\"N\", positive=True)\n",
    "ctx_boundary = pf.PEPContext(\"symbolic_boundary\").set_as_current()\n",
    "x0_boundary = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "xs_boundary = pf.Vector(is_basis=True, tags=[\"x_star\"])\n",
    "xm = pf.Vector(is_basis=True, tags=[\"x_{N-1}\"])\n",
    "Am = pf.Vector(is_basis=True, tags=[\"A_{N-1}\"])\n",
    "Amh = pf.Vector(is_basis=True, tags=[\"A_{N-1/2}\"])\n",
    "AN = pf.Vector(is_basis=True, tags=[\"A_N\"])\n",
    "\n",
    "xmh = xm + sp.Rational(1, 1) / N_sym * (x0_boundary - xm) - (N_sym - 1) / N_sym * Am\n",
    "xN_boundary = xm + sp.Rational(1, 1) / N_sym * (x0_boundary - xm) - Amh\n",
    "\n",
    "VNm1 = (\n",
    "    sp.Rational(2, 1) / N_sym**2 * (x0_boundary - xm) ** 2\n",
    "    - sp.Rational(2, 1) * (xm - xmh) ** 2\n",
    ")\n",
    "VN_boundary = sp.Rational(4, 1) / N_sym**2 * (x0_boundary - xs_boundary) ** 2\n",
    "G_boundary = (Amh - AN) ** 2 - (xmh - xN_boundary) ** 2\n",
    "M_boundary = -(xm - xN_boundary) * (Am - AN)\n",
    "M_star = -(xN_boundary - xs_boundary) * AN\n",
    "S_boundary = (sp.Rational(2, 1) / N_sym * (x0_boundary - xs_boundary) - AN) ** 2\n",
    "perf_boundary = AN**2\n",
    "\n",
    "boundary_residual = (\n",
    "    VN_boundary\n",
    "    - VNm1\n",
    "    + sp.Rational(2, 1) * G_boundary\n",
    "    + sp.Rational(4, 1) * (N_sym - 1) / N_sym * M_boundary\n",
    "    + sp.Rational(4, 1) / N_sym * M_star\n",
    "    - S_boundary\n",
    "    - perf_boundary\n",
    ")\n",
    "pm_boundary = pf.ExpressionManager(ctx_boundary, resolve_parameters={})\n",
    "boundary_matrix = sp.Matrix(\n",
    "    pm_boundary.eval_scalar(boundary_residual).inner_prod_coords\n",
    ").applyfunc(sp.simplify)\n",
    "print(\n",
    "    \"Boundary transition identity zero:\",\n",
    "    boundary_matrix == sp.zeros(*boundary_matrix.shape),\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(boundary_matrix.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_boundary.basis_vectors()],\n",
    "    precision=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "21d98e1c",
   "metadata": {},
   "source": [
    "### Terminal Bound Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d8a221f4",
   "metadata": {},
   "source": [
    "The final boundary form is\n",
    "\n",
    "$$\n",
    "V_N-\\frac{4}{N^2}\\|x_0-x_{\\star}\\|^2=0.\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "c8a3b4ee",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:57:56.385570Z",
     "iopub.status.busy": "2026-05-13T16:57:56.385341Z",
     "iopub.status.idle": "2026-05-13T16:57:56.394306Z",
     "shell.execute_reply": "2026-05-13T16:57:56.392477Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Terminal bound identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & x_0 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_sym = sp.Symbol(\"N\", positive=True)\n",
    "ctx_terminal = pf.PEPContext(\"symbolic_terminal\").set_as_current()\n",
    "x0_terminal = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "xs_terminal = pf.Vector(is_basis=True, tags=[\"x_star\"])\n",
    "VN_terminal = sp.Rational(4, 1) / N_sym**2 * (x0_terminal - xs_terminal) ** 2\n",
    "terminal_residual = (\n",
    "    VN_terminal - sp.Rational(4, 1) / N_sym**2 * (x0_terminal - xs_terminal) ** 2\n",
    ")\n",
    "pm_terminal = pf.ExpressionManager(ctx_terminal, resolve_parameters={})\n",
    "terminal_matrix = sp.Matrix(\n",
    "    pm_terminal.eval_scalar(terminal_residual).inner_prod_coords\n",
    ").applyfunc(sp.simplify)\n",
    "print(\n",
    "    \"Terminal bound identity zero:\", terminal_matrix == sp.zeros(*terminal_matrix.shape)\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(terminal_matrix.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_terminal.basis_vectors()],\n",
    "    precision=3,\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "pepflow (3.11.13)",
   "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.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
