{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "c6e4d0cc",
   "metadata": {},
   "source": [
    "# Optimized Gradient Method\n",
    "\n",
    "We consider a convex and L-smooth function f, a stationary point x_star with grad f(x_star) = 0, and an initial point satisfying ||x_0 - x_star|| <= R. The performance metric is f(x_N) - f(x_star).\n",
    "\n",
    "For a fixed horizon N, define theta_{-1}=0, theta_N = (1/2)(1 + sqrt(8 theta_{N-1}^2 + 1)), and for 0 <= k < N,\n",
    "\n",
    "$$\\theta_k = \\frac{1}{2}\\left(1 + \\sqrt{4\\theta_{k-1}^2 + 1}\\right).$$\n",
    "\n",
    "The optimized gradient method with fixed step size 1/L is\n",
    "\n",
    "$$y_k = x_k - \\frac{1}{L}\\nabla f(x_k),$$\n",
    "$$z_{k+1} = z_k - \\frac{2}{L}\\theta_k \\nabla f(x_k),$$\n",
    "$$x_{k+1} = \\left(1 - \\frac{1}{\\theta_{k+1}}\\right)y_k + \\frac{1}{\\theta_{k+1}}z_{k+1},$$\n",
    "\n",
    "with z_0 = x_0."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d35b32ef",
   "metadata": {},
   "source": [
    "## Proof Statement"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a508f12c",
   "metadata": {},
   "source": [
    "### Theorem\n",
    "\n",
    "Let $f$ be convex and $L$-smooth, let $\\nabla f(x_*)=0$, and assume $\\|x_{0}-x_*\\|\\le R$. For OGM with the horizon-dependent coefficients\n",
    "\n",
    "$$\\theta_{-1}=0,\\qquad \\theta_k=\\frac{1+\\sqrt{4\\theta_{k-1}^{2}+1}}{2}\\quad (0\\le k<N),\\qquad \\theta_N=\\frac{1+\\sqrt{8\\theta_{N-1}^{2}+1}}{2},$$\n",
    "\n",
    "the iterates satisfy\n",
    "\n",
    "$$f(x_N)-f(x_*)\\le \\frac{L R^2}{2\\theta_N^2}.$$\n",
    "\n",
    "A certificate is given by the following Lyapunov terms. Set\n",
    "\n",
    "$$\\alpha_k=\\frac{2\\theta_{k-1}^{2}}{\\theta_N^2}\\quad (1\\le k<N).$$\n",
    "\n",
    "For $1\\le k<N$,\n",
    "\n",
    "$$\\begin{aligned}\n",
    "V_k={}&f(x_N)-\\alpha_k f(x_k)-(1-\\alpha_k)f(x_*)\n",
    "-\\frac{2\\theta_k}{\\theta_N^2}\\langle x_k-x_*,\\nabla f(x_k)\\rangle\n",
    "+\\frac{\\theta_k(\\theta_k+1)}{L\\theta_N^2}\\|\\nabla f(x_k)\\|^2\\\\\n",
    "&-\\frac{L}{2\\theta_N^2}\\|z_{k+1}-x_*\\|^2.\n",
    "\\end{aligned}$$\n",
    "\n",
    "The boundary terms are\n",
    "\n",
    "$$V_0=f(x_N)-f(x_*)-\\frac{L}{2\\theta_N^2}\\|x_0-x_*\\|^2,$$\n",
    "\n",
    "and\n",
    "\n",
    "$$V_N=-\\frac{L}{2\\theta_N^2}\\left\\|z_N-\\frac{\\theta_N}{L}\\nabla f(x_N)-x_*\\right\\|^2.$$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f4d603d8",
   "metadata": {},
   "source": [
    "### Proof outline\n",
    "\n",
    "For smooth convex interpolation, define\n",
    "\n",
    "$$I_f(u,v)=f(v)-f(u)+\\langle \\nabla f(v),u-v\\rangle+\\frac{1}{2L}\\|\\nabla f(u)-\\nabla f(v)\\|^2\\le 0.$$\n",
    "\n",
    "The base identity is\n",
    "\n",
    "$$V_1-V_0+\\frac{2}{\\theta_N^2}I_f(x_0,x_1)+\\frac{2}{\\theta_N^2}I_f(x_*,x_0)=0.$$\n",
    "\n",
    "For $1\\le k<N-1$, the step identity is\n",
    "\n",
    "$$V_{k+1}-V_k+\\frac{2\\theta_k^2}{\\theta_N^2}I_f(x_k,x_{k+1})+\\frac{2\\theta_k}{\\theta_N^2}I_f(x_*,x_k)=0.$$\n",
    "\n",
    "The final boundary square is\n",
    "\n",
    "$$V_N+\\frac{L}{2\\theta_N^2}\\left\\|z_N-\\frac{\\theta_N}{L}\\nabla f(x_N)-x_*\\right\\|^2=0.$$\n",
    "\n",
    "Since all displayed multipliers are nonnegative and $I_f(u,v)\\le 0$, the sequence moves from $V_0$ to the nonpositive terminal square by adding nonnegative quantities. Hence $V_0\\le 0$. Combining this with the initial condition gives\n",
    "\n",
    "$$f(x_N)-f(x_*)\\le \\frac{L}{2\\theta_N^2}\\|x_0-x_*\\|^2\\le \\frac{L R^2}{2\\theta_N^2}.$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "bae22c00",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:41.612220Z",
     "iopub.status.busy": "2026-05-13T17:17:41.611859Z",
     "iopub.status.idle": "2026-05-13T17:17:47.045461Z",
     "shell.execute_reply": "2026-05-13T17:17:47.044165Z"
    }
   },
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "import sys\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "ROOT = Path.cwd()\n",
    "while ROOT != ROOT.parent and not (ROOT / \"pyproject.toml\").exists():\n",
    "    ROOT = ROOT.parent\n",
    "if str(ROOT) not in sys.path:\n",
    "    sys.path.insert(0, str(ROOT))\n",
    "\n",
    "import pepflow as pf  # noqa: E402"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "51956b6d",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.048761Z",
     "iopub.status.busy": "2026-05-13T17:17:47.048166Z",
     "iopub.status.idle": "2026-05-13T17:17:47.062401Z",
     "shell.execute_reply": "2026-05-13T17:17:47.060970Z"
    }
   },
   "outputs": [],
   "source": [
    "L = pf.Parameter(\"L\")\n",
    "R = pf.Parameter(\"R\")\n",
    "f = pf.SmoothConvexFunction(is_basis=True, tags=[\"f\"], L=L)\n",
    "\n",
    "\n",
    "def theta_ogm(i, N, cache=None):\n",
    "    if cache is None:\n",
    "        cache = {}\n",
    "    key = (i, N)\n",
    "    if key in cache:\n",
    "        return cache[key]\n",
    "    if i == -1:\n",
    "        value = 0.0\n",
    "    elif i == N:\n",
    "        value = 0.5 * (1 + np.sqrt(8 * theta_ogm(N - 1, N, cache) ** 2 + 1))\n",
    "    else:\n",
    "        value = 0.5 * (1 + np.sqrt(4 * theta_ogm(i - 1, N, cache) ** 2 + 1))\n",
    "    cache[key] = value\n",
    "    return value\n",
    "\n",
    "\n",
    "def make_ctx_ogm(ctx_name, N):\n",
    "    ctx = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    z = x\n",
    "    f.set_stationary_point(\"x_star\")\n",
    "\n",
    "    for k in range(int(N)):\n",
    "        grad_x = f.grad(x)\n",
    "        y = x - (1 / L) * grad_x\n",
    "        y.add_tag(f\"y_{k}\")\n",
    "        z = z - (2 / L) * theta_ogm(k, int(N)) * grad_x\n",
    "        z.add_tag(f\"z_{k + 1}\")\n",
    "        x = (1 - 1 / theta_ogm(k + 1, int(N))) * y + (1 / theta_ogm(k + 1, int(N))) * z\n",
    "        x.add_tag(f\"x_{k + 1}\")\n",
    "\n",
    "    return ctx\n",
    "\n",
    "\n",
    "def get_pep_setup(N):\n",
    "    ctx = make_ctx_ogm(f\"ctx_{N}\", N)\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(f(ctx[f\"x_{int(N)}\"]) - f(ctx[\"x_star\"]))\n",
    "    return ctx, pb, f"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "8f75550e",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.065120Z",
     "iopub.status.busy": "2026-05-13T17:17:47.064699Z",
     "iopub.status.idle": "2026-05-13T17:17:47.263435Z",
     "shell.execute_reply": "2026-05-13T17:17:47.262065Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N=1: PEP=0.1249989122, L*R^2/(2*theta_N^2)=0.1250000000\n",
      "N=2: PEP=0.0618918528, L*R^2/(2*theta_N^2)=0.0618941824\n",
      "N=3: PEP=0.0376920438, L*R^2/(2*theta_N^2)=0.0376923972\n",
      "N=4: PEP=0.0255836264, L*R^2/(2*theta_N^2)=0.0255839420\n",
      "N=5: PEP=0.0185869483, L*R^2/(2*theta_N^2)=0.0185881367\n",
      "N=6: PEP=0.0141553929, L*R^2/(2*theta_N^2)=0.0141559659\n",
      "N=7: PEP=0.0111568459, L*R^2/(2*theta_N^2)=0.0111604169\n"
     ]
    },
    {
     "data": {
      "image/png": "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VT1JoLZ27VqYM3mdXD9myMHBQeV/MPexRM8995yaVSVfrtHR0WpVVAnKRowYAXM2adIkdOnSRXXH3H///QgLC8PChQvVRlA/biQIke5eaU3UhJ6q1MaNG/VSzddvo0aNMuuaL6lOZFu8eLHe3I0ZM0Zft25dvY2Njd7Ly0t/++236//66y+ti2WUevbsqZ84caLe3A0fPlzv5+enXjMBAQHq+pkzZ7QullH4/fff9c2bN9fb2trqQ0JC9AsXLtS6SEZj7dq16nP35MmTmpVBJ/8wIiQiIqLqxoGpREREpAkGIURERKQJBiFERESkCQYhREREpAkGIURERKQJBiFERESkCQYhREREpAkGIURERKQJBiFEZBJkmXGdTodZs2bdsAy57Cci08MghIhMhp2dHWbPno0rV65oXRQiMgAGIURkMvr06QNfX1/MnDlT66IQkQEwCCEikyErw8qKqB9++CEuXryodXGI6BYxCCEikzJ06FC0atVKLVdPRKaNQQgRmRwZF/LVV1/h+PHjWheFiG4BgxAiMjk9evRAv379MHXqVK2LQkS3wOpW7kxEpBWZqivdMo0bN+aTQGSi2BJCRCapRYsWeOihhzB//nyti0JElcQghIhM1htvvIH8/Hyti0FElaTT6/X6yt6ZiIiIqLLYEkJERESaYBBCREREmmAQQkRERJpgEEJERESaYBBCREREmmAQQkRERJpgEEJERESaYBBCREREmmAQQkRERJpgEEJERESaYBBCREREmmAQQkRERNDC/wO86OKRnpg2KQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sweep_results = [\n",
    "    {\"N\": 1, \"opt_value\": 0.12499891224858557, \"rate_value\": 0.125},\n",
    "    {\"N\": 2, \"opt_value\": 0.0618918527616518, \"rate_value\": 0.06189418239776468},\n",
    "    {\"N\": 3, \"opt_value\": 0.03769204376644394, \"rate_value\": 0.03769239720788239},\n",
    "    {\"N\": 4, \"opt_value\": 0.025583626393635102, \"rate_value\": 0.025583942049932206},\n",
    "    {\"N\": 5, \"opt_value\": 0.01858694825943362, \"rate_value\": 0.01858813666365106},\n",
    "    {\"N\": 6, \"opt_value\": 0.014155392931958322, \"rate_value\": 0.014155965863218723},\n",
    "    {\"N\": 7, \"opt_value\": 0.01115684589824191, \"rate_value\": 0.01116041688775744},\n",
    "]\n",
    "\n",
    "for row in sweep_results:\n",
    "    print(\n",
    "        f\"N={row['N']}: PEP={row['opt_value']:.10f}, L*R^2/(2*theta_N^2)={row['rate_value']:.10f}\"\n",
    "    )\n",
    "\n",
    "Ns = [row[\"N\"] for row in sweep_results]\n",
    "pep_values = [row[\"opt_value\"] for row in sweep_results]\n",
    "rate_values = [row[\"rate_value\"] for row in sweep_results]\n",
    "\n",
    "plt.figure(figsize=(6, 4))\n",
    "plt.plot(Ns, pep_values, \"o-\", label=\"PEP value\")\n",
    "plt.plot(Ns, rate_values, \"s--\", label=\"L R^2 / (2 theta_N^2)\")\n",
    "plt.xlabel(\"N\")\n",
    "plt.ylabel(\"Worst-case value\")\n",
    "plt.grid(True, alpha=0.3)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "575f0817",
   "metadata": {},
   "source": [
    "## Dense and Relaxed Proof Solves\n",
    "\n",
    "This section records the Block 2 full-PEP certificate search at `N=4`. The dense certificate matches the candidate value, and the relaxed certificate keeps only consecutive interpolation inequalities plus the `x_star` row."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e1c61b5f",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.266764Z",
     "iopub.status.busy": "2026-05-13T17:17:47.266247Z",
     "iopub.status.idle": "2026-05-13T17:17:47.276768Z",
     "shell.execute_reply": "2026-05-13T17:17:47.275308Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dense objective:"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " 0.025583626393635102\n",
      "Relaxed objective: 0.025582264054418824\n",
      "Preserved: True\n",
      "Dropped constraints: 26\n",
      "Basis vectors: ['x_0', 'x_star', 'grad_f(x_0)', 'grad_f(x_1)', 'grad_f(x_2)', 'grad_f(x_3)', 'grad_f(x_4)']\n"
     ]
    }
   ],
   "source": [
    "import importlib.util\n",
    "import itertools\n",
    "import json\n",
    "import sympy as sp\n",
    "\n",
    "state_dir = ROOT / \"examples\" / \"ogm\" / \"state\"\n",
    "dense_data = json.loads((state_dir / \"ogm_dense.json\").read_text())\n",
    "relaxed_data = json.loads((state_dir / \"ogm_relaxed.json\").read_text())\n",
    "b2_state = json.loads((state_dir / \"ogm_b2.json\").read_text())\n",
    "\n",
    "print(\"Dense objective:\", dense_data[\"opt_value\"])\n",
    "print(\"Relaxed objective:\", relaxed_data[\"opt_value\"])\n",
    "print(\"Preserved:\", abs(dense_data[\"opt_value\"] - relaxed_data[\"opt_value\"]) < 1e-5)\n",
    "print(\"Dropped constraints:\", len(relaxed_data[\"relaxed_constraints\"]))\n",
    "print(\"Basis vectors:\", relaxed_data[\"basis_vectors\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "ad509d57",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.279700Z",
     "iopub.status.busy": "2026-05-13T17:17:47.279370Z",
     "iopub.status.idle": "2026-05-13T17:17:47.286299Z",
     "shell.execute_reply": "2026-05-13T17:17:47.284846Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Active relaxed lambda entries:\n",
      "  lambda(x_0, x_1) = 0.102335846657\n",
      "  lambda(x_1, x_2) = 0.267918673562\n",
      "  lambda(x_2, x_3) = 0.492395033280\n",
      "  lambda(x_3, x_4) = 0.773797411768\n",
      "  lambda(x_star, x_0) = 0.102335846657\n",
      "  lambda(x_star, x_1) = 0.165582826905\n",
      "  lambda(x_star, x_2) = 0.224476359717\n",
      "  lambda(x_star, x_3) = 0.281402378489\n",
      "  lambda(x_star, x_4) = 0.226202588232\n"
     ]
    }
   ],
   "source": [
    "lambda_matrix = np.array(relaxed_data[\"lambda_matrix\"], dtype=float)\n",
    "lambda_row_names = relaxed_data[\"lambda_row_names\"]\n",
    "lambda_col_names = relaxed_data[\"lambda_col_names\"]\n",
    "\n",
    "print(\"Active relaxed lambda entries:\")\n",
    "for i, ri in enumerate(lambda_row_names):\n",
    "    for j, ci in enumerate(lambda_col_names):\n",
    "        value = lambda_matrix[i, j]\n",
    "        if abs(value) > 1e-7:\n",
    "            print(f\"  lambda({ri}, {ci}) = {value:.12f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9a8800cd",
   "metadata": {},
   "source": [
    "## Closed-Form Lambda\n",
    "\n",
    "For `0 <= i < N`, the active chain multipliers are\n",
    "\n",
    "$$\\lambda_{x_i,x_{i+1}} = \\frac{2\\theta_i^2}{\\theta_N^2}.$$\n",
    "\n",
    "The star-row multipliers are\n",
    "\n",
    "$$\\lambda_{x_*,x_j} = \\frac{2\\theta_j}{\\theta_N^2}, \\quad 0 \\le j < N,$$\n",
    "\n",
    "with the terminal value\n",
    "\n",
    "$$\\lambda_{x_*,x_N} = \\frac{1}{\\theta_N}.$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "885f676e",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.289284Z",
     "iopub.status.busy": "2026-05-13T17:17:47.288945Z",
     "iopub.status.idle": "2026-05-13T17:17:47.310715Z",
     "shell.execute_reply": "2026-05-13T17:17:47.309493Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Lambda max residual: 6.038320075063552e-07\n",
      "Lambda matches: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & x_0 & x_1 & x_2 & x_3 & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.102 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.268 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.492 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.774 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.102 & 0.166 & 0.224 & 0.281 & 0.226 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "setup_path = ROOT / \"examples\" / \"ogm\" / \"ogm_setup.py\"\n",
    "spec = importlib.util.spec_from_file_location(\"ogm_setup\", setup_path)\n",
    "if spec is None or spec.loader is None:\n",
    "    raise ImportError(f\"Cannot load setup module from {setup_path}\")\n",
    "setup = importlib.util.module_from_spec(spec)\n",
    "spec.loader.exec_module(setup)\n",
    "\n",
    "N_int = relaxed_data[\"N\"]\n",
    "L = pf.Parameter(\"L\")\n",
    "params_sp = {\"L\": sp.S(1), \"R\": sp.S(1)}\n",
    "ctx, pb, obj = setup.get_pep_setup(sp.S(N_int), params_sp)\n",
    "pm = pf.ExpressionManager(ctx, resolve_parameters=params_sp)\n",
    "\n",
    "\n",
    "def idx(tag, N=N_int):\n",
    "    return N + 1 if tag == \"x_star\" else int(tag.split(\"_\")[1])\n",
    "\n",
    "\n",
    "def theta_ogm_value(i, N=N_int):\n",
    "    return sp.N(setup.theta_ogm(i, N), 17)\n",
    "\n",
    "\n",
    "def lamb(ri, ci, N=N_int):\n",
    "    i, j = idx(ri, N), idx(ci, N)\n",
    "    theta_N = theta_ogm_value(N, N)\n",
    "    if 0 <= i < N and j == i + 1:\n",
    "        return 2 * theta_ogm_value(i, N) ** 2 / theta_N**2\n",
    "    if ri == \"x_star\" and 0 <= j < N:\n",
    "        return 2 * theta_ogm_value(j, N) / theta_N**2\n",
    "    if ri == \"x_star\" and j == N:\n",
    "        return 1 / theta_N\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "lambda_candidate = np.array(\n",
    "    [[float(lamb(ri, ci)) for ci in lambda_col_names] for ri in lambda_row_names]\n",
    ")\n",
    "print(\"Lambda max residual:\", np.max(np.abs(lambda_candidate - lambda_matrix)))\n",
    "print(\"Lambda matches:\", np.allclose(lambda_candidate, lambda_matrix, atol=1e-4))\n",
    "pf.pprint_labeled_matrix(lambda_candidate, lambda_row_names, lambda_col_names)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "387880e2",
   "metadata": {},
   "source": [
    "## S Decomposition\n",
    "\n",
    "The Gram dual matrix is rank one up to solver tolerance. A direct positive semidefinite certificate is\n",
    "\n",
    "$$S = \\frac{1}{2L}\\left(\\nabla f(x_N) + \\sum_{i=0}^{N-1}\\frac{2\\theta_i}{\\theta_N}\\nabla f(x_i) - \\frac{L}{\\theta_N}(x_0-x_*)\\right)^2.$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "8f679d9c",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.314111Z",
     "iopub.status.busy": "2026-05-13T17:17:47.313579Z",
     "iopub.status.idle": "2026-05-13T17:17:47.379717Z",
     "shell.execute_reply": "2026-05-13T17:17:47.378671Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "S max residual: 9.525434978008285e-07\n",
      "S matches: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) & \\nabla f(x_3) & \\nabla f(x_4) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.026 & -0.026 & -0.051 & -0.083 & -0.112 & -0.141 & -0.113 \\\\x_\\star & -0.026 & 0.026 & 0.051 & 0.083 & 0.112 & 0.141 & 0.113 \\\\\\nabla f(x_0) & -0.051 & 0.051 & 0.102 & 0.166 & 0.224 & 0.281 & 0.226 \\\\\\nabla f(x_1) & -0.083 & 0.083 & 0.166 & 0.268 & 0.363 & 0.455 & 0.366 \\\\\\nabla f(x_2) & -0.112 & 0.112 & 0.224 & 0.363 & 0.492 & 0.617 & 0.496 \\\\\\nabla f(x_3) & -0.141 & 0.141 & 0.281 & 0.455 & 0.617 & 0.774 & 0.622 \\\\\\nabla f(x_4) & -0.113 & 0.113 & 0.226 & 0.366 & 0.496 & 0.622 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "theta_N = theta_ogm_value(N_int, N_int)\n",
    "tau = 1 / (2 * theta_N**2)\n",
    "\n",
    "ell = obj.grad(ctx[f\"x_{N_int}\"])\n",
    "for i in range(N_int):\n",
    "    ell += (2 * theta_ogm_value(i, N_int) / theta_N) * obj.grad(ctx[f\"x_{i}\"])\n",
    "ell += -(L / theta_N) * (ctx[\"x_0\"] - ctx[\"x_star\"])\n",
    "S_guess = (1 / (2 * L)) * ell**2\n",
    "\n",
    "S_candidate = np.array(pm.eval_scalar(S_guess).inner_prod_coords, dtype=float)\n",
    "S_matrix = np.array(relaxed_data[\"S_matrix\"], dtype=float)\n",
    "print(\"S max residual:\", np.max(np.abs(S_candidate - S_matrix)))\n",
    "print(\"S matches:\", np.allclose(S_candidate, S_matrix, atol=1e-4))\n",
    "pf.pprint_labeled_matrix(\n",
    "    S_candidate, relaxed_data[\"S_row_names\"], relaxed_data[\"S_col_names\"]\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "16ca56d8",
   "metadata": {},
   "source": [
    "## Full Proof Identity\n",
    "\n",
    "The fixed-horizon identity checked here is\n",
    "\n",
    "$$f(x_N)-f(x_*) - \tau\\|x_0-x_*\\|^2 - \\sum_{i,j}\\lambda_{ij} I_{ij} + S = 0,$$\n",
    "\n",
    "where $\tau = 1/(2\theta_N^2)$ for `L=R=1` in this numerical certificate."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "4bba41e2",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.382361Z",
     "iopub.status.busy": "2026-05-13T17:17:47.381999Z",
     "iopub.status.idle": "2026-05-13T17:17:47.455880Z",
     "shell.execute_reply": "2026-05-13T17:17:47.454125Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Proof residual max abs: 1.0928757898653885e-16\n",
      "Proof valid: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) & \\nabla f(x_3) & \\nabla f(x_4) \\\\\n",
       "        \\hline\n",
       "        x_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 \\\\\\nabla f(x_0) & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & -0.0 & -0.0 \\\\\\nabla f(x_1) & 0.0 & 0.0 & -0.0 & 0.0 & -0.0 & -0.0 & -0.0 \\\\\\nabla f(x_2) & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 \\\\\\nabla f(x_3) & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & -0.0 \\\\\\nabla f(x_4) & 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": [
    "interp_sum = pf.Scalar.zero()\n",
    "for ri, ci in itertools.product(lambda_row_names, lambda_col_names):\n",
    "    coeff = lamb(ri, ci, N_int)\n",
    "    if coeff != 0:\n",
    "        interp_sum += coeff * obj.interp_ineq(ri, ci, pep_context=ctx)\n",
    "\n",
    "x_N = ctx[f\"x_{N_int}\"]\n",
    "x_0 = ctx[\"x_0\"]\n",
    "x_star = ctx[\"x_star\"]\n",
    "LHS = obj(x_N) - obj(x_star) - tau * (x_0 - x_star) ** 2\n",
    "proof_residual = np.array(\n",
    "    pm.eval_scalar(LHS - interp_sum + S_guess).inner_prod_coords, dtype=float\n",
    ")\n",
    "print(\"Proof residual max abs:\", np.max(np.abs(proof_residual)))\n",
    "print(\"Proof valid:\", np.allclose(proof_residual, 0, atol=1e-8))\n",
    "pf.pprint_labeled_matrix(\n",
    "    proof_residual, relaxed_data[\"S_row_names\"], relaxed_data[\"S_col_names\"]\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "650fddb9",
   "metadata": {},
   "source": [
    "## Partial-Sum Lyapunov Construction\n",
    "\n",
    "The Block 2 certificate is most naturally grouped as a reverse-tail sequence. For `k = 0, ..., N`, define\n",
    "\n",
    "$$V_k = -S + \\sum_{t=k}^{N-1}\\left(\\lambda_{x_t,x_{t+1}} I(x_t,x_{t+1}) + \\lambda_{x_*,x_t} I(x_*,x_t)\\right) + \\mathbf{1}_{k<N}\\lambda_{x_*,x_N}I(x_*,x_N),$$\n",
    "\n",
    "where the terminal star term is included only in the final step block `t=N-1`. This makes `V_0` equal to the full right-hand certificate and `V_N=-S`. The interior terms have constant rank, which is the signal used by the next block to identify a fixed spanning vector family."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "7ec3e735",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.459569Z",
     "iopub.status.busy": "2026-05-13T17:17:47.459147Z",
     "iopub.status.idle": "2026-05-13T17:17:47.611609Z",
     "shell.execute_reply": "2026-05-13T17:17:47.610242Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rank V_0: 1\n",
      "\n",
      "rank V_1: 3\n",
      "rank V_2: 3\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rank V_3: 3\n",
      "rank V_4: 1\n",
      "Interior rank is constant: True\n"
     ]
    }
   ],
   "source": [
    "b3_state = json.loads((state_dir / \"ogm_b3.json\").read_text())\n",
    "rank_tolerance = b3_state[\"rank_tolerance\"]\n",
    "\n",
    "\n",
    "def step_terms(step):\n",
    "    terms = [(f\"x_{step}\", f\"x_{step + 1}\"), (\"x_star\", f\"x_{step}\")]\n",
    "    if step == N_int - 1:\n",
    "        terms.append((\"x_star\", f\"x_{N_int}\"))\n",
    "    return terms\n",
    "\n",
    "\n",
    "lyap = []\n",
    "for k in range(N_int + 1):\n",
    "    tail = -S_guess\n",
    "    for step in range(k, N_int):\n",
    "        for ri, ci in step_terms(step):\n",
    "            coeff = lamb(ri, ci, N_int)\n",
    "            if coeff != 0:\n",
    "                tail += coeff * obj.interp_ineq(ri, ci, pep_context=ctx)\n",
    "    lyap.append(tail)\n",
    "\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": 10,
   "id": "19d3c79a",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.613856Z",
     "iopub.status.busy": "2026-05-13T17:17:47.613554Z",
     "iopub.status.idle": "2026-05-13T17:17:47.619540Z",
     "shell.execute_reply": "2026-05-13T17:17:47.618371Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lyap[4] rank: 1\n",
      "Coverage check: lyap[N] rank should be 0 or 1 (boundary identity term)\n",
      "Stored rank profile: [1, 3, 3, 3, 1]\n",
      "Computed rank profile matches stored: True\n"
     ]
    }
   ],
   "source": [
    "M_final = pm.eval_scalar(lyap[N_int]).inner_prod_coords.astype(float)\n",
    "rank_final = int(np.linalg.matrix_rank(M_final, tol=rank_tolerance))\n",
    "print(f\"lyap[{N_int}] rank:\", rank_final)\n",
    "print(\"Coverage check: lyap[N] rank should be 0 or 1 (boundary identity term)\")\n",
    "print(\"Stored rank profile:\", b3_state[\"rank_profile\"])\n",
    "print(\"Computed rank profile matches stored:\", ranks == b3_state[\"rank_profile\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3c809e5c",
   "metadata": {},
   "source": [
    "## Identify the vectors composing the Lyapunov function\n",
    "\n",
    "Block 4 starts from the reverse-tail partial sums and searches for interpretable vectors spanning their rank-three interior quadratic parts. The final boundary term is rank one and is handled separately."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "99dc50cc",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.622370Z",
     "iopub.status.busy": "2026-05-13T17:17:47.622034Z",
     "iopub.status.idle": "2026-05-13T17:17:47.628482Z",
     "shell.execute_reply": "2026-05-13T17:17:47.627262Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Stored rank profile: [1, 3, 3, 3, 1]\n",
      "Current rank profile: [1, 3, 3, 3, 1]\n",
      "Rank profile matches: True\n"
     ]
    }
   ],
   "source": [
    "b4_state = json.loads((state_dir / \"ogm_b4.json\").read_text())\n",
    "print(\"Stored rank profile:\", b4_state[\"rank_profile\"])\n",
    "print(\"Current rank profile:\", ranks)\n",
    "print(\"Rank profile matches:\", ranks == b4_state[\"rank_profile\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "74dea514",
   "metadata": {},
   "source": [
    "### Candidate-vector scan\n",
    "\n",
    "The candidate pool includes tagged iterates, gradients, OGM auxiliary iterates `z_k`, point-to-solution gaps, step differences, and the terminal square vector suggested by the Block 2 certificate."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "3f52e14f",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.630794Z",
     "iopub.status.busy": "2026-05-13T17:17:47.630485Z",
     "iopub.status.idle": "2026-05-13T17:17:47.647783Z",
     "shell.execute_reply": "2026-05-13T17:17:47.646792Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Candidate count: 27\n"
     ]
    }
   ],
   "source": [
    "from pepflow.lyapunov_utils import (\n",
    "    vectors_in_column_space,\n",
    "    decompose_rankr_symmetric,\n",
    "    independent_subset,\n",
    ")\n",
    "\n",
    "candidate_labels = []\n",
    "candidate_vectors = []\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_vectors:\n",
    "        existing_coords = np.asarray(pm.eval_vector(existing).coords, dtype=float)\n",
    "        if np.linalg.norm(coords - existing_coords) < 1e-9:\n",
    "            return\n",
    "    candidate_labels.append(label)\n",
    "    candidate_vectors.append(vector)\n",
    "\n",
    "\n",
    "for i in range(N_int + 1):\n",
    "    xi = ctx[f\"x_{i}\"]\n",
    "    add_candidate(f\"x_{i}-x_star\", xi - ctx[\"x_star\"])\n",
    "    add_candidate(f\"grad_f(x_{i})\", obj.grad(xi))\n",
    "    if i > 0:\n",
    "        add_candidate(f\"z_{i}-x_star\", ctx[f\"z_{i}\"] - ctx[\"x_star\"])\n",
    "        add_candidate(f\"z_{i}-x_{i}\", ctx[f\"z_{i}\"] - xi)\n",
    "\n",
    "for i in range(N_int):\n",
    "    add_candidate(f\"x_{i + 1}-x_{i}\", ctx[f\"x_{i + 1}\"] - ctx[f\"x_{i}\"])\n",
    "    add_candidate(\n",
    "        f\"grad_f(x_{i + 1})-grad_f(x_{i})\",\n",
    "        obj.grad(ctx[f\"x_{i + 1}\"]) - obj.grad(ctx[f\"x_{i}\"]),\n",
    "    )\n",
    "\n",
    "terminal_vector = (\n",
    "    ctx[f\"z_{N_int}\"] - theta_N / L * obj.grad(ctx[f\"x_{N_int}\"]) - ctx[\"x_star\"]\n",
    ")\n",
    "add_candidate(f\"z_{N_int}-theta_N/L*grad_f(x_{N_int})-x_star\", terminal_vector)\n",
    "\n",
    "print(\"Candidate count:\", len(candidate_vectors))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "6145eafd",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.650014Z",
     "iopub.status.busy": "2026-05-13T17:17:47.649781Z",
     "iopub.status.idle": "2026-05-13T17:17:47.736438Z",
     "shell.execute_reply": "2026-05-13T17:17:47.734578Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_1 column-space candidates:\n",
      "   x_0-x_star\n",
      "   grad_f(x_0)\n",
      "   x_1-x_star\n",
      "   grad_f(x_1)\n",
      "   z_1-x_star\n",
      "   z_1-x_1\n",
      "   x_2-x_star\n",
      "   z_2-x_star\n",
      "   z_2-x_2\n",
      "   x_1-x_0\n",
      "   grad_f(x_1)-grad_f(x_0)\n",
      "   x_2-x_1\n",
      "V_2 column-space candidates:\n",
      "   x_2-x_star\n",
      "   grad_f(x_2)\n",
      "   z_2-x_star\n",
      "   z_2-x_2\n",
      "   x_3-x_star\n",
      "   z_3-x_star\n",
      "   z_3-x_3\n",
      "   x_3-x_2\n",
      "V_3 column-space candidates:\n",
      "   x_3-x_star\n",
      "   grad_f(x_3)\n",
      "   z_3-x_star\n",
      "   z_3-x_3\n",
      "   x_4-x_star\n",
      "   z_4-x_star\n",
      "   z_4-x_4\n",
      "   x_4-x_3\n"
     ]
    }
   ],
   "source": [
    "for k in range(1, N_int):\n",
    "    in_col = vectors_in_column_space(\n",
    "        lyap[k],\n",
    "        candidate_vectors,\n",
    "        pep_context=ctx,\n",
    "        resolve_parameters=params_sp,\n",
    "        rtol=1e-4,\n",
    "        atol=1e-4,\n",
    "    )\n",
    "    print(f\"V_{k} column-space candidates:\")\n",
    "    for vector in in_col:\n",
    "        print(\"  \", candidate_labels[candidate_vectors.index(vector)])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "14a2d674",
   "metadata": {},
   "source": [
    "### Selected basis pattern\n",
    "\n",
    "For interior indices `1 <= k < N`, use the basis\n",
    "\n",
    "$$[x_k-x_*,\\ \\nabla f(x_k),\\ z_{k+1}-x_*].$$\n",
    "\n",
    "For the terminal boundary term `k=N`, use\n",
    "\n",
    "$$[z_N - \\frac{\\theta_N}{L}\\nabla f(x_N) - x_*].$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "e0f8dc52",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.739709Z",
     "iopub.status.busy": "2026-05-13T17:17:47.739291Z",
     "iopub.status.idle": "2026-05-13T17:17:47.758808Z",
     "shell.execute_reply": "2026-05-13T17:17:47.757777Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: rank 3 basis ['x_1-x_star', 'grad_f(x_1)', 'z_2-x_star']\n",
      "k=2: rank 3 basis ['x_2-x_star', 'grad_f(x_2)', 'z_3-x_star']\n",
      "k=3: rank 3 basis ['x_3-x_star', 'grad_f(x_3)', 'z_4-x_star']\n",
      "k=4: rank 1 basis ['z_4-theta_N/L*grad_f(x_4)-x_star']\n"
     ]
    }
   ],
   "source": [
    "def V_k_basis(k):\n",
    "    if 1 <= k < N_int:\n",
    "        return [\n",
    "            ctx[f\"x_{k}\"] - ctx[\"x_star\"],\n",
    "            obj.grad(ctx[f\"x_{k}\"]),\n",
    "            ctx[f\"z_{k + 1}\"] - ctx[\"x_star\"],\n",
    "        ]\n",
    "    if k == N_int:\n",
    "        return [\n",
    "            ctx[f\"z_{N_int}\"]\n",
    "            - theta_N / L * obj.grad(ctx[f\"x_{N_int}\"])\n",
    "            - ctx[\"x_star\"]\n",
    "        ]\n",
    "    return []\n",
    "\n",
    "\n",
    "def V_k_basis_labels(k):\n",
    "    if 1 <= k < N_int:\n",
    "        return [f\"x_{k}-x_star\", f\"grad_f(x_{k})\", f\"z_{k + 1}-x_star\"]\n",
    "    if k == N_int:\n",
    "        return [f\"z_{N_int}-theta_N/L*grad_f(x_{N_int})-x_star\"]\n",
    "    return []\n",
    "\n",
    "\n",
    "for k in range(1, N_int + 1):\n",
    "    basis = V_k_basis(k)\n",
    "    independent, _ = independent_subset(\n",
    "        basis, pep_context=ctx, resolve_parameters=params_sp\n",
    "    )\n",
    "    print(f\"k={k}: rank {len(independent)} basis {V_k_basis_labels(k)}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bf0ff846",
   "metadata": {},
   "source": [
    "### Coefficient matrices\n",
    "\n",
    "For `1 <= k < N`, in the basis order `[x_k-x_*, grad_f(x_k), z_{k+1}-x_*]`, the nonzero coefficient pattern is\n",
    "\n",
    "$$C_{01}=C_{10}=-\\frac{\\theta_k}{\\theta_N^2},\\quad C_{11}=\\frac{\\theta_k(\\theta_k+1)}{\\theta_N^2},\\quad C_{22}=-\\frac{1}{2\\theta_N^2}.$$\n",
    "\n",
    "For `k=N`, the terminal coefficient is `-1/(2 theta_N^2)` in the one-vector boundary basis."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "73e8f7ac",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.761450Z",
     "iopub.status.busy": "2026-05-13T17:17:47.761179Z",
     "iopub.status.idle": "2026-05-13T17:17:47.872287Z",
     "shell.execute_reply": "2026-05-13T17:17:47.871268Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: formula residual 1.055e-15\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccc}\n",
       "         & x_1-x_\\star & \\nabla f(x_1) & z_2-x_\\star \\\\\n",
       "        \\hline\n",
       "        x_1-x_\\star & 0.0 & -0.083 & 0.0 \\\\\\nabla f(x_1) & -0.083 & 0.217 & 0.0 \\\\z_2-x_\\star & 0.0 & 0.0 & -0.026 \\\\\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 4.025e-16\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccc}\n",
       "         & x_2-x_\\star & \\nabla f(x_2) & z_3-x_\\star \\\\\n",
       "        \\hline\n",
       "        x_2-x_\\star & 0.0 & -0.112 & 0.0 \\\\\\nabla f(x_2) & -0.112 & 0.358 & 0.0 \\\\z_3-x_\\star & 0.0 & 0.0 & -0.026 \\\\\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 1.610e-15\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccc}\n",
       "         & x_3-x_\\star & \\nabla f(x_3) & z_4-x_\\star \\\\\n",
       "        \\hline\n",
       "        x_3-x_\\star & 0.0 & -0.141 & 0.0 \\\\\\nabla f(x_3) & -0.141 & 0.528 & 0.0 \\\\z_4-x_\\star & 0.0 & 0.0 & -0.026 \\\\\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 3.469e-18\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|c}\n",
       "         & z_4-theta_N/L*\\nabla f(x_4)-x_\\star \\\\\n",
       "        \\hline\n",
       "        z_4-theta_N/L*\\nabla f(x_4)-x_\\star & -0.026 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def coeff_pattern(k, N=N_int):\n",
    "    theta_N_local = theta_ogm_value(N, N)\n",
    "    if 1 <= k < N:\n",
    "        theta_k = theta_ogm_value(k, N)\n",
    "        C = np.zeros((3, 3), dtype=float)\n",
    "        C[0, 1] = C[1, 0] = float(-theta_k / theta_N_local**2)\n",
    "        C[1, 1] = float(theta_k * (theta_k + 1) / theta_N_local**2)\n",
    "        C[2, 2] = float(-1 / (2 * theta_N_local**2))\n",
    "        return C\n",
    "    if k == N:\n",
    "        return np.array([[float(-1 / (2 * theta_N_local**2))]])\n",
    "    return np.zeros((0, 0))\n",
    "\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 = coeff_pattern(k, N_int)\n",
    "    print(f\"k={k}: formula residual {np.max(np.abs(C - C_formula)):.3e}\")\n",
    "    pf.pprint_labeled_matrix(C_formula, labels, labels)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "73de8a3a",
   "metadata": {},
   "source": [
    "### Block 4 conclusion\n",
    "\n",
    "The interior Lyapunov terms are represented by a rank-three basis tied directly to the current OGM iterate, current gradient, and next auxiliary `z` iterate. Block 5 will symbolically verify the step, base, and boundary identities for this closed-form candidate."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4e59f303",
   "metadata": {},
   "source": [
    "## Symbolic Step Recursion Verification\n",
    "\n",
    "For $1\\le k<N-1$, this cell verifies\n",
    "\n",
    "$$V_{k+1}-V_k+\\frac{2\\theta_k^2}{\\theta_N^2}I_f(x_k,x_{k+1})+\\frac{2\\theta_k}{\\theta_N^2}I_f(x_*,x_k)=0.$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero using $\\theta_{k+1}^2-\\theta_{k+1}=\\theta_k^2$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "3c7c2449",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.875338Z",
     "iopub.status.busy": "2026-05-13T17:17:47.874746Z",
     "iopub.status.idle": "2026-05-13T17:17:47.970501Z",
     "shell.execute_reply": "2026-05-13T17:17:47.969468Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Step func residual zero: True\n",
      "Step matrix residual zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_k & x_{k+1} & x_\\star & \\nabla f_{step}(x_k) & \\nabla f_{step}(x_{k+1}) \\\\\n",
       "        \\hline\n",
       "        x_k & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{k+1} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f_{step}(x_k) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f_{step}(x_{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": [
    "def reduce_theta_step(expr, theta_k_symbol, theta_k1_symbol):\n",
    "    numerator = sp.factor(sp.together(expr).as_numer_denom()[0])\n",
    "    relation = theta_k1_symbol**2 - theta_k1_symbol - theta_k_symbol**2\n",
    "    return sp.factor(sp.rem(numerator, relation, theta_k1_symbol))\n",
    "\n",
    "\n",
    "ctx_step = pf.PEPContext(\"ogm_symbolic_step\").set_as_current()\n",
    "L_step = pf.Parameter(\"L_step\")\n",
    "theta_k_param = pf.Parameter(\"theta_k\")\n",
    "theta_k1_param = pf.Parameter(\"theta_{k+1}\")\n",
    "theta_N_param = pf.Parameter(\"theta_N\")\n",
    "f_step = pf.SmoothConvexFunction(is_basis=True, tags=[\"f_{step}\"], L=L_step)\n",
    "x_k_step = pf.Vector(is_basis=True, tags=[\"x_k\"])\n",
    "x_k1_step = pf.Vector(is_basis=True, tags=[\"x_{k+1}\"])\n",
    "x_star_step = f_step.set_stationary_point(\"x_star\")\n",
    "g_k_step = f_step.grad(x_k_step)\n",
    "g_k1_step = f_step.grad(x_k1_step)\n",
    "\n",
    "z_k1_step = theta_k1_param * x_k1_step - (theta_k1_param - 1) * (\n",
    "    x_k_step - g_k_step / L_step\n",
    ")\n",
    "z_k2_step = z_k1_step - (2 * theta_k1_param / L_step) * g_k1_step\n",
    "alpha_k_step = 2 * (theta_k_param**2 - theta_k_param) / theta_N_param**2\n",
    "alpha_k1_step = 2 * theta_k_param**2 / theta_N_param**2\n",
    "\n",
    "V_k_step = (\n",
    "    -alpha_k_step * f_step(x_k_step)\n",
    "    - (1 - alpha_k_step) * f_step(x_star_step)\n",
    "    - (2 * theta_k_param / theta_N_param**2) * (x_k_step - x_star_step) * g_k_step\n",
    "    + (theta_k_param * (theta_k_param + 1) / (L_step * theta_N_param**2)) * g_k_step**2\n",
    "    - (L_step / (2 * theta_N_param**2)) * (z_k1_step - x_star_step) ** 2\n",
    ")\n",
    "V_k1_step = (\n",
    "    -alpha_k1_step * f_step(x_k1_step)\n",
    "    - (1 - alpha_k1_step) * f_step(x_star_step)\n",
    "    - (2 * theta_k1_param / theta_N_param**2) * (x_k1_step - x_star_step) * g_k1_step\n",
    "    + (theta_k1_param * (theta_k1_param + 1) / (L_step * theta_N_param**2))\n",
    "    * g_k1_step**2\n",
    "    - (L_step / (2 * theta_N_param**2)) * (z_k2_step - x_star_step) ** 2\n",
    ")\n",
    "step_residual = (2 * theta_k_param**2 / theta_N_param**2) * f_step.interp_ineq(\n",
    "    \"x_k\", \"x_{k+1}\", pep_context=ctx_step\n",
    ") + (2 * theta_k_param / theta_N_param**2) * f_step.interp_ineq(\n",
    "    \"x_star\", \"x_k\", pep_context=ctx_step\n",
    ")\n",
    "step_diff = V_k1_step - V_k_step + step_residual\n",
    "\n",
    "L_sym, theta_k_sym, theta_k1_sym, theta_N_sym = sp.symbols(\n",
    "    \"L theta_k theta_k1 theta_N\", positive=True\n",
    ")\n",
    "pm_step = pf.ExpressionManager(\n",
    "    ctx_step,\n",
    "    resolve_parameters={\n",
    "        \"L_step\": L_sym,\n",
    "        \"theta_k\": theta_k_sym,\n",
    "        \"theta_{k+1}\": theta_k1_sym,\n",
    "        \"theta_N\": theta_N_sym,\n",
    "    },\n",
    ")\n",
    "step_eval = pm_step.eval_scalar(step_diff, sympy_mode=True)\n",
    "step_func_residual = [\n",
    "    reduce_theta_step(e, theta_k_sym, theta_k1_sym) for e in step_eval.func_coords\n",
    "]\n",
    "step_matrix_residual = sp.Matrix(step_eval.inner_prod_coords).applyfunc(\n",
    "    lambda e: reduce_theta_step(e, theta_k_sym, theta_k1_sym)\n",
    ")\n",
    "print(\"Step func residual zero:\", all(e == 0 for e in step_func_residual))\n",
    "print(\n",
    "    \"Step matrix residual zero:\",\n",
    "    step_matrix_residual == sp.zeros(*step_matrix_residual.shape),\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(step_matrix_residual.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_step.basis_vectors()],\n",
    "    [str(v) for v in ctx_step.basis_vectors()],\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4922685e",
   "metadata": {},
   "source": [
    "## Base Case Symbolic Verification\n",
    "\n",
    "This cell verifies\n",
    "\n",
    "$$V_1-V_0+\\frac{2}{\\theta_N^2}I_f(x_0,x_1)+\\frac{2}{\\theta_N^2}I_f(x_*,x_0)=0.$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero using $\\theta_1^2-\\theta_1=1$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "08de7d94",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:47.972762Z",
     "iopub.status.busy": "2026-05-13T17:17:47.972503Z",
     "iopub.status.idle": "2026-05-13T17:17:48.011572Z",
     "shell.execute_reply": "2026-05-13T17:17:48.010418Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Base func residual zero: True\n",
      "Base matrix residual zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_0 & x_\\star & \\nabla f_{base}(x_0) & \\nabla f_{base}(x_1) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f_{base}(x_0) & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f_{base}(x_1) & 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": [
    "ctx_base = pf.PEPContext(\"ogm_symbolic_base\").set_as_current()\n",
    "L_base = pf.Parameter(\"L_base\")\n",
    "theta_1_param = pf.Parameter(\"theta_1\")\n",
    "theta_N_base_param = pf.Parameter(\"theta_N_base\")\n",
    "f_base = pf.SmoothConvexFunction(is_basis=True, tags=[\"f_{base}\"], L=L_base)\n",
    "x_0_base = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "x_star_base = f_base.set_stationary_point(\"x_star\")\n",
    "g_0_base = f_base.grad(x_0_base)\n",
    "x_1_base = x_0_base - (theta_1_param / L_base) * g_0_base\n",
    "x_1_base.add_tag(\"x_1\")\n",
    "g_1_base = f_base.grad(x_1_base)\n",
    "z_1_base = x_0_base - (2 / L_base) * g_0_base\n",
    "z_2_base = z_1_base - (2 * theta_1_param / L_base) * g_1_base\n",
    "\n",
    "V_0_base = (\n",
    "    -f_base(x_star_base)\n",
    "    - (L_base / (2 * theta_N_base_param**2)) * (x_0_base - x_star_base) ** 2\n",
    ")\n",
    "V_1_base = (\n",
    "    -(2 / theta_N_base_param**2) * f_base(x_1_base)\n",
    "    - (1 - 2 / theta_N_base_param**2) * f_base(x_star_base)\n",
    "    - (2 * theta_1_param / theta_N_base_param**2) * (x_1_base - x_star_base) * g_1_base\n",
    "    + (theta_1_param * (theta_1_param + 1) / (L_base * theta_N_base_param**2))\n",
    "    * g_1_base**2\n",
    "    - (L_base / (2 * theta_N_base_param**2)) * (z_2_base - x_star_base) ** 2\n",
    ")\n",
    "base_residual = (2 / theta_N_base_param**2) * f_base.interp_ineq(\n",
    "    \"x_0\", \"x_1\", pep_context=ctx_base\n",
    ") + (2 / theta_N_base_param**2) * f_base.interp_ineq(\n",
    "    \"x_star\", \"x_0\", pep_context=ctx_base\n",
    ")\n",
    "base_diff = V_1_base - V_0_base + base_residual\n",
    "L_base_sym, theta_1_sym, theta_N_base_sym = sp.symbols(\n",
    "    \"L theta_1 theta_N\", positive=True\n",
    ")\n",
    "pm_base = pf.ExpressionManager(\n",
    "    ctx_base,\n",
    "    resolve_parameters={\n",
    "        \"L_base\": L_base_sym,\n",
    "        \"theta_1\": theta_1_sym,\n",
    "        \"theta_N_base\": theta_N_base_sym,\n",
    "    },\n",
    ")\n",
    "base_eval = pm_base.eval_scalar(base_diff, sympy_mode=True)\n",
    "base_relation = theta_1_sym**2 - theta_1_sym - 1\n",
    "\n",
    "\n",
    "def reduce_theta_base(expr):\n",
    "    numerator = sp.factor(sp.together(expr).as_numer_denom()[0])\n",
    "    return sp.factor(sp.rem(numerator, base_relation, theta_1_sym))\n",
    "\n",
    "\n",
    "base_func_residual = [reduce_theta_base(e) for e in base_eval.func_coords]\n",
    "base_matrix_residual = sp.Matrix(base_eval.inner_prod_coords).applyfunc(\n",
    "    reduce_theta_base\n",
    ")\n",
    "print(\"Base func residual zero:\", all(e == 0 for e in base_func_residual))\n",
    "print(\n",
    "    \"Base matrix residual zero:\",\n",
    "    base_matrix_residual == sp.zeros(*base_matrix_residual.shape),\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(base_matrix_residual.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_base.basis_vectors()],\n",
    "    [str(v) for v in ctx_base.basis_vectors()],\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "920a072a",
   "metadata": {},
   "source": [
    "### Boundary Identity Symbolic Verification\n",
    "\n",
    "This cell verifies\n",
    "\n",
    "$$V_N+\\frac{L}{2\\theta_N^2}\\left\\|z_N-\\frac{\\theta_N}{L}\\nabla f(x_N)-x_*\\right\\|^2=0.$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "6a628c09",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T17:17:48.014285Z",
     "iopub.status.busy": "2026-05-13T17:17:48.013885Z",
     "iopub.status.idle": "2026-05-13T17:17:48.028514Z",
     "shell.execute_reply": "2026-05-13T17:17:48.027111Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Boundary func residual zero: True\n",
      "Boundary matrix residual zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & z_N & x_N & x_\\star & \\nabla f_{boundary}(x_N) \\\\\n",
       "        \\hline\n",
       "        z_N & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_N & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f_{boundary}(x_N) & 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": [
    "ctx_boundary = pf.PEPContext(\"ogm_symbolic_boundary\").set_as_current()\n",
    "L_boundary = pf.Parameter(\"L_boundary\")\n",
    "theta_N_boundary = pf.Parameter(\"theta_N_boundary\")\n",
    "f_boundary = pf.SmoothConvexFunction(is_basis=True, tags=[\"f_{boundary}\"], L=L_boundary)\n",
    "z_N_boundary = pf.Vector(is_basis=True, tags=[\"z_N\"])\n",
    "x_N_boundary = pf.Vector(is_basis=True, tags=[\"x_N\"])\n",
    "x_star_boundary = f_boundary.set_stationary_point(\"x_star\")\n",
    "g_N_boundary = f_boundary.grad(x_N_boundary)\n",
    "q_boundary = (\n",
    "    z_N_boundary - theta_N_boundary / L_boundary * g_N_boundary - x_star_boundary\n",
    ")\n",
    "V_N_boundary = -(L_boundary / (2 * theta_N_boundary**2)) * q_boundary**2\n",
    "S_boundary = (L_boundary / (2 * theta_N_boundary**2)) * q_boundary**2\n",
    "boundary_diff = V_N_boundary + S_boundary\n",
    "L_boundary_sym, theta_N_boundary_sym = sp.symbols(\"L theta_N\", positive=True)\n",
    "pm_boundary = pf.ExpressionManager(\n",
    "    ctx_boundary,\n",
    "    resolve_parameters={\n",
    "        \"L_boundary\": L_boundary_sym,\n",
    "        \"theta_N_boundary\": theta_N_boundary_sym,\n",
    "    },\n",
    ")\n",
    "boundary_eval = pm_boundary.eval_scalar(boundary_diff, sympy_mode=True)\n",
    "boundary_func_residual = [sp.simplify(e) for e in boundary_eval.func_coords]\n",
    "boundary_matrix_residual = sp.Matrix(boundary_eval.inner_prod_coords).applyfunc(\n",
    "    sp.simplify\n",
    ")\n",
    "print(\"Boundary func residual zero:\", all(e == 0 for e in boundary_func_residual))\n",
    "print(\n",
    "    \"Boundary matrix residual zero:\",\n",
    "    boundary_matrix_residual == sp.zeros(*boundary_matrix_residual.shape),\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(boundary_matrix_residual.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_boundary.basis_vectors()],\n",
    "    [str(v) for v in ctx_boundary.basis_vectors()],\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
