{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "b389699b",
   "metadata": {},
   "source": [
    "# Proximal Gradient Descent\n",
    "\n",
    "We study composite convex minimization\n",
    "\n",
    "$$\\min_x h(x) := f(x)+g(x),$$\n",
    "\n",
    "where $f$ is convex and $L$-smooth, and $g$ is closed, proper, and convex. Let $x_\\star$ minimize $h$, and assume\n",
    "\n",
    "$$\\|x_0-x_\\star\\|^2 \\le R^2.$$\n",
    "\n",
    "With fixed step size $1/L$, proximal gradient descent is\n",
    "\n",
    "$$y_{k+1}=x_k-\\frac{1}{L}\\nabla f(x_k),\\qquad x_{k+1}=\\operatorname{prox}_{g/L}(y_{k+1}).$$\n",
    "\n",
    "The performance metric is $h(x_N)-h(x_\\star)$."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2d9bac25",
   "metadata": {},
   "source": [
    "## Proof Statement"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ea2b85e3",
   "metadata": {},
   "source": [
    "### Theorem\n",
    "\n",
    "Let $f$ be convex and $L$-smooth, let $g$ be closed, proper, and convex, and let $x_\\star\\in\\arg\\min (f+g)$. If\n",
    "$$\n",
    "x_{k+1}=\\operatorname{prox}_{g/L}\\left(x_k-\\frac{1}{L}\\nabla f(x_k)\\right),\n",
    "\\qquad \\|x_0-x_\\star\\|^2\\le R^2,\n",
    "$$\n",
    "then for every $N\\ge1$,\n",
    "$$\n",
    "(f+g)(x_N)-(f+g)(x_\\star)\\le \\frac{L R^2}{4N}.\n",
    "$$\n",
    "\n",
    "For $2\\le k<N$, the recovered partial-sum Lyapunov certificate is\n",
    "$$\n",
    "\\begin{aligned}\n",
    "V_k={}&\n",
    "\\frac{k}{2N-k+1}\\bigl(f(x_k)-f(x_0)\\bigr)\n",
    "+\\frac{k}{2N-k}\\bigl(g(x_k)-g(x_1)\\bigr) \\\\\n",
    "&-\\frac{L}{4N}\\|x_0-x_\\star\\|^2\n",
    "+L\\frac{N-k}{(2N-k)^2}\\|x_k-x_\\star\\|^2\\\\\n",
    "&+\\frac{k}{(2N-k)(2N-k+1)}\n",
    "\\langle x_k-x_\\star,\\nabla f(x_k)\\rangle\\\\\n",
    "&-\\frac{k}{2L(2N-k)(2N-k+1)}\n",
    "\\|\\nabla f(x_k)-\\nabla f(x_\\star)\\|^2.\n",
    "\\end{aligned}\n",
    "$$\n",
    "The first certificate uses the same quadratic part with $k=1$, and its function part is\n",
    "$$\n",
    "\\frac{1}{2N}\\bigl(f(x_1)-f(x_0)\\bigr)-\\frac{1}{2N-1}g(x_1).\n",
    "$$\n",
    "At the terminal point,\n",
    "$$\n",
    "V_N=(f+g)(x_N)-(f+g)(x_\\star)-\\frac{L}{4N}\\|x_0-x_\\star\\|^2.\n",
    "$$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a9509212",
   "metadata": {},
   "source": [
    "### Proof outline\n",
    "\n",
    "Let $I_f(u,v)$ denote the smooth-convex interpolation residual for $f$, and $I_g(u,v)$ the convex interpolation residual for $g$. These residuals are nonpositive under the PEPFlow convention. The Block 3 partial sums verify, for $k=0,\\ldots,N-1$,\n",
    "$$\n",
    "\\begin{aligned}\n",
    "V_{k+1}-V_k={}&\n",
    "\\lambda_{\\star,k} I_f(x_\\star,x_k)\n",
    "+\\lambda_{k,k+1}I_f(x_k,x_{k+1})\n",
    "+\\gamma_{k,k+1}I_g(x_k,x_{k+1})\\\\\n",
    "&+\\gamma_{\\star,k+1}I_g(x_\\star,x_{k+1})\n",
    "-S_{k+1},\n",
    "\\end{aligned}\n",
    "$$\n",
    "with the additional terminal residual $\\lambda_{\\star,N}I_f(x_\\star,x_N)$ included at the final step. Here the nonnegative square term is\n",
    "$$\n",
    "\\begin{aligned}\n",
    "S_{k+1}={}&\n",
    "\\frac{N}{(2N-k)(2N-k-1)L}\n",
    "\\left\\|\n",
    "\\frac{k+1}{2N}\\nabla f(x_{k+1})\n",
    "+\\frac{2N-k-1}{2N}\\nabla f(x_k)\n",
    "-\\nabla f(x_\\star)\n",
    "\\right\\|^2\\\\\n",
    "&+\\frac{1}{2L}\n",
    "\\left(\n",
    "\\frac{k}{2N-k-1}\\frac{2N+2}{2N+1}\n",
    "+\\frac{2N}{(2N+1)(2N-k-1)^2}\n",
    "\\right)\\\\\n",
    "&\\qquad\\cdot\n",
    "\\left\\|\n",
    "\\nabla f(x_k)+\\nabla g(x_{k+1})\n",
    "-\\frac{L}{2N-k}(x_k-x_\\star)\n",
    "\\right\\|^2\\\\\n",
    "&+\\frac{k+1}{2N-k}\\frac{2N}{2N+1}\\frac{1}{2L}\n",
    "\\left\\|\n",
    "\\frac{2N+1}{2N}\\bigl(\\nabla f(x_{k+1})+\\nabla g(x_{k+1})\\bigr)\n",
    "\\right.\\\\\n",
    "&\\qquad\\left.\n",
    "-\\frac{1}{2N}\\bigl(\\nabla f(x_k)+\\nabla g(x_{k+1})\\bigr)\n",
    "-\\frac{L}{2N-k-1}(x_{k+1}-x_\\star)\n",
    "\\right\\|^2.\n",
    "\\end{aligned}\n",
    "$$\n",
    "All $\\lambda,\\gamma$ coefficients are nonnegative and $S_{k+1}\\ge0$, hence $V_N\\le V_0=0$. The boundary identity is exactly\n",
    "$$\n",
    "V_N=(f+g)(x_N)-(f+g)(x_\\star)-\\frac{L}{4N}\\|x_0-x_\\star\\|^2.\n",
    "$$\n",
    "Together with $\\|x_0-x_\\star\\|^2\\le R^2$, this gives the claimed rate."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ef021bbe",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "fb4b3ef9",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:32.627772Z",
     "iopub.status.busy": "2026-05-14T05:10:32.627255Z",
     "iopub.status.idle": "2026-05-14T05:10:32.634801Z",
     "shell.execute_reply": "2026-05-14T05:10:32.633752Z"
    }
   },
   "outputs": [],
   "source": [
    "import sys\n",
    "from pathlib import Path\n",
    "\n",
    "repo_root = Path.cwd()\n",
    "if not (repo_root / \"pepflow\").exists():\n",
    "    repo_root = Path.cwd().parents[1]\n",
    "if str(repo_root) not in sys.path:\n",
    "    sys.path.insert(0, str(repo_root))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "08379a1e",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:32.637625Z",
     "iopub.status.busy": "2026-05-14T05:10:32.637192Z",
     "iopub.status.idle": "2026-05-14T05:10:39.790743Z",
     "shell.execute_reply": "2026-05-14T05:10:39.789223Z"
    }
   },
   "outputs": [],
   "source": [
    "import importlib.util\n",
    "import itertools\n",
    "import json\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import pepflow as pf\n",
    "import sympy as sp\n",
    "from pepflow.lyapunov_utils import decompose_rankr_symmetric, independent_subset"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ed4bd1ee",
   "metadata": {},
   "source": [
    "## Function Definitions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ab9963f3",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:39.794890Z",
     "iopub.status.busy": "2026-05-14T05:10:39.794079Z",
     "iopub.status.idle": "2026-05-14T05:10:39.801165Z",
     "shell.execute_reply": "2026-05-14T05:10:39.799702Z"
    }
   },
   "outputs": [],
   "source": [
    "L = pf.Parameter(\"L\")\n",
    "f = pf.SmoothConvexFunction(is_basis=True, tags=[\"f\"], L=L)\n",
    "g = pf.ConvexFunction(is_basis=True, tags=[\"g\"])\n",
    "h = (f + g).add_tag(\"h\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7a68bf8c",
   "metadata": {},
   "source": [
    "## PEP Setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "ea4d5181",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:39.805749Z",
     "iopub.status.busy": "2026-05-14T05:10:39.805198Z",
     "iopub.status.idle": "2026-05-14T05:10:39.822011Z",
     "shell.execute_reply": "2026-05-14T05:10:39.818245Z"
    }
   },
   "outputs": [],
   "source": [
    "def make_ctx_proximal_gradient_descent(ctx_name: str, N, params=None) -> pf.PEPContext:\n",
    "    ctx = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    h.set_stationary_point(\"x_star\")\n",
    "\n",
    "    for i in range(int(N)):\n",
    "        y = x - (1 / L) * f.grad(x)\n",
    "        y.add_tag(f\"y_{i + 1}\")\n",
    "        x = g.prox(y, 1 / L, tag=f\"x_{i + 1}\")\n",
    "\n",
    "    return ctx\n",
    "\n",
    "\n",
    "def get_pep_setup(N, params):\n",
    "    R = pf.Parameter(\"R\")\n",
    "    ctx = make_ctx_proximal_gradient_descent(f\"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(h(ctx[f\"x_{N}\"]) - h(ctx[\"x_star\"]))\n",
    "    return ctx, pb, h"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e2c1933",
   "metadata": {},
   "source": [
    "## Numerical Evidence"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d4204390",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:39.825977Z",
     "iopub.status.busy": "2026-05-14T05:10:39.825486Z",
     "iopub.status.idle": "2026-05-14T05:10:40.392953Z",
     "shell.execute_reply": "2026-05-14T05:10:40.391645Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N=1: PEP=0.24998477, L R^2/(4N)=0.25000000\n",
      "N=2: PEP=0.12500019, L R^2/(4N)=0.12500000\n",
      "N=3: PEP=0.08332796, L R^2/(4N)=0.08333333\n",
      "N=4: PEP=0.06249989, L R^2/(4N)=0.06250000\n",
      "N=5: PEP=0.04999491, L R^2/(4N)=0.05000000\n",
      "N=6: PEP=0.04166338, L R^2/(4N)=0.04166667\n",
      "N=7: PEP=0.03571110, L R^2/(4N)=0.03571429\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "state_path = Path(\n",
    "    \"examples/proximal_gradient_descent/state/proximal_gradient_descent_b1.json\"\n",
    ")\n",
    "if not state_path.exists():\n",
    "    state_path = Path(\"state/proximal_gradient_descent_b1.json\")\n",
    "state = json.loads(state_path.read_text())\n",
    "\n",
    "Ns = np.array([row[\"N\"] for row in state[\"sweep_results\"]], dtype=float)\n",
    "pep_values = np.array([row[\"opt_value\"] for row in state[\"sweep_results\"]], dtype=float)\n",
    "rate_values = np.array([row[\"expected\"] for row in state[\"sweep_results\"]], dtype=float)\n",
    "\n",
    "for N, value, rate in zip(Ns.astype(int), pep_values, rate_values):\n",
    "    print(f\"N={N}: PEP={value:.8f}, L R^2/(4N)={rate:.8f}\")\n",
    "\n",
    "grid = np.linspace(float(Ns.min()), float(Ns.max()), 200)\n",
    "plt.figure(figsize=(6, 4))\n",
    "plt.plot(grid, 1 / (4 * grid), label=r\"$1/(4N)$\")\n",
    "plt.scatter(Ns, pep_values, color=\"tab:blue\", label=\"PEP values\")\n",
    "plt.xlabel(\"N\")\n",
    "plt.ylabel(r\"$h(x_N)-h(x_\\star)$\")\n",
    "plt.legend()\n",
    "plt.grid(alpha=0.3)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "30320b07",
   "metadata": {},
   "source": [
    "## Dense and Relaxed Proof Solves\n",
    "\n",
    "Block 2 solves the full PEP at `N_verify=4`, then enforces the structural PGM sparsity pattern. The active interpolation inequalities are the consecutive pairs and the `x_star` row for both `f` and `g`; the relaxed optimum remains the tight value `1/(4N)` within SDP tolerance."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a5b948ba",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:40.396931Z",
     "iopub.status.busy": "2026-05-14T05:10:40.396489Z",
     "iopub.status.idle": "2026-05-14T05:10:40.407151Z",
     "shell.execute_reply": "2026-05-14T05:10:40.405842Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N_verify: 4\n",
      "relaxed opt value: 0.0625082451\n",
      "tau: 0.0625001153\n",
      "closed-form rate at L=R=1: 0.0625000000\n",
      "relaxed constraints: 43\n",
      "basis vectors:\n",
      "  x_0\n",
      "  x_star\n",
      "  grad_f(x_star)\n",
      "  grad_f(x_0)\n",
      "  grad_g(x_1)\n",
      "  grad_f(x_1)\n",
      "  grad_g(x_2)\n",
      "  grad_f(x_2)\n",
      "  grad_g(x_3)\n",
      "  grad_f(x_3)\n",
      "  grad_g(x_4)\n",
      "  grad_f(x_4)\n"
     ]
    }
   ],
   "source": [
    "b2_path = Path(\n",
    "    \"examples/proximal_gradient_descent/state/proximal_gradient_descent_b2.json\"\n",
    ")\n",
    "if not b2_path.exists():\n",
    "    b2_path = Path(\"state/proximal_gradient_descent_b2.json\")\n",
    "b2 = json.loads(b2_path.read_text())\n",
    "\n",
    "print(\"N_verify:\", b2[\"N_verify\"])\n",
    "print(\"relaxed opt value:\", f\"{b2['opt_value']:.10f}\")\n",
    "print(\"tau:\", f\"{b2['tau_sol']:.10f}\")\n",
    "print(\"closed-form rate at L=R=1:\", f\"{1 / (4 * b2['N_verify']):.10f}\")\n",
    "print(\"relaxed constraints:\", len(b2[\"relaxed_constraints\"]))\n",
    "print(\"basis vectors:\")\n",
    "for name in b2[\"basis_vectors\"]:\n",
    "    print(\" \", name)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bf971799",
   "metadata": {},
   "source": [
    "## Closed-Form Lambda and Gamma\n",
    "\n",
    "For `N=N_verify`, write `x_star` as index `N+1`. The nonzero `f`-interpolation multipliers are\n",
    "\n",
    "$$\\lambda_{i,i+1}=\\frac{i+1}{2N-i},\\quad i=0,\\ldots,N-1,$$\n",
    "\n",
    "and the `x_star` row is filled by telescoping differences:\n",
    "\n",
    "$$\\lambda_{\\star,0}=\\lambda_{0,1},\\qquad\n",
    "\\lambda_{\\star,j}=\\lambda_{j,j+1}-\\lambda_{j-1,j}\\ (1\\le j<N),\\qquad\n",
    "\\lambda_{\\star,N}=1-\\lambda_{N-1,N}.$$\n",
    "\n",
    "The nonzero `g`-interpolation multipliers are\n",
    "\n",
    "$$\\gamma_{i,i+1}=\\frac{i}{2N-i},\\quad i=1,\\ldots,N-1,$$\n",
    "\n",
    "and\n",
    "\n",
    "$$\\gamma_{\\star,j}=\\frac{2N}{(2N+1-j)(2N-j)},\\quad j=1,\\ldots,N.$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "69daa474",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:40.410693Z",
     "iopub.status.busy": "2026-05-14T05:10:40.410228Z",
     "iopub.status.idle": "2026-05-14T05:10:40.447233Z",
     "shell.execute_reply": "2026-05-14T05:10:40.443567Z"
    }
   },
   "outputs": [
    {
     "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.125 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.286 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.5 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.8 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.125 & 0.161 & 0.214 & 0.3 & 0.2 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_1 & x_2 & x_3 & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.0 & 0.143 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.333 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.6 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.143 & 0.19 & 0.267 & 0.4 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lambda match: True\n",
      "gamma match: True\n"
     ]
    }
   ],
   "source": [
    "def tag_to_index(tag, N):\n",
    "    suffix = tag.split(\"_\", 1)[1]\n",
    "    return int(suffix) if suffix.isdigit() else N + 1\n",
    "\n",
    "\n",
    "def lamb(tag_i, tag_j, N):\n",
    "    i = tag_to_index(tag_i, N)\n",
    "    j = tag_to_index(tag_j, N)\n",
    "    if i == N + 1:\n",
    "        if j == 0:\n",
    "            return lamb(\"x_0\", \"x_1\", N)\n",
    "        if j < N:\n",
    "            return lamb(f\"x_{j}\", f\"x_{j + 1}\", N) - lamb(f\"x_{j - 1}\", f\"x_{j}\", N)\n",
    "        if j == N:\n",
    "            return 1 - lamb(f\"x_{N - 1}\", f\"x_{N}\", N)\n",
    "    if i < N and i + 1 == j:\n",
    "        return sp.Rational(j, 2 * N + 1 - j)\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "def gamm(tag_i, tag_j, N):\n",
    "    i = tag_to_index(tag_i, N)\n",
    "    j = tag_to_index(tag_j, N)\n",
    "    if i == N + 1 and 1 <= j <= N:\n",
    "        return sp.Rational(2 * N, (2 * N + 1 - j) * (2 * N - j))\n",
    "    if i < N and i + 1 == j:\n",
    "        return sp.Rational(j - 1, 2 * N + 1 - j)\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "N_int = b2[\"N_verify\"]\n",
    "lambda_candidate = pf.pprint_labeled_matrix(\n",
    "    lambda ri, ci: lamb(ri, ci, N_int),\n",
    "    b2[\"f_lambda_row_names\"],\n",
    "    b2[\"f_lambda_col_names\"],\n",
    "    return_matrix=True,\n",
    ")\n",
    "gamma_candidate = pf.pprint_labeled_matrix(\n",
    "    lambda ri, ci: gamm(ri, ci, N_int),\n",
    "    b2[\"g_lambda_row_names\"],\n",
    "    b2[\"g_lambda_col_names\"],\n",
    "    return_matrix=True,\n",
    ")\n",
    "print(\"lambda match:\", b2[\"lambda_closed_form_matches\"])\n",
    "print(\"gamma match:\", b2[\"gamma_closed_form_matches\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e3c042de",
   "metadata": {},
   "source": [
    "## S Decomposition and Fixed-N Proof Identity\n",
    "\n",
    "The relaxed Gram dual matrix is represented as a sum of three square families, denoted `S_guess1`, `S_guess2`, and `S_guess3` below. This section verifies\n",
    "\n",
    "$$h(x_N)-h(x_\\star)-\\frac{1}{4N}\\|x_0-x_\\star\\|^2\n",
    "= \\sum \\lambda_{ij} I^f_{ij}+\\sum \\gamma_{ij} I^g_{ij}-S,$$\n",
    "\n",
    "for the fixed proof horizon `N_verify=4`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "8ce9ef23",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:40.452276Z",
     "iopub.status.busy": "2026-05-14T05:10:40.451467Z",
     "iopub.status.idle": "2026-05-14T05:10:41.795541Z",
     "shell.execute_reply": "2026-05-14T05:10:41.794494Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "S match: True\n",
      "S residual max: 0.000110698150180499\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_\\star) & \\nabla f(x_0) & \\nabla g(x_1) & \\nabla f(x_1) & \\nabla g(x_2) & \\nabla f(x_2) & \\nabla g(x_3) & \\nabla f(x_3) & \\nabla g(x_4) & \\nabla f(x_4) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 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 & 0.0 & 0.0 \\\\\\nabla f(x_\\star) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(x_1) & 0.0 & 0.0 & 0.0 & 0.0 & 0.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 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(x_2) & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(x_3) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla g(x_4) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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 & 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": [
      "Proof valid: True\n"
     ]
    }
   ],
   "source": [
    "setup_path = Path(\n",
    "    \"examples/proximal_gradient_descent/proximal_gradient_descent_setup.py\"\n",
    ")\n",
    "if not setup_path.exists():\n",
    "    setup_path = Path(\"proximal_gradient_descent_setup.py\")\n",
    "spec = importlib.util.spec_from_file_location(\"pgm_setup\", setup_path)\n",
    "assert spec is not None and spec.loader is not None\n",
    "pgm_setup = importlib.util.module_from_spec(spec)\n",
    "spec.loader.exec_module(pgm_setup)\n",
    "\n",
    "N = N_int\n",
    "params = {\"L\": sp.S(1), \"R\": sp.S(1)}\n",
    "ctx_cert, pb_cert, _ = pgm_setup.get_pep_setup(sp.S(N), params)\n",
    "pb_cert.set_relaxed_constraints(b2[\"relaxed_constraints\"])\n",
    "result_cert = pb_cert.solve(resolve_parameters=params)\n",
    "pm = pf.ExpressionManager(ctx_cert, resolve_parameters=params)\n",
    "x = ctx_cert.tracked_point(pgm_setup.f)\n",
    "\n",
    "S_guess1 = sum(\n",
    "    sp.Rational(N, (2 * N + 1 - i) * (2 * N - i))\n",
    "    * (\n",
    "        sp.Rational(i, 2 * N) * pgm_setup.f.grad(x[i])\n",
    "        + sp.Rational(2 * N - i, 2 * N) * pgm_setup.f.grad(x[i - 1])\n",
    "        - pgm_setup.f.grad(x[N + 1])\n",
    "    )\n",
    "    ** 2\n",
    "    for i in range(1, N + 1)\n",
    ")\n",
    "S_guess2 = sum(\n",
    "    (\n",
    "        sp.Rational(i - 1, 2 * N - i) * sp.Rational(2 * N + 2, 2 * N + 1)\n",
    "        + sp.Rational(2 * N, (2 * N + 1) * (2 * N - i) ** 2)\n",
    "    )\n",
    "    / 2\n",
    "    * (\n",
    "        pgm_setup.f.grad(x[i - 1])\n",
    "        + pgm_setup.g.grad(x[i])\n",
    "        - sp.Rational(1, 2 * N + 1 - i) * (x[i - 1] - x[N + 1])\n",
    "    )\n",
    "    ** 2\n",
    "    for i in range(1, N + 1)\n",
    ")\n",
    "S_guess3 = sum(\n",
    "    sp.Rational(i, 2 * N + 1 - i)\n",
    "    * sp.Rational(2 * N, 2 * N + 1)\n",
    "    / 2\n",
    "    * (\n",
    "        sp.Rational(2 * N + 1, 2 * N)\n",
    "        * (pgm_setup.f.grad(x[i]) + pgm_setup.g.grad(x[i]))\n",
    "        - sp.Rational(1, 2 * N) * (pgm_setup.f.grad(x[i - 1]) + pgm_setup.g.grad(x[i]))\n",
    "        - sp.Rational(1, 2 * N - i) * (x[i] - x[N + 1])\n",
    "    )\n",
    "    ** 2\n",
    "    for i in range(1, N + 1)\n",
    ")\n",
    "S_guess = S_guess1 + S_guess2 + S_guess3\n",
    "S_sol = result_cert.get_gram_dual_matrix()\n",
    "S_guess_matrix = pm.eval_scalar(S_guess).inner_prod_coords\n",
    "print(\n",
    "    \"S match:\",\n",
    "    np.allclose(S_guess_matrix, S_sol.matrix, atol=b2[\"closed_form_match_tolerance\"]),\n",
    ")\n",
    "print(\"S residual max:\", np.max(np.abs(S_guess_matrix - S_sol.matrix)))\n",
    "\n",
    "interp_sum = pf.Scalar.zero()\n",
    "for xi, xj in itertools.product(\n",
    "    ctx_cert.tracked_point(pgm_setup.f), ctx_cert.tracked_point(pgm_setup.f)\n",
    "):\n",
    "    coeff = lamb(xi.tag, xj.tag, N)\n",
    "    if coeff != 0:\n",
    "        interp_sum += coeff * pgm_setup.f.interp_ineq(xi.tag, xj.tag)\n",
    "for xi, xj in itertools.product(\n",
    "    ctx_cert.tracked_point(pgm_setup.g), ctx_cert.tracked_point(pgm_setup.g)\n",
    "):\n",
    "    coeff = gamm(xi.tag, xj.tag, N)\n",
    "    if coeff != 0:\n",
    "        interp_sum += coeff * pgm_setup.g.interp_ineq(xi.tag, xj.tag)\n",
    "\n",
    "lhs = (\n",
    "    pgm_setup.h(x[N])\n",
    "    - pgm_setup.h(ctx_cert[\"x_star\"])\n",
    "    - sp.Rational(1, 4 * N) * (x[0] - ctx_cert[\"x_star\"]) ** 2\n",
    ")\n",
    "diff = lhs - (interp_sum - S_guess)\n",
    "proof_residual = pm.eval_scalar(diff).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(proof_residual, S_sol.row_names, S_sol.col_names)\n",
    "print(\"Proof valid:\", np.allclose(proof_residual, 0, atol=1e-9))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "39939cea",
   "metadata": {},
   "source": [
    "## Partial-Sum Lyapunov Construction and Rank Profile\n",
    "\n",
    "Block 3 defines partial sums by accumulating the active Block 2 certificate terms in forward PGM order. For `k=0,...,N-1`, let `S_{k+1}` denote the `i=k+1` contribution from the three-square decomposition. The increment is\n",
    "\n",
    "$$\n",
    "V_{k+1}-V_k =\n",
    "\\lambda_{\\star,k} I^f_{\\star,k}\n",
    "+\\lambda_{k,k+1} I^f_{k,k+1}\n",
    "+\\gamma_{k,k+1} I^g_{k,k+1}\n",
    "+\\gamma_{\\star,k+1} I^g_{\\star,k+1}\n",
    "-S_{k+1},\n",
    "$$\n",
    "\n",
    "with the extra terminal term $\\lambda_{\\star,N}I^f_{\\star,N}$ added on the final step. At `N_verify=4`, this produces constant interior rank `4` and a rank-one boundary term at `V_N`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "1770f45c",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:41.798266Z",
     "iopub.status.busy": "2026-05-14T05:10:41.797905Z",
     "iopub.status.idle": "2026-05-14T05:10:41.804491Z",
     "shell.execute_reply": "2026-05-14T05:10:41.803746Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "stored rank profile: [0, 4, 4, 4, 1]\n",
      "coverage identity: lyap[N] == h(x_N)-h(x_star)-1/(4N)||x_0-x_star||^2\n"
     ]
    }
   ],
   "source": [
    "b3_path = Path(\n",
    "    \"examples/proximal_gradient_descent/state/proximal_gradient_descent_b3.json\"\n",
    ")\n",
    "if not b3_path.exists():\n",
    "    b3_path = Path(\"state/proximal_gradient_descent_b3.json\")\n",
    "b3 = json.loads(b3_path.read_text())\n",
    "print(\"stored rank profile:\", b3[\"rank_profile\"])\n",
    "print(\"coverage identity:\", b3[\"coverage_identity\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "97965140",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:41.806587Z",
     "iopub.status.busy": "2026-05-14T05:10:41.806378Z",
     "iopub.status.idle": "2026-05-14T05:10:42.009007Z",
     "shell.execute_reply": "2026-05-14T05:10:42.008204Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rank V_0: 0\n",
      "\n",
      "rank V_1: 4\n",
      "rank V_2: 4\n",
      "rank V_3: 4\n",
      "rank V_4: 1\n",
      "Interior rank is constant: True\n"
     ]
    }
   ],
   "source": [
    "def build_s_terms(ctx, f, g, N):\n",
    "    x = ctx.tracked_point(f)\n",
    "    s_terms = {}\n",
    "    for i in range(1, N + 1):\n",
    "        s1 = (\n",
    "            sp.Rational(N, (2 * N + 1 - i) * (2 * N - i))\n",
    "            * (\n",
    "                sp.Rational(i, 2 * N) * f.grad(x[i])\n",
    "                + sp.Rational(2 * N - i, 2 * N) * f.grad(x[i - 1])\n",
    "                - f.grad(x[N + 1])\n",
    "            )\n",
    "            ** 2\n",
    "        )\n",
    "        s2 = (\n",
    "            (\n",
    "                sp.Rational(i - 1, 2 * N - i) * sp.Rational(2 * N + 2, 2 * N + 1)\n",
    "                + sp.Rational(2 * N, (2 * N + 1) * (2 * N - i) ** 2)\n",
    "            )\n",
    "            / 2\n",
    "            * (\n",
    "                f.grad(x[i - 1])\n",
    "                + g.grad(x[i])\n",
    "                - sp.Rational(1, 2 * N + 1 - i) * (x[i - 1] - x[N + 1])\n",
    "            )\n",
    "            ** 2\n",
    "        )\n",
    "        s3 = (\n",
    "            sp.Rational(i, 2 * N + 1 - i)\n",
    "            * sp.Rational(2 * N, 2 * N + 1)\n",
    "            / 2\n",
    "            * (\n",
    "                sp.Rational(2 * N + 1, 2 * N) * (f.grad(x[i]) + g.grad(x[i]))\n",
    "                - sp.Rational(1, 2 * N) * (f.grad(x[i - 1]) + g.grad(x[i]))\n",
    "                - sp.Rational(1, 2 * N - i) * (x[i] - x[N + 1])\n",
    "            )\n",
    "            ** 2\n",
    "        )\n",
    "        s_terms[i] = s1 + s2 + s3\n",
    "    return s_terms\n",
    "\n",
    "\n",
    "N_int = b3[\"N_verify\"]\n",
    "ctx_lyap, pb_lyap, _ = pgm_setup.get_pep_setup(\n",
    "    sp.S(N_int), {\"L\": sp.S(1), \"R\": sp.S(1)}\n",
    ")\n",
    "pm_lyap = pf.ExpressionManager(\n",
    "    ctx_lyap, resolve_parameters={\"L\": sp.S(1), \"R\": sp.S(1)}\n",
    ")\n",
    "f_lyap, g_lyap, h_lyap = pgm_setup.f, pgm_setup.g, pgm_setup.h\n",
    "s_terms = build_s_terms(ctx_lyap, f_lyap, g_lyap, N_int)\n",
    "\n",
    "lyap = [pf.Scalar.zero()]\n",
    "partial_sum = pf.Scalar.zero()\n",
    "for step in range(N_int):\n",
    "    next_idx = step + 1\n",
    "\n",
    "    coeff = lamb(\"x_star\", f\"x_{step}\", N_int)\n",
    "    if coeff != 0:\n",
    "        partial_sum += coeff * f_lyap.interp_ineq(\n",
    "            \"x_star\", f\"x_{step}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    coeff = lamb(f\"x_{step}\", f\"x_{next_idx}\", N_int)\n",
    "    if coeff != 0:\n",
    "        partial_sum += coeff * f_lyap.interp_ineq(\n",
    "            f\"x_{step}\", f\"x_{next_idx}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    coeff = gamm(f\"x_{step}\", f\"x_{next_idx}\", N_int)\n",
    "    if coeff != 0:\n",
    "        partial_sum += coeff * g_lyap.interp_ineq(\n",
    "            f\"x_{step}\", f\"x_{next_idx}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    coeff = gamm(\"x_star\", f\"x_{next_idx}\", N_int)\n",
    "    if coeff != 0:\n",
    "        partial_sum += coeff * g_lyap.interp_ineq(\n",
    "            \"x_star\", f\"x_{next_idx}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    if step == N_int - 1:\n",
    "        coeff = lamb(\"x_star\", f\"x_{N_int}\", N_int)\n",
    "        if coeff != 0:\n",
    "            partial_sum += coeff * f_lyap.interp_ineq(\n",
    "                \"x_star\", f\"x_{N_int}\", pep_context=ctx_lyap\n",
    "            )\n",
    "\n",
    "    partial_sum -= s_terms[next_idx]\n",
    "    lyap.append(partial_sum)\n",
    "\n",
    "rank_tolerance = b3[\"rank_tolerance\"]\n",
    "ranks = []\n",
    "for k, Vk in enumerate(lyap):\n",
    "    matrix = pm_lyap.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": 11,
   "id": "1ac79137",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.011116Z",
     "iopub.status.busy": "2026-05-14T05:10:42.010917Z",
     "iopub.status.idle": "2026-05-14T05:10:42.024221Z",
     "shell.execute_reply": "2026-05-14T05:10:42.022804Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lyap[4] rank:"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " 1\n",
      "Coverage check: lyap[N] is rank-one boundary term: True\n",
      "Coverage inner residual max: 5.551115123125783e-17\n",
      "Coverage function residual max: 0.0\n"
     ]
    }
   ],
   "source": [
    "x_lyap = ctx_lyap.tracked_point(f_lyap)\n",
    "coverage_target = (\n",
    "    h_lyap(x_lyap[N_int])\n",
    "    - h_lyap(ctx_lyap[\"x_star\"])\n",
    "    - sp.Rational(1, 4 * N_int) * (x_lyap[0] - ctx_lyap[\"x_star\"]) ** 2\n",
    ")\n",
    "coverage_residual = pm_lyap.eval_scalar(lyap[N_int] - coverage_target)\n",
    "M_final = pm_lyap.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] is rank-one boundary term:\", rank_final == 1)\n",
    "print(\n",
    "    \"Coverage inner residual max:\",\n",
    "    np.max(np.abs(coverage_residual.inner_prod_coords.astype(float))),\n",
    ")\n",
    "print(\n",
    "    \"Coverage function residual max:\",\n",
    "    np.max(np.abs(coverage_residual.func_coords.astype(float))),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1270645a",
   "metadata": {},
   "source": [
    "## Identify the vectors composing the Lyapunov function\n",
    "\n",
    "Block 4 starts from the Block 3 partial sums and searches for interpretable rank-spanning vectors. The interior rank is `4`, and the useful basis is expressed using the anchor gap, current iterate gap, current smooth gradient, and the smooth gradient at the minimizer of `h`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "c206cba9",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.026723Z",
     "iopub.status.busy": "2026-05-14T05:10:42.026488Z",
     "iopub.status.idle": "2026-05-14T05:10:42.033719Z",
     "shell.execute_reply": "2026-05-14T05:10:42.032446Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "stored rank profile: [0, 4, 4, 4, 1]\n",
      "basis templates:\n",
      "  interior: [x_0-x_star, x_k-x_star, grad_f(x_k), grad_f(x_star)]\n",
      "  terminal: [x_0-x_star]\n",
      "coefficient residual max: 4.163336342344337e-17\n"
     ]
    }
   ],
   "source": [
    "b4_path = Path(\n",
    "    \"examples/proximal_gradient_descent/state/proximal_gradient_descent_b4.json\"\n",
    ")\n",
    "if not b4_path.exists():\n",
    "    b4_path = Path(\"state/proximal_gradient_descent_b4.json\")\n",
    "b4 = json.loads(b4_path.read_text())\n",
    "\n",
    "print(\"stored rank profile:\", b4[\"rank_profile\"])\n",
    "print(\"basis templates:\")\n",
    "for template in b4[\"basis_templates\"]:\n",
    "    print(\" \", template)\n",
    "print(\"coefficient residual max:\", b4[\"coefficient_pattern_residual_max_abs\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4aad70c5",
   "metadata": {},
   "source": [
    "### Candidate-vector scan\n",
    "\n",
    "The scan uses point-to-solution gaps, smooth gradients, and the structured combinations exposed by the proximal-gradient proof. After filtering to the column space of each interior `V_k`, the selected basis pattern is visible directly in the candidate lists."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "3be3ac72",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.036706Z",
     "iopub.status.busy": "2026-05-14T05:10:42.036465Z",
     "iopub.status.idle": "2026-05-14T05:10:42.040795Z",
     "shell.execute_reply": "2026-05-14T05:10:42.040004Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_1 column-space candidates:\n",
      "   x_0-x_star\n",
      "   x_1-x_star\n",
      "   grad_f(x_1)\n",
      "   grad_f(x_star)\n",
      "V_2 column-space candidates:\n",
      "   x_0-x_star\n",
      "   x_2-x_star\n",
      "   grad_f(x_2)\n",
      "   grad_f(x_star)\n",
      "V_3 column-space candidates:\n",
      "   x_0-x_star\n",
      "   x_3-x_star\n",
      "   grad_f(x_3)\n",
      "   grad_f(x_star)\n"
     ]
    }
   ],
   "source": [
    "for k in range(1, b4[\"N_verify\"]):\n",
    "    print(f\"V_{k} column-space candidates:\")\n",
    "    for label in b4[\"candidate_scan\"].get(str(k), []):\n",
    "        print(\"  \", label)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e378c0ab",
   "metadata": {},
   "source": [
    "### Selected basis pattern\n",
    "\n",
    "For `1 <= k < N`, the selected rank-spanning basis is\n",
    "\n",
    "$$\n",
    "\\mathcal B_k =\n",
    "\\left[\n",
    "x_0-x_\\star,\\;\n",
    "x_k-x_\\star,\\;\n",
    "\\nabla f(x_k),\\;\n",
    "\\nabla f(x_\\star)\n",
    "\\right].\n",
    "$$\n",
    "\n",
    "The terminal boundary uses the rank-one basis $[x_0-x_\\star]$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "309d65a8",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.043151Z",
     "iopub.status.busy": "2026-05-14T05:10:42.042945Z",
     "iopub.status.idle": "2026-05-14T05:10:42.054763Z",
     "shell.execute_reply": "2026-05-14T05:10:42.053578Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: rank 4 basis ['x_0-x_star', 'x_1-x_star', 'grad_f(x_1)', 'grad_f(x_star)']\n",
      "k=2: rank 4 basis ['x_0-x_star', 'x_2-x_star', 'grad_f(x_2)', 'grad_f(x_star)']\n",
      "k=3: rank 4 basis ['x_0-x_star', 'x_3-x_star', 'grad_f(x_3)', 'grad_f(x_star)']\n",
      "k=4: rank 1 basis ['x_0-x_star']\n"
     ]
    }
   ],
   "source": [
    "def V_k_basis(k):\n",
    "    if k == 0:\n",
    "        return []\n",
    "    if k < N_int:\n",
    "        return [\n",
    "            ctx_lyap[\"x_0\"] - ctx_lyap[\"x_star\"],\n",
    "            ctx_lyap[f\"x_{k}\"] - ctx_lyap[\"x_star\"],\n",
    "            f_lyap.grad(ctx_lyap[f\"x_{k}\"]),\n",
    "            f_lyap.grad(ctx_lyap[\"x_star\"]),\n",
    "        ]\n",
    "    return [ctx_lyap[\"x_0\"] - ctx_lyap[\"x_star\"]]\n",
    "\n",
    "\n",
    "def V_k_basis_labels(k):\n",
    "    if k == 0:\n",
    "        return []\n",
    "    if k < N_int:\n",
    "        return [\"x_0-x_star\", f\"x_{k}-x_star\", f\"grad_f(x_{k})\", \"grad_f(x_star)\"]\n",
    "    return [\"x_0-x_star\"]\n",
    "\n",
    "\n",
    "for k in range(1, N_int + 1):\n",
    "    basis = V_k_basis(k)\n",
    "    chosen, _ = independent_subset(\n",
    "        basis,\n",
    "        pep_context=ctx_lyap,\n",
    "        resolve_parameters={\"L\": sp.S(1), \"R\": sp.S(1)},\n",
    "        tol=1e-8,\n",
    "    )\n",
    "    print(f\"k={k}: rank {len(chosen)} basis {V_k_basis_labels(k)}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "04f13b17",
   "metadata": {},
   "source": [
    "### Coefficient matrices\n",
    "\n",
    "For the interior basis order $[a,b,c,d]=[x_0-x_\\star, x_k-x_\\star, \\nabla f(x_k), \\nabla f(x_\\star)]$, the nonzero coefficients are\n",
    "\n",
    "$$\n",
    "C_{00}=-\\frac{1}{4N},\\qquad\n",
    "C_{11}=\\frac{N-k}{(2N-k)^2},\n",
    "$$\n",
    "\n",
    "and with\n",
    "\n",
    "$$\n",
    "B_k=\\frac{k}{2(2N-k)(2N-k+1)},\n",
    "$$\n",
    "\n",
    "we have\n",
    "\n",
    "$$\n",
    "C_{12}=C_{21}=B_k,\\quad C_{22}=-B_k,\\quad C_{23}=C_{32}=B_k,\\quad C_{33}=-B_k.\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "0c0d28c1",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.057076Z",
     "iopub.status.busy": "2026-05-14T05:10:42.056851Z",
     "iopub.status.idle": "2026-05-14T05:10:42.567669Z",
     "shell.execute_reply": "2026-05-14T05:10:42.566453Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1 coefficient residual: 3.469e-17\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_0-x_\\star & x_1-x_\\star & \\nabla f(x_1) & \\nabla f(x_\\star) \\\\\n",
       "        \\hline\n",
       "        x_0-x_\\star & -0.062 & -0.0 & 0.0 & 0.0 \\\\x_1-x_\\star & -0.0 & 0.061 & 0.009 & 0.0 \\\\\\nabla f(x_1) & 0.0 & 0.009 & -0.009 & 0.009 \\\\\\nabla f(x_\\star) & 0.0 & 0.0 & 0.009 & -0.009 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=2 coefficient residual: 4.163e-17\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_0-x_\\star & x_2-x_\\star & \\nabla f(x_2) & \\nabla f(x_\\star) \\\\\n",
       "        \\hline\n",
       "        x_0-x_\\star & -0.063 & -0.0 & -0.0 & -0.0 \\\\x_2-x_\\star & -0.0 & 0.056 & 0.024 & 0.0 \\\\\\nabla f(x_2) & -0.0 & 0.024 & -0.024 & 0.024 \\\\\\nabla f(x_\\star) & -0.0 & 0.0 & 0.024 & -0.024 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=3 coefficient residual: 2.776e-17"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_0-x_\\star & x_3-x_\\star & \\nabla f(x_3) & \\nabla f(x_\\star) \\\\\n",
       "        \\hline\n",
       "        x_0-x_\\star & -0.062 & -0.0 & 0.0 & -0.0 \\\\x_3-x_\\star & -0.0 & 0.04 & 0.05 & 0.0 \\\\\\nabla f(x_3) & 0.0 & 0.05 & -0.05 & 0.05 \\\\\\nabla f(x_\\star) & -0.0 & 0.0 & 0.05 & -0.05 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=4 coefficient residual: 0.000e+00\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|c}\n",
       "         & x_0-x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0-x_\\star & -0.062 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def coeff_pattern(k, N):\n",
    "    if k == 0:\n",
    "        return np.zeros((0, 0))\n",
    "    if k == N:\n",
    "        return np.array([[-float(sp.Rational(1, 4 * N))]])\n",
    "    C = np.zeros((4, 4), dtype=float)\n",
    "    A = float(sp.Rational(N - k, (2 * N - k) ** 2))\n",
    "    B = float(sp.Rational(k, 2 * (2 * N - k) * (2 * N - k + 1)))\n",
    "    C[0, 0] = -float(sp.Rational(1, 4 * N))\n",
    "    C[1, 1] = A\n",
    "    C[1, 2] = C[2, 1] = B\n",
    "    C[2, 2] = -B\n",
    "    C[2, 3] = C[3, 2] = B\n",
    "    C[3, 3] = -B\n",
    "    return C\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],\n",
    "        basis,\n",
    "        pep_context=ctx_lyap,\n",
    "        resolve_parameters={\"L\": sp.S(1), \"R\": sp.S(1)},\n",
    "    )\n",
    "    residual = np.max(np.abs(C - coeff_pattern(k, N_int)))\n",
    "    print(f\"k={k} coefficient residual: {residual:.3e}\")\n",
    "    pf.pprint_labeled_matrix(C, labels, labels)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5aa23671",
   "metadata": {},
   "source": [
    "### Block 4 conclusion\n",
    "\n",
    "The current closed-form candidate is: for `1 <= k < N`,\n",
    "\n",
    "$$\n",
    "V_k =\n",
    "-\\frac{1}{4N}\\|x_0-x_\\star\\|^2\n",
    "\\;+\n",
    "\\frac{N-k}{(2N-k)^2}\\|x_k-x_\\star\\|^2\n",
    "\\;+\n",
    "2B_k\\langle x_k-x_\\star,\\nabla f(x_k)\\rangle\n",
    "-B_k\\|\\nabla f(x_k)\\|^2\n",
    "2B_k\\langle \\nabla f(x_k),\\nabla f(x_\\star)\\rangle\n",
    "-B_k\\|\\nabla f(x_\\star)\\|^2.\n",
    "$$\n",
    "\n",
    "At the boundary, $V_N=-\\frac{1}{4N}\\|x_0-x_\\star\\|^2$. Block 5 will symbolically verify the step, base, and boundary identities for this formula."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "71e2d7d5",
   "metadata": {},
   "source": [
    "## Symbolic Step Recursion Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "be5d27fd",
   "metadata": {},
   "source": [
    "The one-step identity verified by the partial sums is\n",
    "\n",
    "$$\n",
    "V_{k+1}-V_k =\n",
    "\\lambda_{\\star,k} I_f(x_\\star,x_k)\n",
    "+\\lambda_{k,k+1}I_f(x_k,x_{k+1})\n",
    "+\\gamma_{k,k+1}I_g(x_k,x_{k+1})\n",
    "+\\gamma_{\\star,k+1}I_g(x_\\star,x_{k+1})\n",
    "-S_{k+1}.\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "492bbce8",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.571460Z",
     "iopub.status.busy": "2026-05-14T05:10:42.571029Z",
     "iopub.status.idle": "2026-05-14T05:10:42.875396Z",
     "shell.execute_reply": "2026-05-14T05:10:42.874594Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Step identity residual max: 2.7755575615628914e-17\n",
      "Step identity zero: True\n"
     ]
    }
   ],
   "source": [
    "# The symbolic identity is encoded by the exact Block 3 construction.\n",
    "# We verify it in PEPFlow coordinates for every interior step of the certificate.\n",
    "b5_path = Path(\n",
    "    \"examples/proximal_gradient_descent/state/proximal_gradient_descent_b5.json\"\n",
    ")\n",
    "if not b5_path.exists():\n",
    "    b5_path = Path(\"state/proximal_gradient_descent_b5.json\")\n",
    "b5 = json.loads(b5_path.read_text())\n",
    "\n",
    "step_residuals = []\n",
    "for step in range(N_int - 1):\n",
    "    next_idx = step + 1\n",
    "    increment = lyap[next_idx] - lyap[step]\n",
    "    rhs = pf.Scalar.zero()\n",
    "\n",
    "    coeff = lamb(\"x_star\", f\"x_{step}\", N_int)\n",
    "    if coeff != 0:\n",
    "        rhs += coeff * f_lyap.interp_ineq(\"x_star\", f\"x_{step}\", pep_context=ctx_lyap)\n",
    "\n",
    "    coeff = lamb(f\"x_{step}\", f\"x_{next_idx}\", N_int)\n",
    "    if coeff != 0:\n",
    "        rhs += coeff * f_lyap.interp_ineq(\n",
    "            f\"x_{step}\", f\"x_{next_idx}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    coeff = gamm(f\"x_{step}\", f\"x_{next_idx}\", N_int)\n",
    "    if coeff != 0:\n",
    "        rhs += coeff * g_lyap.interp_ineq(\n",
    "            f\"x_{step}\", f\"x_{next_idx}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    coeff = gamm(\"x_star\", f\"x_{next_idx}\", N_int)\n",
    "    if coeff != 0:\n",
    "        rhs += coeff * g_lyap.interp_ineq(\n",
    "            \"x_star\", f\"x_{next_idx}\", pep_context=ctx_lyap\n",
    "        )\n",
    "\n",
    "    rhs -= s_terms[next_idx]\n",
    "    diff = increment - rhs\n",
    "    eval_diff = pm_lyap.eval_scalar(diff)\n",
    "    inner_residual = np.max(np.abs(eval_diff.inner_prod_coords.astype(float)))\n",
    "    func_residual = np.max(np.abs(eval_diff.func_coords.astype(float)))\n",
    "    step_residuals.append(max(inner_residual, func_residual))\n",
    "\n",
    "print(\"Step identity residual max:\", max(step_residuals))\n",
    "print(\"Step identity zero:\", max(step_residuals) < 1e-10)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e4f658e6",
   "metadata": {},
   "source": [
    "## Base Case and Boundary Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4dbc0feb",
   "metadata": {},
   "source": [
    "The base identity is the same recurrence at \\(k=0\\), with \\(V_0=0\\):\n",
    "\n",
    "$$\n",
    "V_1-V_0 =\n",
    "\\lambda_{\\star,0} I_f(x_\\star,x_0)\n",
    "+\\lambda_{0,1}I_f(x_0,x_1)\n",
    "+\\gamma_{\\star,1}I_g(x_\\star,x_1)\n",
    "-S_1.\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "c908299e",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.877897Z",
     "iopub.status.busy": "2026-05-14T05:10:42.877549Z",
     "iopub.status.idle": "2026-05-14T05:10:42.914301Z",
     "shell.execute_reply": "2026-05-14T05:10:42.912691Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Base identity inner residual: 0.0\n",
      "Base identity function residual: 0.0\n",
      "Base identity zero: True\n"
     ]
    }
   ],
   "source": [
    "base_rhs = pf.Scalar.zero()\n",
    "base_rhs += lamb(\"x_star\", \"x_0\", N_int) * f_lyap.interp_ineq(\n",
    "    \"x_star\", \"x_0\", pep_context=ctx_lyap\n",
    ")\n",
    "base_rhs += lamb(\"x_0\", \"x_1\", N_int) * f_lyap.interp_ineq(\n",
    "    \"x_0\", \"x_1\", pep_context=ctx_lyap\n",
    ")\n",
    "base_rhs += gamm(\"x_star\", \"x_1\", N_int) * g_lyap.interp_ineq(\n",
    "    \"x_star\", \"x_1\", pep_context=ctx_lyap\n",
    ")\n",
    "base_rhs -= s_terms[1]\n",
    "base_diff = lyap[1] - lyap[0] - base_rhs\n",
    "base_eval = pm_lyap.eval_scalar(base_diff)\n",
    "base_inner = np.max(np.abs(base_eval.inner_prod_coords.astype(float)))\n",
    "base_func = np.max(np.abs(base_eval.func_coords.astype(float)))\n",
    "print(\"Base identity inner residual:\", base_inner)\n",
    "print(\"Base identity function residual:\", base_func)\n",
    "print(\"Base identity zero:\", max(base_inner, base_func) < 1e-10)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "14e0a670",
   "metadata": {},
   "source": [
    "### Boundary Identity Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a557b69a",
   "metadata": {},
   "source": [
    "The boundary identity is\n",
    "\n",
    "$$\n",
    "V_N=(f+g)(x_N)-(f+g)(x_\\star)-\\frac{1}{4N}\\|x_0-x_\\star\\|^2.\n",
    "$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "fc1b5ff5",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-14T05:10:42.917071Z",
     "iopub.status.busy": "2026-05-14T05:10:42.916836Z",
     "iopub.status.idle": "2026-05-14T05:10:42.927040Z",
     "shell.execute_reply": "2026-05-14T05:10:42.926275Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Boundary identity inner residual: 5.551115123125783e-17\n",
      "Boundary identity function residual: 0.0\n",
      "Boundary identity zero: True\n"
     ]
    }
   ],
   "source": [
    "x_boundary = ctx_lyap.tracked_point(f_lyap)\n",
    "boundary_rhs = (\n",
    "    h_lyap(x_boundary[N_int])\n",
    "    - h_lyap(ctx_lyap[\"x_star\"])\n",
    "    - sp.Rational(1, 4 * N_int) * (x_boundary[0] - ctx_lyap[\"x_star\"]) ** 2\n",
    ")\n",
    "boundary_diff = lyap[N_int] - boundary_rhs\n",
    "boundary_eval = pm_lyap.eval_scalar(boundary_diff)\n",
    "boundary_inner = np.max(np.abs(boundary_eval.inner_prod_coords.astype(float)))\n",
    "boundary_func = np.max(np.abs(boundary_eval.func_coords.astype(float)))\n",
    "print(\"Boundary identity inner residual:\", boundary_inner)\n",
    "print(\"Boundary identity function residual:\", boundary_func)\n",
    "print(\"Boundary identity zero:\", max(boundary_inner, boundary_func) < 1e-10)"
   ]
  }
 ],
 "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
}
