{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "c59dfead",
   "metadata": {},
   "source": [
    "# Bregman Proximal Point Method\n",
    "\n",
    "We study composite convex minimization with a closed, proper, convex objective $f$ that is differentiable relative to a Bregman kernel $h$. The kernel $h$ is closed, proper, convex, differentiable, and defines\n",
    "\n",
    "$$D_h(x,y)=h(x)-h(y)-\\langle \\nabla h(y),x-y\\rangle.$$\n",
    "\n",
    "For a fixed step size $\\alpha>0$, the Bregman proximal point method is\n",
    "\n",
    "$$x_{k+1}=\\operatorname{prox}^{h}_{\\alpha f}(x_k),\\qquad k\\ge 0,$$\n",
    "\n",
    "implemented in PEPFlow as\n",
    "\n",
    "```python\n",
    "x = f.bregman_prox(x, stepsize, h)\n",
    "```\n",
    "\n",
    "The initial condition is $D_h(x_\\star,x_0)\\le R$, where $x_\\star$ minimizes $f$, and the performance metric is $f(x_N)-f(x_\\star)$. The conjectured rate is\n",
    "\n",
    "$$f(x_N)-f(x_\\star)\\le \\frac{R}{\\alpha N}.$$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "acc12fc1",
   "metadata": {},
   "source": [
    "## Proof Statement"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5df36ee",
   "metadata": {},
   "source": [
    "### Theorem\n",
    "\n",
    "Let $f$ be closed, proper, convex, and differentiable relative to the Bregman kernel $h$. Let $h$ be closed, proper, convex, and differentiable, and define\n",
    "\n",
    "$$D_h(x,y)=h(x)-h(y)-\\langle\\nabla h(y),x-y\\rangle.$$\n",
    "\n",
    "For a fixed $\\alpha>0$, consider\n",
    "\n",
    "$$x_{k+1}=\\operatorname{prox}^{h}_{\\alpha f}(x_k).$$\n",
    "\n",
    "Let $x_\\star$ minimize $f$, and assume $D_h(x_\\star,x_0)\\le R$. Define, for $k=0,\\ldots,N$,\n",
    "\n",
    "$$V_k=k\\bigl(f(x_k)-f(x_\\star)\\bigr)+\\frac{1}{\\alpha}D_h(x_\\star,x_k).$$\n",
    "\n",
    "Then\n",
    "\n",
    "$$f(x_N)-f(x_\\star)\\le \\frac{R}{\\alpha N}.$$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dabb2fc2",
   "metadata": {},
   "source": [
    "### Proof outline\n",
    "\n",
    "For any convex function $g$, write the interpolation residual in PEPFlow's sign convention as\n",
    "\n",
    "$$I_g(u,v)=g(v)-g(u)+\\langle\\nabla g(v),u-v\\rangle\\le 0.$$\n",
    "\n",
    "The Bregman proximal update gives\n",
    "\n",
    "$$\\nabla h(x_k)=\\nabla h(x_{k-1})-\\alpha\\nabla f(x_k).$$\n",
    "\n",
    "The Lyapunov increment satisfies the exact identity\n",
    "\n",
    "$$\\begin{aligned}\n",
    "V_k-V_{k-1}={}&I_f(x_\\star,x_k)+(k-1)I_f(x_{k-1},x_k)\\\\\n",
    "&+\\frac{k-1}{\\alpha}I_h(x_{k-1},x_k)+\\frac{k}{\\alpha}I_h(x_k,x_{k-1}),\n",
    "\\end{aligned}$$\n",
    "\n",
    "for $k=1,\\ldots,N$, with the two terms multiplied by $k-1$ equal to zero at $k=1$. All coefficients are nonnegative and all residuals are nonpositive, hence $V_k\\le V_{k-1}$ and $V_N\\le V_0$. Since\n",
    "\n",
    "$$V_0=\\frac{1}{\\alpha}D_h(x_\\star,x_0)\\le\\frac{R}{\\alpha}$$\n",
    "\n",
    "and\n",
    "\n",
    "$$V_N=N\\bigl(f(x_N)-f(x_\\star)\\bigr)+\\frac{1}{\\alpha}D_h(x_\\star,x_N)\\ge N\\bigl(f(x_N)-f(x_\\star)\\bigr),$$\n",
    "\n",
    "we obtain $f(x_N)-f(x_\\star)\\le R/(\\alpha N)$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "a5a7aff4",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:19.103231Z",
     "iopub.status.busy": "2026-05-13T15:14:19.102933Z",
     "iopub.status.idle": "2026-05-13T15:14:24.930154Z",
     "shell.execute_reply": "2026-05-13T15:14:24.929068Z"
    }
   },
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "import sys\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "ROOT = Path.cwd().resolve()\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": "markdown",
   "id": "e5664cbf",
   "metadata": {},
   "source": [
    "## PEP Setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "d753c2df",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:24.932838Z",
     "iopub.status.busy": "2026-05-13T15:14:24.932473Z",
     "iopub.status.idle": "2026-05-13T15:14:24.939576Z",
     "shell.execute_reply": "2026-05-13T15:14:24.938432Z"
    }
   },
   "outputs": [],
   "source": [
    "stepsize = pf.Parameter(\"stepsize\")\n",
    "f = pf.ConvexFunction(is_basis=True, tags=[\"f\"])\n",
    "h = pf.ConvexFunction(is_basis=True, tags=[\"h\"])\n",
    "\n",
    "\n",
    "def bregman_distance(kernel, x, y):\n",
    "    return kernel(x) - kernel(y) - kernel.grad(y) * (x - y)\n",
    "\n",
    "\n",
    "def make_ctx_bppm(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",
    "    f.set_stationary_point(\"x_star\")\n",
    "\n",
    "    for i in range(int(N)):\n",
    "        x = f.bregman_prox(x, stepsize, h, 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_bppm(f\"ctx_{N}\", N, params)\n",
    "    pb = pf.PEPBuilder(ctx)\n",
    "    pb.add_initial_constraint(\n",
    "        bregman_distance(h, ctx[\"x_star\"], ctx[\"x_0\"]).le(R, name=\"initial_condition\")\n",
    "    )\n",
    "    pb.set_performance_metric(f(ctx[f\"x_{N}\"]) - f(ctx[\"x_star\"]))\n",
    "    return ctx, pb, f"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a30b0543",
   "metadata": {},
   "source": [
    "## Numerical Evidence"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "17f201c7",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:24.942730Z",
     "iopub.status.busy": "2026-05-13T15:14:24.942425Z",
     "iopub.status.idle": "2026-05-13T15:14:25.335075Z",
     "shell.execute_reply": "2026-05-13T15:14:25.333921Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N=1: PEP=0.999999433, R/(stepsize*N)=1.000000000\n",
      "N=2: PEP=0.499994290, R/(stepsize*N)=0.500000000\n",
      "N=3: PEP=0.333323951, R/(stepsize*N)=0.333333333\n",
      "N=4: PEP=0.250005744, R/(stepsize*N)=0.250000000\n",
      "N=5: PEP=0.199999165, R/(stepsize*N)=0.200000000\n",
      "N=6: PEP=0.166666658, R/(stepsize*N)=0.166666667\n",
      "N=7: PEP=0.142857975, R/(stepsize*N)=0.142857143\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sweep_results = [\n",
    "    {\"N\": 1, \"opt_value\": 0.999999433171909, \"rate_value\": 1.0},\n",
    "    {\"N\": 2, \"opt_value\": 0.49999429001860696, \"rate_value\": 0.5},\n",
    "    {\"N\": 3, \"opt_value\": 0.33332395148468097, \"rate_value\": 1 / 3},\n",
    "    {\"N\": 4, \"opt_value\": 0.250005743584715, \"rate_value\": 0.25},\n",
    "    {\"N\": 5, \"opt_value\": 0.1999991645331413, \"rate_value\": 0.2},\n",
    "    {\"N\": 6, \"opt_value\": 0.16666665835283603, \"rate_value\": 1 / 6},\n",
    "    {\"N\": 7, \"opt_value\": 0.1428579747978089, \"rate_value\": 1 / 7},\n",
    "]\n",
    "\n",
    "for row in sweep_results:\n",
    "    print(\n",
    "        f\"N={row['N']}: PEP={row['opt_value']:.9f}, \"\n",
    "        f\"R/(stepsize*N)={row['rate_value']:.9f}\"\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 optimum\")\n",
    "plt.plot(Ns, rate_values, \"--\", label=r\"$R/(\\alpha N)$\")\n",
    "plt.xlabel(\"N\")\n",
    "plt.ylabel(r\"$f(x_N)-f(x_\\star)$\")\n",
    "plt.title(\"BPPM numerical rate evidence\")\n",
    "plt.grid(True, alpha=0.3)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0cfd8b7e",
   "metadata": {},
   "source": [
    "## Block 2: Full PEP Certificate\n",
    "\n",
    "At `N=4`, the dense solve and the sparse relaxed solve both attain the candidate value `1/4`. The sparse certificate keeps the star-row `f` inequalities, consecutive `f` descent inequalities, and the `h` inequalities that telescope the Bregman distance and descent terms. The resulting proof residual is identically zero, so the PSD certificate is `S = 0`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b1309e5e",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.338246Z",
     "iopub.status.busy": "2026-05-13T15:14:25.337966Z",
     "iopub.status.idle": "2026-05-13T15:14:25.348129Z",
     "shell.execute_reply": "2026-05-13T15:14:25.346897Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "dense opt:   0.2500057436\n",
      "relaxed opt: 0.2499985477\n",
      "tau:         0.2500002758\n",
      "constraints dropped: 35\n",
      "proof valid: True\n",
      "rate: R/(stepsize*N)\n"
     ]
    }
   ],
   "source": [
    "import importlib.util\n",
    "import json\n",
    "import numpy as np\n",
    "import sympy as sp\n",
    "\n",
    "ROOT = Path.cwd()\n",
    "while ROOT != ROOT.parent and not (ROOT / \"pyproject.toml\").exists():\n",
    "    ROOT = ROOT.parent\n",
    "\n",
    "b2_path = ROOT / \"examples\" / \"bppm\" / \"state\" / \"bppm_b2.json\"\n",
    "b2 = json.load(open(b2_path))\n",
    "print(\"dense opt:  \", f\"{b2['dense_opt_value']:.10f}\")\n",
    "print(\"relaxed opt:\", f\"{b2['opt_value']:.10f}\")\n",
    "print(\"tau:        \", f\"{b2['tau_sol']:.10f}\")\n",
    "print(\"constraints dropped:\", len(b2[\"relaxed_constraints\"]))\n",
    "print(\"proof valid:\", b2[\"proof_valid\"])\n",
    "print(\"rate:\", b2[\"conjectured_rate\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3b3effd3",
   "metadata": {},
   "source": [
    "The nonzero closed-form multipliers are\n",
    "\n",
    "$$\\lambda_f(x_\\star,x_j)=\\frac1N,\\qquad j=1,\\ldots,N,$$\n",
    "\n",
    "$$\\lambda_f(x_i,x_{i+1})=\\frac{i}{N},\\qquad i=1,\n",
    "\\ldots,N-1,$$\n",
    "\n",
    "and, for the Bregman kernel,\n",
    "\n",
    "$$\\lambda_h(x_\\star,x_N)=\\frac{1}{\\alpha N},$$\n",
    "\n",
    "$$\\lambda_h(x_i,x_{i-1})=\\frac{i}{\\alpha N},\\qquad i=1,\n",
    "\\ldots,N,$$\n",
    "\n",
    "$$\\lambda_h(x_i,x_{i+1})=\\frac{i}{\\alpha N},\\qquad i=1,\n",
    "\\ldots,N-1.$$\n",
    "\n",
    "All other interpolation inequalities are relaxed."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "39939c07",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.350890Z",
     "iopub.status.busy": "2026-05-13T15:14:25.350555Z",
     "iopub.status.idle": "2026-05-13T15:14:25.365187Z",
     "shell.execute_reply": "2026-05-13T15:14:25.364204Z"
    }
   },
   "outputs": [
    {
     "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.25 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.5 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.75 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.25 & 0.25 & 0.25 & 0.25 & 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|cccccc}\n",
       "         & x_0 & x_1 & x_2 & x_3 & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.25 & 0.0 & 0.25 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.5 & 0.0 & 0.5 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.75 & 0.0 & 0.75 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 1.0 & 0.0 & 0.0 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 & 0.25 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_int = b2[\"N_verify\"]\n",
    "stepsize_value = sp.S(1)\n",
    "lambda_row_names = b2[\"lambda_row_names\"]\n",
    "lambda_col_names = b2[\"lambda_col_names\"]\n",
    "h_lambda_row_names = b2[\"h_lambda_row_names\"]\n",
    "h_lambda_col_names = b2[\"h_lambda_col_names\"]\n",
    "\n",
    "\n",
    "def idx(tag, N=N_int):\n",
    "    if tag == \"x_star\":\n",
    "        return N + 1\n",
    "    return int(tag.split(\"_\")[1])\n",
    "\n",
    "\n",
    "def lamb(ri, ci, N=N_int):\n",
    "    if ri == \"x_star\" and ci.startswith(\"x_\") and ci != \"x_0\":\n",
    "        j = idx(ci, N)\n",
    "        if 1 <= j <= N:\n",
    "            return sp.Rational(1, N)\n",
    "    if ri.startswith(\"x_\") and ci.startswith(\"x_\"):\n",
    "        i, j = idx(ri, N), idx(ci, N)\n",
    "        if 1 <= i < N and j == i + 1:\n",
    "            return sp.Rational(i, N)\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "def h_lamb(ri, ci, N=N_int, stepsize=stepsize_value):\n",
    "    if ri == \"x_star\" and ci == f\"x_{N}\":\n",
    "        return sp.Rational(1, 1) / (stepsize * N)\n",
    "    if ri.startswith(\"x_\") and ci.startswith(\"x_\"):\n",
    "        i, j = idx(ri, N), idx(ci, N)\n",
    "        if 1 <= i <= N and j == i - 1:\n",
    "            return sp.Rational(i, 1) / (stepsize * N)\n",
    "        if 1 <= i < N and j == i + 1:\n",
    "            return sp.Rational(i, 1) / (stepsize * N)\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "f_candidate = [[lamb(ri, ci) for ci in lambda_col_names] for ri in lambda_row_names]\n",
    "h_candidate = [\n",
    "    [h_lamb(ri, ci) for ci in h_lambda_col_names] for ri in h_lambda_row_names\n",
    "]\n",
    "\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(f_candidate, dtype=object), lambda_row_names, lambda_col_names\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(h_candidate, dtype=object), h_lambda_row_names, h_lambda_col_names\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "85df398d",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.367592Z",
     "iopub.status.busy": "2026-05-13T15:14:25.367262Z",
     "iopub.status.idle": "2026-05-13T15:14:25.541933Z",
     "shell.execute_reply": "2026-05-13T15:14:25.540655Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "S certificate: 0"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "inner-product residual zero: True\n",
      "function-value residual zero: True\n"
     ]
    }
   ],
   "source": [
    "spec = importlib.util.spec_from_file_location(\n",
    "    \"bppm_setup\", ROOT / \"examples\" / \"bppm\" / \"bppm_setup.py\"\n",
    ")\n",
    "if spec is None or spec.loader is None:\n",
    "    raise ImportError(\"Cannot load bppm setup module\")\n",
    "setup = importlib.util.module_from_spec(spec)\n",
    "spec.loader.exec_module(setup)\n",
    "ctx, pb, obj = setup.get_pep_setup(\n",
    "    sp.S(N_int), {\"stepsize\": stepsize_value, \"R\": sp.S(1)}\n",
    ")\n",
    "pm = pf.ExpressionManager(\n",
    "    ctx, resolve_parameters={\"stepsize\": stepsize_value, \"R\": sp.S(1)}\n",
    ")\n",
    "\n",
    "IC = setup.bregman_distance(setup.h, ctx[\"x_star\"], ctx[\"x_0\"])\n",
    "perf = setup.f(ctx[f\"x_{N_int}\"]) - setup.f(ctx[\"x_star\"])\n",
    "proof_residual = perf - sp.Rational(1, 1) / (stepsize_value * N_int) * IC\n",
    "\n",
    "for ri in lambda_row_names:\n",
    "    for ci in lambda_col_names:\n",
    "        coeff = lamb(ri, ci, N_int)\n",
    "        if coeff != 0:\n",
    "            proof_residual -= coeff * setup.f.interp_ineq(ri, ci, pep_context=ctx)\n",
    "\n",
    "for ri in h_lambda_row_names:\n",
    "    for ci in h_lambda_col_names:\n",
    "        coeff = h_lamb(ri, ci, N_int, stepsize_value)\n",
    "        if coeff != 0:\n",
    "            proof_residual -= coeff * setup.h.interp_ineq(ri, ci, pep_context=ctx)\n",
    "\n",
    "residual_eval = pm.eval_scalar(proof_residual)\n",
    "inner_ok = np.allclose(\n",
    "    np.array(residual_eval.inner_prod_coords, dtype=float), 0, atol=1e-10\n",
    ")\n",
    "func_ok = np.allclose(np.array(residual_eval.func_coords, dtype=float), 0, atol=1e-10)\n",
    "print(\"S certificate: 0\")\n",
    "print(\"inner-product residual zero:\", inner_ok)\n",
    "print(\"function-value residual zero:\", func_ok)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "18ec9bd3",
   "metadata": {},
   "source": [
    "## Block 3: Lyapunov Partial Sums And Rank Profile\n",
    "\n",
    "The sparse certificate suggests the standard Bregman proximal point Lyapunov sequence\n",
    "\n",
    "$$V_k = k\\bigl(f(x_k)-f(x_\\star)\\bigr)+\\frac{1}{\\alpha}D_h(x_\\star,x_k).$$\n",
    "\n",
    "Its one-step decrement is a nonnegative combination of interpolation inequalities:\n",
    "\n",
    "$$\\begin{aligned}\n",
    "V_k-V_{k-1}={}&I_f(x_\\star,x_k)+(k-1)I_f(x_{k-1},x_k)\\\\\n",
    "&+\\frac{k-1}{\\alpha}I_h(x_{k-1},x_k)+\\frac{k}{\\alpha}I_h(x_k,x_{k-1}),\n",
    "\\end{aligned}$$\n",
    "\n",
    "where the terms with coefficient $k-1$ vanish for $k=1$. Since each interpolation expression is nonpositive in PEPFlow's convention, $V_k\\le V_{k-1}$. The terminal identity\n",
    "\n",
    "$$V_N=N\\bigl(f(x_N)-f(x_\\star)\\bigr)+\\frac{1}{\\alpha}D_h(x_\\star,x_N)$$\n",
    "\n",
    "leaves the nonnegative Bregman boundary term, giving $f(x_N)-f(x_\\star)\\le D_h(x_\\star,x_0)/(\\alpha N)$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "ac1d8fd5",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.544841Z",
     "iopub.status.busy": "2026-05-13T15:14:25.544415Z",
     "iopub.status.idle": "2026-05-13T15:14:25.551938Z",
     "shell.execute_reply": "2026-05-13T15:14:25.550901Z"
    }
   },
   "outputs": [],
   "source": [
    "b3_path = ROOT / \"examples\" / \"bppm\" / \"state\" / \"bppm_b3.json\"\n",
    "b3 = json.load(open(b3_path))\n",
    "print(\"Block 3 state path:\", b3_path)\n",
    "print(\"decrement identity max residual:\", b3[\"decrement_identity_max_residual\"])\n",
    "print(\"Lyapunov formula:\", b3[\"lyap_formula\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "95eff9c8",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.555867Z",
     "iopub.status.busy": "2026-05-13T15:14:25.555319Z",
     "iopub.status.idle": "2026-05-13T15:14:25.631390Z",
     "shell.execute_reply": "2026-05-13T15:14:25.630264Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rank V_0: 2\n",
      "\n",
      "rank V_1: 2\n",
      "rank V_2: 2\n",
      "rank V_3: 2\n",
      "rank V_4: 2\n",
      "Interior rank is constant: True\n"
     ]
    }
   ],
   "source": [
    "N_int = b3[\"N_verify\"]\n",
    "stepsize_value = sp.S(1)\n",
    "rank_tolerance = b3[\"rank_tolerance\"]\n",
    "\n",
    "spec = importlib.util.spec_from_file_location(\n",
    "    \"bppm_setup\", ROOT / \"examples\" / \"bppm\" / \"bppm_setup.py\"\n",
    ")\n",
    "if spec is None or spec.loader is None:\n",
    "    raise ImportError(\"Cannot load bppm setup module\")\n",
    "setup = importlib.util.module_from_spec(spec)\n",
    "spec.loader.exec_module(setup)\n",
    "ctx, pb, obj = setup.get_pep_setup(\n",
    "    sp.S(N_int), {\"stepsize\": stepsize_value, \"R\": sp.S(1)}\n",
    ")\n",
    "pm = pf.ExpressionManager(\n",
    "    ctx, resolve_parameters={\"stepsize\": stepsize_value, \"R\": sp.S(1)}\n",
    ")\n",
    "\n",
    "\n",
    "def triplet_value(fun, tag):\n",
    "    return ctx.get_triplet_by_point_tag(tag, func=fun).expand()[1]\n",
    "\n",
    "\n",
    "def triplet_grad(fun, tag):\n",
    "    return ctx.get_triplet_by_point_tag(tag, func=fun).expand()[2]\n",
    "\n",
    "\n",
    "def bregman_distance_by_tag(kernel, x_tag, y_tag):\n",
    "    return (\n",
    "        triplet_value(kernel, x_tag)\n",
    "        - triplet_value(kernel, y_tag)\n",
    "        - triplet_grad(kernel, y_tag) * (ctx[x_tag] - ctx[y_tag])\n",
    "    )\n",
    "\n",
    "\n",
    "lyap = []\n",
    "for k in range(N_int + 1):\n",
    "    xk = f\"x_{k}\"\n",
    "    gap = pf.Scalar.zero() if k == 0 else setup.f(ctx[xk]) - setup.f(ctx[\"x_star\"])\n",
    "    lyap.append(\n",
    "        sp.S(k) * gap + bregman_distance_by_tag(setup.h, \"x_star\", xk) / stepsize_value\n",
    "    )\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": 9,
   "id": "dcd8dedc",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.633836Z",
     "iopub.status.busy": "2026-05-13T15:14:25.633515Z",
     "iopub.status.idle": "2026-05-13T15:14:25.716679Z",
     "shell.execute_reply": "2026-05-13T15:14:25.715553Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "one-step decrement identities verified: True\n",
      "lyap[4] rank: 2\n",
      "Terminal boundary term: D_h(x_star, x_N) / stepsize\n"
     ]
    }
   ],
   "source": [
    "max_residual = 0.0\n",
    "for k in range(1, N_int + 1):\n",
    "    rhs = setup.f.interp_ineq(\"x_star\", f\"x_{k}\", pep_context=ctx)\n",
    "    if k > 1:\n",
    "        rhs += sp.S(k - 1) * setup.f.interp_ineq(\n",
    "            f\"x_{k - 1}\", f\"x_{k}\", pep_context=ctx\n",
    "        )\n",
    "        rhs += (\n",
    "            sp.S(k - 1)\n",
    "            / stepsize_value\n",
    "            * setup.h.interp_ineq(f\"x_{k - 1}\", f\"x_{k}\", pep_context=ctx)\n",
    "        )\n",
    "    rhs += (\n",
    "        sp.S(k)\n",
    "        / stepsize_value\n",
    "        * setup.h.interp_ineq(f\"x_{k}\", f\"x_{k - 1}\", pep_context=ctx)\n",
    "    )\n",
    "\n",
    "    residual = pm.eval_scalar(lyap[k] - lyap[k - 1] - rhs)\n",
    "    max_residual = max(\n",
    "        max_residual, float(np.max(np.abs(residual.inner_prod_coords.astype(float))))\n",
    "    )\n",
    "    max_residual = max(\n",
    "        max_residual, float(np.max(np.abs(residual.func_coords.astype(float))))\n",
    "    )\n",
    "\n",
    "final_rank = int(\n",
    "    np.linalg.matrix_rank(\n",
    "        pm.eval_scalar(lyap[N_int]).inner_prod_coords.astype(float),\n",
    "        tol=rank_tolerance,\n",
    "    )\n",
    ")\n",
    "print(\"one-step decrement identities verified:\", max_residual <= 1e-10)\n",
    "print(f\"lyap[{N_int}] rank:\", final_rank)\n",
    "print(\"Terminal boundary term:\", r\"D_h(x_star, x_N) / stepsize\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eb3ad965",
   "metadata": {},
   "source": [
    "## Identify the vectors composing the Lyapunov function\n",
    "\n",
    "Block 4 starts from the Block 3 partial sums and identifies a small, interpretable basis for the quadratic coordinates of each $V_k$. The function-value part is already explicit in\n",
    "\n",
    "$$V_k=k(f(x_k)-f(x_\\star))+\\frac{h(x_\\star)-h(x_k)}{\\alpha}+\\frac{1}{\\alpha}\\langle\\nabla h(x_k),x_k-x_\\star\\rangle.$$\n",
    "\n",
    "Thus the only quadratic term should live in the span of $\\nabla h(x_k)$ and $x_k-x_\\star$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "839d8f23",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.720325Z",
     "iopub.status.busy": "2026-05-13T15:14:25.719374Z",
     "iopub.status.idle": "2026-05-13T15:14:25.733257Z",
     "shell.execute_reply": "2026-05-13T15:14:25.731308Z"
    }
   },
   "outputs": [],
   "source": [
    "from pepflow.lyapunov_utils import decompose_rankr_symmetric, vectors_in_column_space\n",
    "\n",
    "b4_path = ROOT / \"examples\" / \"bppm\" / \"state\" / \"bppm_b4.json\"\n",
    "b4 = json.load(open(b4_path))\n",
    "print(\"Block 4 state path:\", b4_path)\n",
    "print(\"stored rank profile:\", b4[\"rank_profile\"])\n",
    "print(\"basis template:\", b4[\"basis_templates\"][0])\n",
    "print(\"coefficient formula residual:\", b4[\"coefficient_formula_max_residual\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "097962f0",
   "metadata": {},
   "source": [
    "### Candidate-vector scan\n",
    "\n",
    "The scan uses tagged iterates, point-to-solution gaps, $h$-gradients at tagged iterates, and available $f$-gradients. For the interior Lyapunov terms, only the matching pair $x_k-x_\\star$ and $\\nabla h(x_k)$ appears in the column space."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "b18db973",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.738548Z",
     "iopub.status.busy": "2026-05-13T15:14:25.737836Z",
     "iopub.status.idle": "2026-05-13T15:14:25.809000Z",
     "shell.execute_reply": "2026-05-13T15:14:25.807719Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_1 column-space candidates:\n",
      "   x_1-x_star\n",
      "   grad_h(x_1)\n",
      "V_2 column-space candidates:\n",
      "   x_2-x_star\n",
      "   grad_h(x_2)\n",
      "V_3 column-space candidates:\n",
      "   x_3-x_star\n",
      "   grad_h(x_3)\n"
     ]
    }
   ],
   "source": [
    "N_int = b4[\"N_verify\"]\n",
    "stepsize_value = sp.S(1)\n",
    "\n",
    "spec = importlib.util.spec_from_file_location(\n",
    "    \"bppm_setup\", ROOT / \"examples\" / \"bppm\" / \"bppm_setup.py\"\n",
    ")\n",
    "if spec is None or spec.loader is None:\n",
    "    raise ImportError(\"Cannot load bppm setup module\")\n",
    "setup = importlib.util.module_from_spec(spec)\n",
    "spec.loader.exec_module(setup)\n",
    "ctx, pb, obj = setup.get_pep_setup(\n",
    "    sp.S(N_int), {\"stepsize\": stepsize_value, \"R\": sp.S(1)}\n",
    ")\n",
    "pm = pf.ExpressionManager(\n",
    "    ctx, resolve_parameters={\"stepsize\": stepsize_value, \"R\": sp.S(1)}\n",
    ")\n",
    "\n",
    "\n",
    "# Rebuild Block 3 partial sums visibly enough for this scan.\n",
    "def scan_triplet_value(fun, tag):\n",
    "    return ctx.get_triplet_by_point_tag(tag, func=fun).expand()[1]\n",
    "\n",
    "\n",
    "def scan_triplet_grad(fun, tag):\n",
    "    return ctx.get_triplet_by_point_tag(tag, func=fun).expand()[2]\n",
    "\n",
    "\n",
    "def scan_bregman_distance_by_tag(kernel, x_tag, y_tag):\n",
    "    return (\n",
    "        scan_triplet_value(kernel, x_tag)\n",
    "        - scan_triplet_value(kernel, y_tag)\n",
    "        - scan_triplet_grad(kernel, y_tag) * (ctx[x_tag] - ctx[y_tag])\n",
    "    )\n",
    "\n",
    "\n",
    "lyap = []\n",
    "for k in range(N_int + 1):\n",
    "    xk = f\"x_{k}\"\n",
    "    gap = pf.Scalar.zero() if k == 0 else setup.f(ctx[xk]) - setup.f(ctx[\"x_star\"])\n",
    "    lyap.append(\n",
    "        sp.S(k) * gap\n",
    "        + scan_bregman_distance_by_tag(setup.h, \"x_star\", xk) / stepsize_value\n",
    "    )\n",
    "\n",
    "candidate_pairs = []\n",
    "for k in range(N_int + 1):\n",
    "    candidate_pairs.append((f\"x_{k}\", ctx[f\"x_{k}\"]))\n",
    "    candidate_pairs.append((f\"x_{k}-x_star\", ctx[f\"x_{k}\"] - ctx[\"x_star\"]))\n",
    "    candidate_pairs.append((f\"grad_h(x_{k})\", scan_triplet_grad(setup.h, f\"x_{k}\")))\n",
    "candidate_pairs.append((\"x_star\", ctx[\"x_star\"]))\n",
    "for k in range(1, N_int + 1):\n",
    "    candidate_pairs.append((f\"grad_f(x_{k})\", scan_triplet_grad(setup.f, f\"x_{k}\")))\n",
    "\n",
    "candidates = [v for _, v in candidate_pairs]\n",
    "for k in range(1, N_int):\n",
    "    in_col = vectors_in_column_space(\n",
    "        lyap[k],\n",
    "        candidates,\n",
    "        pep_context=ctx,\n",
    "        resolve_parameters={\"stepsize\": stepsize_value, \"R\": sp.S(1)},\n",
    "        rtol=1e-4,\n",
    "        atol=1e-4,\n",
    "    )\n",
    "    print(f\"V_{k} column-space candidates:\")\n",
    "    for label, vector in candidate_pairs:\n",
    "        if any(str(vector) == str(v) for v in in_col):\n",
    "            print(\"  \", label)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "90275109",
   "metadata": {},
   "source": [
    "### Selected basis pattern\n",
    "\n",
    "For every $k=0,\\ldots,N$, use the rank-spanning basis\n",
    "\n",
    "$$\\mathcal B_k=\\left[\\nabla h(x_k),\\; x_k-x_\\star\\right].$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "4bf0e7f4",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.812050Z",
     "iopub.status.busy": "2026-05-13T15:14:25.811681Z",
     "iopub.status.idle": "2026-05-13T15:14:25.822270Z",
     "shell.execute_reply": "2026-05-13T15:14:25.820885Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=0: rank 2 basis ['grad_h(x_0)', 'x_0-x_star']\n",
      "k=1: rank 2 basis ['grad_h(x_1)', 'x_1-x_star']\n",
      "k=2: rank 2 basis ['grad_h(x_2)', 'x_2-x_star']\n",
      "k=3: rank 2 basis ['grad_h(x_3)', 'x_3-x_star']\n",
      "k=4: rank 2 basis ['grad_h(x_4)', 'x_4-x_star']\n"
     ]
    }
   ],
   "source": [
    "def h_grad_by_tag(tag):\n",
    "    return ctx.get_triplet_by_point_tag(tag, func=setup.h).expand()[2]\n",
    "\n",
    "\n",
    "def V_k_basis(k):\n",
    "    return [h_grad_by_tag(f\"x_{k}\"), ctx[f\"x_{k}\"] - ctx[\"x_star\"]]\n",
    "\n",
    "\n",
    "def V_k_basis_labels(k):\n",
    "    return [f\"grad_h(x_{k})\", f\"x_{k}-x_star\"]\n",
    "\n",
    "\n",
    "for k in range(N_int + 1):\n",
    "    B = V_k_basis(k)\n",
    "    coords = [pm.eval_vector(v).coords.astype(float) for v in B]\n",
    "    rank = int(np.linalg.matrix_rank(np.column_stack(coords), tol=1e-8))\n",
    "    print(f\"k={k}: rank {rank} basis {V_k_basis_labels(k)}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "add0c619",
   "metadata": {},
   "source": [
    "### Coefficient matrices\n",
    "\n",
    "In the basis order $[\\nabla h(x_k), x_k-x_\\star]$, the quadratic coefficient matrix is independent of $k$:\n",
    "\n",
    "$$C_k=\\begin{bmatrix}0 & \\frac{1}{2\\alpha}\\\\[2pt]\\frac{1}{2\\alpha} & 0\\end{bmatrix}.$$\n",
    "\n",
    "The function-value coordinates are $k f(x_k)-k f(x_\\star)+(h(x_\\star)-h(x_k))/\\alpha$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "3fbb0174",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.825296Z",
     "iopub.status.busy": "2026-05-13T15:14:25.825003Z",
     "iopub.status.idle": "2026-05-13T15:14:25.952003Z",
     "shell.execute_reply": "2026-05-13T15:14:25.950622Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=0: formula residual 0.00e+00\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & \\nabla h(x_0) & x_0-x_\\star \\\\\n",
       "        \\hline\n",
       "        \\nabla h(x_0) & 0.0 & 0.5 \\\\x_0-x_\\star & 0.5 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: formula residual 1.11e-16\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & \\nabla h(x_1) & x_1-x_\\star \\\\\n",
       "        \\hline\n",
       "        \\nabla h(x_1) & 0.0 & 0.5 \\\\x_1-x_\\star & 0.5 & 0.0 \\\\\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 1.11e-16\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & \\nabla h(x_2) & x_2-x_\\star \\\\\n",
       "        \\hline\n",
       "        \\nabla h(x_2) & 0.0 & 0.5 \\\\x_2-x_\\star & 0.5 & 0.0 \\\\\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 0.00e+00\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & \\nabla h(x_3) & x_3-x_\\star \\\\\n",
       "        \\hline\n",
       "        \\nabla h(x_3) & 0.0 & 0.5 \\\\x_3-x_\\star & 0.5 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=4: formula residual 5.55e-17\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cc}\n",
       "         & \\nabla h(x_4) & x_4-x_\\star \\\\\n",
       "        \\hline\n",
       "        \\nabla h(x_4) & 0.0 & 0.5 \\\\x_4-x_\\star & 0.5 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def coeff_pattern(k, N, stepsize=stepsize_value):\n",
    "    return sp.Matrix(\n",
    "        [\n",
    "            [0, sp.Rational(1, 2) / stepsize],\n",
    "            [sp.Rational(1, 2) / stepsize, 0],\n",
    "        ]\n",
    "    )\n",
    "\n",
    "\n",
    "for k in range(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,\n",
    "        resolve_parameters={\"stepsize\": stepsize_value, \"R\": sp.S(1)},\n",
    "    )\n",
    "    formula = np.array(coeff_pattern(k, N_int, stepsize_value), dtype=float)\n",
    "    residual = float(np.max(np.abs(np.array(C, dtype=float) - formula)))\n",
    "    print(f\"k={k}: formula residual {residual:.2e}\")\n",
    "    pf.pprint_labeled_matrix(C, labels, labels)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3883595d",
   "metadata": {},
   "source": [
    "### Block 4 conclusion\n",
    "\n",
    "The closed-form Lyapunov candidate is\n",
    "\n",
    "$$V_k=k(f(x_k)-f(x_\\star))+\\frac{h(x_\\star)-h(x_k)}{\\alpha}+\\frac{1}{\\alpha}\\langle\\nabla h(x_k),x_k-x_\\star\\rangle.$$\n",
    "\n",
    "Block 5 will verify the base, step, and boundary identities symbolically for general $k$ and $N$."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c3fa8418",
   "metadata": {},
   "source": [
    "## Symbolic Step Recursion Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f3864402",
   "metadata": {},
   "source": [
    "For a generic step $x_{k-1}\\mapsto x_k$, verify\n",
    "\n",
    "$$\\begin{aligned}\n",
    "V_k-V_{k-1}={}&I_f(x_\\star,x_k)+(k-1)I_f(x_{k-1},x_k)\\\\\n",
    "&+\\frac{k-1}{\\alpha}I_h(x_{k-1},x_k)+\\frac{k}{\\alpha}I_h(x_k,x_{k-1}).\n",
    "\\end{aligned}$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero in both inner-product and function-value coordinates."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "c6c43a5c",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:25.956946Z",
     "iopub.status.busy": "2026-05-13T15:14:25.956361Z",
     "iopub.status.idle": "2026-05-13T15:14:26.300598Z",
     "shell.execute_reply": "2026-05-13T15:14:26.299563Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "step identity zero: True\n",
      "inner residual entries: [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, 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",
      "function residual entries: [0, 0, 0, 0, 0, 0]\n"
     ]
    }
   ],
   "source": [
    "b5_path = ROOT / \"examples\" / \"bppm\" / \"state\" / \"bppm_b5.json\"\n",
    "b5 = json.load(open(b5_path))\n",
    "alpha_param = pf.Parameter(\"alpha\")\n",
    "alpha_sym = sp.Symbol(\"alpha\", positive=True)\n",
    "k_sym = sp.Symbol(\"k\", integer=True, positive=True)\n",
    "\n",
    "\n",
    "def coordinate_entries(evaled_scalar):\n",
    "    inner = np.array(evaled_scalar.inner_prod_coords, dtype=object).reshape(-1)\n",
    "    funcs = np.array(evaled_scalar.func_coords, dtype=object).reshape(-1)\n",
    "    return [sp.simplify(x) for x in inner], [sp.simplify(x) for x in funcs]\n",
    "\n",
    "\n",
    "def coordinates_zero(evaled_scalar):\n",
    "    inner, funcs = coordinate_entries(evaled_scalar)\n",
    "    return all(x == 0 for x in inner) and all(x == 0 for x in funcs)\n",
    "\n",
    "\n",
    "def symbolic_bregman(kernel, ctx_obj, x_tag, y_tag):\n",
    "    return (\n",
    "        kernel(ctx_obj[x_tag])\n",
    "        - kernel(ctx_obj[y_tag])\n",
    "        - kernel.grad(ctx_obj[y_tag]) * (ctx_obj[x_tag] - ctx_obj[y_tag])\n",
    "    )\n",
    "\n",
    "\n",
    "ctx_step = pf.PEPContext(\"bppm_step_symbolic_nb\").set_as_current()\n",
    "f_step = pf.ConvexFunction(is_basis=True, tags=[\"f_{step}\"])\n",
    "h_step = pf.ConvexFunction(is_basis=True, tags=[\"h_{step}\"])\n",
    "x_prev = pf.Vector(is_basis=True, tags=[\"x_{k-1}\"])\n",
    "f_step(x_prev)\n",
    "h_step(x_prev)\n",
    "f_step.set_stationary_point(\"x_star\")\n",
    "h_step(ctx_step[\"x_star\"])\n",
    "x_curr = f_step.bregman_prox(x_prev, alpha_param, h_step, tag=\"x_k\")\n",
    "f_step(x_curr)\n",
    "h_step(x_curr)\n",
    "\n",
    "V_prev = (k_sym - 1) * (f_step(ctx_step[\"x_{k-1}\"]) - f_step(ctx_step[\"x_star\"]))\n",
    "V_prev += symbolic_bregman(h_step, ctx_step, \"x_star\", \"x_{k-1}\") / alpha_param\n",
    "V_curr = k_sym * (f_step(ctx_step[\"x_k\"]) - f_step(ctx_step[\"x_star\"]))\n",
    "V_curr += symbolic_bregman(h_step, ctx_step, \"x_star\", \"x_k\") / alpha_param\n",
    "\n",
    "rhs_step = f_step.interp_ineq(\"x_star\", \"x_k\", pep_context=ctx_step)\n",
    "rhs_step += (k_sym - 1) * f_step.interp_ineq(\"x_{k-1}\", \"x_k\", pep_context=ctx_step)\n",
    "rhs_step += (\n",
    "    (k_sym - 1)\n",
    "    / alpha_param\n",
    "    * h_step.interp_ineq(\"x_{k-1}\", \"x_k\", pep_context=ctx_step)\n",
    ")\n",
    "rhs_step += (\n",
    "    k_sym / alpha_param * h_step.interp_ineq(\"x_k\", \"x_{k-1}\", pep_context=ctx_step)\n",
    ")\n",
    "\n",
    "pm_step = pf.ExpressionManager(ctx_step, resolve_parameters={\"alpha\": alpha_sym})\n",
    "step_residual = pm_step.eval_scalar(V_curr - V_prev - rhs_step, sympy_mode=True)\n",
    "print(\"step identity zero:\", coordinates_zero(step_residual))\n",
    "print(\"inner residual entries:\", coordinate_entries(step_residual)[0])\n",
    "print(\"function residual entries:\", coordinate_entries(step_residual)[1])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a46ed4bd",
   "metadata": {},
   "source": [
    "## Base Case Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a27507fb",
   "metadata": {},
   "source": [
    "For the first step $x_0\\mapsto x_1$, verify\n",
    "\n",
    "$$V_1-V_0=I_f(x_\\star,x_1)+\\frac{1}{\\alpha}I_h(x_1,x_0).$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "44ad2b62",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:26.303139Z",
     "iopub.status.busy": "2026-05-13T15:14:26.302809Z",
     "iopub.status.idle": "2026-05-13T15:14:26.317292Z",
     "shell.execute_reply": "2026-05-13T15:14:26.316120Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "base identity zero: True\n",
      "inner residual entries: [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, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]\n",
      "function residual entries: [0, 0, 0, 0, 0]\n"
     ]
    }
   ],
   "source": [
    "ctx_base = pf.PEPContext(\"bppm_base_symbolic_nb\").set_as_current()\n",
    "f_base = pf.ConvexFunction(is_basis=True, tags=[\"f_{base}\"])\n",
    "h_base = pf.ConvexFunction(is_basis=True, tags=[\"h_{base}\"])\n",
    "x0_base = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "h_base(x0_base)\n",
    "f_base.set_stationary_point(\"x_star\")\n",
    "h_base(ctx_base[\"x_star\"])\n",
    "x1_base = f_base.bregman_prox(x0_base, alpha_param, h_base, tag=\"x_1\")\n",
    "f_base(x1_base)\n",
    "h_base(x1_base)\n",
    "\n",
    "V0_base = symbolic_bregman(h_base, ctx_base, \"x_star\", \"x_0\") / alpha_param\n",
    "V1_base = f_base(ctx_base[\"x_1\"]) - f_base(ctx_base[\"x_star\"])\n",
    "V1_base += symbolic_bregman(h_base, ctx_base, \"x_star\", \"x_1\") / alpha_param\n",
    "rhs_base = f_base.interp_ineq(\"x_star\", \"x_1\", pep_context=ctx_base)\n",
    "rhs_base += h_base.interp_ineq(\"x_1\", \"x_0\", pep_context=ctx_base) / alpha_param\n",
    "\n",
    "pm_base = pf.ExpressionManager(ctx_base, resolve_parameters={\"alpha\": alpha_sym})\n",
    "base_residual = pm_base.eval_scalar(V1_base - V0_base - rhs_base, sympy_mode=True)\n",
    "print(\"base identity zero:\", coordinates_zero(base_residual))\n",
    "print(\"inner residual entries:\", coordinate_entries(base_residual)[0])\n",
    "print(\"function residual entries:\", coordinate_entries(base_residual)[1])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7c399742",
   "metadata": {},
   "source": [
    "### Boundary Identity Symbolic Verification"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "76862751",
   "metadata": {},
   "source": [
    "At the terminal index, verify the defining boundary identity\n",
    "\n",
    "$$V_N=N\\bigl(f(x_N)-f(x_\\star)\\bigr)+\\frac{1}{\\alpha}D_h(x_\\star,x_N).$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero. This identity, together with $D_h(x_\\star,x_N)\\ge0$, converts $V_N\\le V_0$ into the final performance bound."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "a20dade7",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T15:14:26.320170Z",
     "iopub.status.busy": "2026-05-13T15:14:26.319659Z",
     "iopub.status.idle": "2026-05-13T15:14:26.335225Z",
     "shell.execute_reply": "2026-05-13T15:14:26.334084Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "boundary identity zero: True\n",
      "inner residual entries: [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]\n",
      "function residual entries: [0, 0, 0, 0]\n"
     ]
    }
   ],
   "source": [
    "N_sym = sp.Symbol(\"N\", integer=True, positive=True)\n",
    "ctx_boundary = pf.PEPContext(\"bppm_boundary_symbolic_nb\").set_as_current()\n",
    "f_boundary = pf.ConvexFunction(is_basis=True, tags=[\"f_{boundary}\"])\n",
    "h_boundary = pf.ConvexFunction(is_basis=True, tags=[\"h_{boundary}\"])\n",
    "xN_boundary = pf.Vector(is_basis=True, tags=[\"x_N\"])\n",
    "f_boundary(xN_boundary)\n",
    "h_boundary(xN_boundary)\n",
    "f_boundary.set_stationary_point(\"x_star\")\n",
    "h_boundary(ctx_boundary[\"x_star\"])\n",
    "\n",
    "V_N_boundary = N_sym * (\n",
    "    f_boundary(ctx_boundary[\"x_N\"]) - f_boundary(ctx_boundary[\"x_star\"])\n",
    ")\n",
    "V_N_boundary += (\n",
    "    symbolic_bregman(h_boundary, ctx_boundary, \"x_star\", \"x_N\") / alpha_param\n",
    ")\n",
    "rhs_boundary = N_sym * (\n",
    "    f_boundary(ctx_boundary[\"x_N\"]) - f_boundary(ctx_boundary[\"x_star\"])\n",
    ")\n",
    "rhs_boundary += (\n",
    "    symbolic_bregman(h_boundary, ctx_boundary, \"x_star\", \"x_N\") / alpha_param\n",
    ")\n",
    "\n",
    "pm_boundary = pf.ExpressionManager(\n",
    "    ctx_boundary, resolve_parameters={\"alpha\": alpha_sym}\n",
    ")\n",
    "boundary_residual = pm_boundary.eval_scalar(\n",
    "    V_N_boundary - rhs_boundary, sympy_mode=True\n",
    ")\n",
    "print(\"boundary identity zero:\", coordinates_zero(boundary_residual))\n",
    "print(\"inner residual entries:\", coordinate_entries(boundary_residual)[0])\n",
    "print(\"function residual entries:\", coordinate_entries(boundary_residual)[1])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "003c50ee",
   "metadata": {},
   "source": [
    "## Final Theorem\n",
    "\n",
    "The verified Lyapunov certificate is\n",
    "\n",
    "$$V_k=k\\bigl(f(x_k)-f(x_\\star)\\bigr)+\\frac{1}{\\alpha}D_h(x_\\star,x_k).$$\n",
    "\n",
    "The symbolic base, step, and boundary identities above prove\n",
    "\n",
    "$$f(x_N)-f(x_\\star)\\le\\frac{D_h(x_\\star,x_0)}{\\alpha N}\\le\\frac{R}{\\alpha N}.$$"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "pepflow (3.11.13)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
