{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "9c7e4e55",
   "metadata": {},
   "source": [
    "# Dual FEG Lyapunov Workflow\n",
    "\n",
    "We study Dual Fast Extragradient for a monotone and $L$-Lipschitz continuous operator $A$ with $\\alpha = 1/L$. The zero point $x_\\star$ satisfies $A(x_\\star)=0$, the initial condition is $\\|x_0-x_\\star\\|^2 \\le R^2$, and this Block 1 setup uses the residual performance metric $\\|A(x_N)\\|^2$. The recursion is initialized with $z_0=0$.\n",
    "\n",
    "For $k=0,\\ldots,N-1$,\n",
    "\n",
    "$$x_{k+1/2}=x_k-\\alpha z_k-\\alpha A(x_k),$$\n",
    "\n",
    "$$x_{k+1}=x_{k+1/2}-\\frac{N-k-1}{N-k}\\alpha\\left(A(x_{k+1/2})-A(x_k)\\right),$$\n",
    "\n",
    "$$z_{k+1}=\\frac{N-k-1}{N-k}z_k-\\frac{1}{N-k}A(x_{k+1/2}).$$\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0b18d570",
   "metadata": {},
   "source": [
    "## Proof Statement\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3716050d",
   "metadata": {},
   "source": [
    "### Theorem\n",
    "\n",
    "Assume that $A$ is monotone and $L$-Lipschitz continuous, and let $x_{\\star}$ satisfy $A(x_{\\star})=0$. Run Dual FEG with $\\alpha=1/L$ and $z_{0}=0$:\n",
    "\n",
    "$$x_{k+1/2}=x_{k}-\\alpha z_{k}-\\alpha A(x_{k}),$$\n",
    "\n",
    "$$x_{k+1}=x_{k+1/2}-\\frac{N-k-1}{N-k}\\alpha\\left(A(x_{k+1/2})-A(x_{k})\\right),$$\n",
    "\n",
    "$$z_{k+1}=\\frac{N-k-1}{N-k}z_{k}-\\frac{1}{N-k}A(x_{k+1/2}).$$\n",
    "\n",
    "For $1\\le k\\le N-1$, define $q_{k}=z_{k}+A(x_{N})$ and\n",
    "\n",
    "$$V_{k}=2\\|q_{k}\\|^{2}-\\frac{4}{N-k}\\langle q_{k},x_{k}-x_{N}\\rangle-2\\|A(x_{N})\\|^{2}+\\frac{4}{N}\\langle A(x_{N}),x_{0}-x_{N}\\rangle.$$\n",
    "\n",
    "Set $V_{0}=0$ and $V_{N}=\\frac{4}{N^{2}}\\|x_{0}-x_{\\star}\\|^{2}$. Then\n",
    "\n",
    "$$\\|A(x_{N})\\|^{2}\\le \\frac{4L^{2}}{N^{2}}\\|x_{0}-x_{\\star}\\|^{2}\\le \\frac{4L^{2}R^{2}}{N^{2}}.$$\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "28a4330b",
   "metadata": {},
   "source": [
    "### Proof outline\n",
    "\n",
    "Use the residual notation\n",
    "\n",
    "$$M(u,v)=-\\langle u-v,A(u)-A(v)\\rangle\\le 0$$\n",
    "\n",
    "for monotonicity, and\n",
    "\n",
    "$$G(u,v)=\\|A(u)-A(v)\\|^{2}-L^{2}\\|u-v\\|^{2}\\le 0$$\n",
    "\n",
    "for Lipschitz continuity. In the normalized notebook identities, $L=1$.\n",
    "\n",
    "For $0\\le k\\le N-2$,\n",
    "\n",
    "$$V_{k+1}-V_{k}=-\\frac{2}{(N-k)^{2}}G(x_{k},x_{k+1/2})-\\frac{4}{(N-k-1)(N-k)}M(x_{k+1/2},x_{N}).$$\n",
    "\n",
    "The base case is this identity at $k=0$, with $V_{0}=0$. The final boundary increment is\n",
    "\n",
    "$$V_{N}-V_{N-1}=-2G(x_{N-1},x_{N})-\\frac{4}{N}M(x_{N},x_{\\star})+\\left\\|\\frac{2}{N}(x_{0}-x_{\\star})-A(x_{N})\\right\\|^{2}+\\|A(x_{N})\\|^{2}.$$\n",
    "\n",
    "Since $M\\le 0$, $G\\le 0$, and the square is nonnegative, these identities telescope to\n",
    "\n",
    "$$\\frac{4}{N^{2}}\\|x_{0}-x_{\\star}\\|^{2}\\ge \\|A(x_{N})\\|^{2}.$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "bfae9c23",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:15.016969Z",
     "iopub.status.busy": "2026-05-13T16:14:15.016749Z",
     "iopub.status.idle": "2026-05-13T16:14:20.118989Z",
     "shell.execute_reply": "2026-05-13T16:14:20.117770Z"
    }
   },
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "import json\n",
    "import sys\n",
    "\n",
    "REPO_ROOT = Path.cwd()\n",
    "while not (REPO_ROOT / \"pyproject.toml\").exists() and REPO_ROOT != REPO_ROOT.parent:\n",
    "    REPO_ROOT = REPO_ROOT.parent\n",
    "if str(REPO_ROOT) not in sys.path:\n",
    "    sys.path.insert(0, str(REPO_ROOT))\n",
    "\n",
    "import matplotlib.pyplot as plt  # noqa: E402\n",
    "import pepflow as pf  # noqa: E402\n",
    "\n",
    "from examples.dual_feg.dual_feg_setup import A, L, alpha  # noqa: E402"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "856fca7d",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.121772Z",
     "iopub.status.busy": "2026-05-13T16:14:20.121439Z",
     "iopub.status.idle": "2026-05-13T16:14:20.127836Z",
     "shell.execute_reply": "2026-05-13T16:14:20.126783Z"
    }
   },
   "outputs": [],
   "source": [
    "def make_ctx_dual_feg(ctx_name: str, N, params):\n",
    "    del params\n",
    "    ctx = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    z = pf.Vector.zero()\n",
    "    z.add_tag(\"z_0\")\n",
    "    A.set_zero_point(\"x_star\")\n",
    "\n",
    "    for k in range(int(N)):\n",
    "        rho = (N - k - 1) / (N - k)\n",
    "        eta = 1 / (N - k)\n",
    "        Ax = A(x)\n",
    "        x_half = x - alpha * z - alpha * Ax\n",
    "        x_half.add_tag(f\"x_{k + 1}_half\")\n",
    "        A_x_half = A(x_half)\n",
    "        x = x_half - rho * alpha * (A_x_half - Ax)\n",
    "        x.add_tag(f\"x_{k + 1}\")\n",
    "        z = rho * z - eta * A_x_half\n",
    "        z.add_tag(f\"z_{k + 1}\")\n",
    "\n",
    "    return ctx\n",
    "\n",
    "\n",
    "def get_pep_setup_notebook(N, params):\n",
    "    R = pf.Parameter(\"R\")\n",
    "    ctx = make_ctx_dual_feg(f\"ctx_notebook_{N}\", N, params)\n",
    "    pb = pf.PEPBuilder(ctx)\n",
    "    pb.add_initial_constraint(\n",
    "        ((ctx[\"x_0\"] - ctx[\"x_star\"]) ** 2).le(R**2, name=\"initial_condition\")\n",
    "    )\n",
    "    pb.set_performance_metric(A(ctx[f\"x_{N}\"]) ** 2)\n",
    "    return ctx, pb, A"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ee4311cb",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.130364Z",
     "iopub.status.busy": "2026-05-13T16:14:20.129982Z",
     "iopub.status.idle": "2026-05-13T16:14:20.440941Z",
     "shell.execute_reply": "2026-05-13T16:14:20.440338Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N=1: 1.99999864\n",
      "N=2: 1.00005152\n",
      "N=3: 0.44447345\n",
      "N=4: 0.25000040\n",
      "N=5: 0.16000072\n",
      "N=6: 0.11111809\n",
      "N=7: 0.08163549\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "state_path = REPO_ROOT / \"examples/dual_feg/state/dual_feg_b1.json\"\n",
    "state = json.loads(state_path.read_text())\n",
    "results = state[\"sweep_results\"]\n",
    "Ns = [r[\"N\"] for r in results]\n",
    "values = [float(r[\"opt_value\"]) for r in results]\n",
    "guess = [4 / (N**2) for N in Ns]\n",
    "\n",
    "for N, value in zip(Ns, values):\n",
    "    print(f\"N={N}: {value:.8f}\")\n",
    "\n",
    "plt.figure(figsize=(6, 4))\n",
    "plt.plot(Ns, values, \"o-\", label=\"PEP value\")\n",
    "plt.plot(Ns, guess, \"--\", label=r\"$4/N^2$ guess\")\n",
    "plt.xlabel(\"N\")\n",
    "plt.ylabel(r\"$\\|A(x_N)\\|^2$\")\n",
    "plt.legend()\n",
    "plt.grid(True, alpha=0.3)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2565277d",
   "metadata": {},
   "source": [
    "## Dense Proof Solve\n",
    "\n",
    "At `N_verify = 4`, the dense PEP solve gives a value numerically equal to the conjectured residual rate `4/N^2 = 1/4`. The dense certificate shows a small active set in the monotonicity and Lipschitz interpolation inequalities.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b820ff80",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.443640Z",
     "iopub.status.busy": "2026-05-13T16:14:20.443430Z",
     "iopub.status.idle": "2026-05-13T16:14:20.449973Z",
     "shell.execute_reply": "2026-05-13T16:14:20.448660Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "N: 4\n",
      "opt_value: 0.2500004040056193\n",
      "tau_sol: 0.2500003952284651\n",
      "basis_vectors: ['x_0', 'x_star', 'A(x_0)', 'A(x_1_half)', 'A(x_1)', 'A(x_2_half)', 'A(x_2)', 'A(x_3_half)', 'A(x_3)', 'A(x_4_half)', 'A(x_4)']\n",
      "\n",
      "Monotone Operator Inequality\n",
      "  x_1_half   x_4        0.33333358\n",
      "  x_2_half   x_4        0.66666410\n",
      "  x_3_half   x_4        1.99999725\n",
      "  x_4        x_star     0.99999777\n",
      "\n",
      "Lipschitz Operator Inequality\n",
      "  x_0        x_1_half   0.12500015\n",
      "  x_1        x_2_half   0.22222206\n",
      "  x_2        x_3_half   0.49999987\n",
      "  x_3        x_4        1.99999871\n",
      "  x_4        x_4_half   0.55281785\n"
     ]
    }
   ],
   "source": [
    "dense = json.loads(\n",
    "    (REPO_ROOT / \"examples/dual_feg/state/dual_feg_dense.json\").read_text()\n",
    ")\n",
    "print(\"N:\", dense[\"N\"])\n",
    "print(\"opt_value:\", dense[\"opt_value\"])\n",
    "print(\"tau_sol:\", dense[\"tau_sol\"])\n",
    "print(\"basis_vectors:\", dense[\"basis_vectors\"])\n",
    "\n",
    "for group, data in dense[\"lambda_groups\"].items():\n",
    "    print(\"\\n\" + group)\n",
    "    for i, ri in enumerate(data[\"row_names\"]):\n",
    "        for j, ci in enumerate(data[\"col_names\"]):\n",
    "            value = float(data[\"matrix\"][i][j])\n",
    "            if abs(value) > 1e-5:\n",
    "                print(f\"  {ri:10s} {ci:10s} {value:.8f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f1119f43",
   "metadata": {},
   "source": [
    "## Sparse Relaxation\n",
    "\n",
    "The preserved sparse certificate keeps only the following entries: monotone pairs `(x_j_half, x_N)` for `j=1,...,N-1`, monotone pair `(x_N, x_star)`, Lipschitz pairs `(x_i, x_{i+1}_half)` for `i=0,...,N-2`, and Lipschitz pair `(x_{N-1}, x_N)`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "3013714e",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.452463Z",
     "iopub.status.busy": "2026-05-13T16:14:20.452231Z",
     "iopub.status.idle": "2026-05-13T16:14:20.458342Z",
     "shell.execute_reply": "2026-05-13T16:14:20.457648Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "dense opt: 0.2500004040056193\n",
      "relaxed opt: 0.2500011995865052\n",
      "preserved: True\n",
      "relaxed constraints dropped: 154\n",
      "\n",
      "Monotone Operator Inequality\n",
      "  x_1_half   x_4        0.33333263\n",
      "  x_2_half   x_4        0.66666532\n",
      "  x_3_half   x_4        1.99999604\n",
      "  x_4        x_star     0.99999819\n",
      "\n",
      "Lipschitz Operator Inequality\n",
      "  x_0        x_1_half   0.12499972\n",
      "  x_1        x_2_half   0.22222176\n",
      "  x_2        x_3_half   0.49999900\n",
      "  x_3        x_4        1.99999608\n"
     ]
    }
   ],
   "source": [
    "relaxed = json.loads(\n",
    "    (REPO_ROOT / \"examples/dual_feg/state/dual_feg_relaxed.json\").read_text()\n",
    ")\n",
    "print(\"dense opt:\", dense[\"opt_value\"])\n",
    "print(\"relaxed opt:\", relaxed[\"opt_value\"])\n",
    "print(\"preserved:\", abs(float(dense[\"opt_value\"]) - float(relaxed[\"opt_value\"])) < 1e-5)\n",
    "print(\"relaxed constraints dropped:\", len(relaxed[\"relaxed_constraints\"]))\n",
    "\n",
    "for group, data in relaxed[\"lambda_groups\"].items():\n",
    "    print(\"\\n\" + group)\n",
    "    for i, ri in enumerate(data[\"row_names\"]):\n",
    "        for j, ci in enumerate(data[\"col_names\"]):\n",
    "            value = float(data[\"matrix\"][i][j])\n",
    "            if abs(value) > 1e-6:\n",
    "                print(f\"  {ri:10s} {ci:10s} {value:.8f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3b5361e8",
   "metadata": {},
   "source": [
    "## Closed-Form Lambda\n",
    "\n",
    "For `N >= 2`, the active lambda coefficients are\n",
    "\n",
    "$$\\lambda^{mon}_{x_{j+1/2},x_N}=\\frac{4}{(N-j)(N-j+1)},\\quad j=1,\\ldots,N-1,$$\n",
    "\n",
    "$$\\lambda^{mon}_{x_N,x_\\star}=\\frac{4}{N},$$\n",
    "\n",
    "$$\\lambda^{Lip}_{x_i,x_{i+1/2}}=\\frac{2}{(N-i)^2},\\quad i=0,\\ldots,N-2,$$\n",
    "\n",
    "$$\\lambda^{Lip}_{x_{N-1},x_N}=2.$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "75f2c45c",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.460607Z",
     "iopub.status.busy": "2026-05-13T16:14:20.460401Z",
     "iopub.status.idle": "2026-05-13T16:14:20.471248Z",
     "shell.execute_reply": "2026-05-13T16:14:20.470203Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccccccc}\n",
       "         & x_1 & x_1_half & x_2 & x_2_half & x_3 & x_3_half & x_4 & x_4_half & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.333 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_2_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.667 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_3_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 2.0 & 0.0 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 1.0 \\\\x_4_half & 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"
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccccccc}\n",
       "         & x_1 & x_1_half & x_2 & x_2_half & x_3 & x_3_half & x_4 & x_4_half & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.125 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.0 & 0.222 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_1_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.5 & 0.0 & 0.0 & 0.0 \\\\x_2_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 2.0 & 0.0 & 0.0 \\\\x_3_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_4_half & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import sympy as sp\n",
    "import numpy as np\n",
    "\n",
    "N_int = 4\n",
    "\n",
    "\n",
    "def monotone_lamb(ri, ci, N=N_int):\n",
    "    if ci == f\"x_{N}\" and ri.endswith(\"_half\"):\n",
    "        j = int(ri.split(\"_\")[1])\n",
    "        if 1 <= j < N:\n",
    "            m = N - j\n",
    "            return sp.Rational(4, m * (m + 1))\n",
    "    if ri == f\"x_{N}\" and ci == \"x_star\":\n",
    "        return sp.Rational(4, N)\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "def lipschitz_lamb(ri, ci, N=N_int):\n",
    "    if ci.endswith(\"_half\") and ri.startswith(\"x_\") and not ri.endswith(\"_half\"):\n",
    "        i = int(ri.split(\"_\")[1])\n",
    "        if ci == f\"x_{i + 1}_half\" and 0 <= i <= N - 2:\n",
    "            return sp.Rational(2, (N - i) ** 2)\n",
    "    if ri == f\"x_{N - 1}\" and ci == f\"x_{N}\":\n",
    "        return sp.S(2)\n",
    "    return sp.S(0)\n",
    "\n",
    "\n",
    "for group, lamb in [\n",
    "    (\"Monotone Operator Inequality\", monotone_lamb),\n",
    "    (\"Lipschitz Operator Inequality\", lipschitz_lamb),\n",
    "]:\n",
    "    data = relaxed[\"lambda_groups\"][group]\n",
    "    candidate = np.array(\n",
    "        [[lamb(ri, ci) for ci in data[\"col_names\"]] for ri in data[\"row_names\"]],\n",
    "        dtype=object,\n",
    "    )\n",
    "    pf.pprint_labeled_matrix(candidate, data[\"row_names\"], data[\"col_names\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2d5b98e7",
   "metadata": {},
   "source": [
    "## Rank-One S Certificate\n",
    "\n",
    "The Gram slack has the direct decomposition\n",
    "\n",
    "$$S=\\left(\\frac{2}{N}(x_0-x_\\star)-A(x_N)\\right)^2.$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "ef9a5522",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.473366Z",
     "iopub.status.busy": "2026-05-13T16:14:20.473159Z",
     "iopub.status.idle": "2026-05-13T16:14:20.482354Z",
     "shell.execute_reply": "2026-05-13T16:14:20.480924Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccccccccc}\n",
       "         & x_0 & x_\\star & A(x_0) & A(x_1_half) & A(x_1) & A(x_2_half) & A(x_2) & A(x_3_half) & A(x_3) & A(x_4_half) & A(x_4) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.25 & -0.25 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.5 \\\\x_\\star & -0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.5 \\\\A(x_0) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_1_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_1) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_2_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_2) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4) & -0.5 & 0.5 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 1.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ctx, pb, obj = get_pep_setup_notebook(N_int, {\"L\": sp.S(1), \"R\": sp.S(1)})\n",
    "pm = pf.ExpressionManager(ctx, resolve_parameters={\"L\": sp.S(1), \"R\": sp.S(1)})\n",
    "S_guess = (\n",
    "    sp.Rational(2, N_int) * (ctx[\"x_0\"] - ctx[\"x_star\"]) - obj(ctx[f\"x_{N_int}\"])\n",
    ") ** 2\n",
    "S_guess_matrix = pm.eval_scalar(S_guess).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(\n",
    "    S_guess_matrix,\n",
    "    [str(v) for v in ctx.basis_vectors()],\n",
    "    [str(v) for v in ctx.basis_vectors()],\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fb2ab88a",
   "metadata": {},
   "source": [
    "## Fixed-N Full Proof Identity\n",
    "\n",
    "The certificate verifies\n",
    "\n",
    "$$\\|A(x_N)\\|^2-\\frac{4}{N^2}\\|x_0-x_\\star\\|^2-\\sum \\lambda^{mon}g^{mon}-\\sum \\lambda^{Lip}g^{Lip}+S=0,$$\n",
    "\n",
    "where each interpolation expression `g` is nonpositive. Therefore `||A(x_N)||^2 <= 4||x_0-x_star||^2/N^2`. Restoring scale gives `||A(x_N)||^2 <= 4L^2R^2/N^2`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "14ea052d",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.484853Z",
     "iopub.status.busy": "2026-05-13T16:14:20.484620Z",
     "iopub.status.idle": "2026-05-13T16:14:20.598864Z",
     "shell.execute_reply": "2026-05-13T16:14:20.597403Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Proof valid: True\n",
      "max residual: 6.938893903907228e-17\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccccccccc}\n",
       "         & x_0 & x_\\star & A(x_0) & A(x_1_half) & A(x_1) & A(x_2_half) & A(x_2) & A(x_3_half) & A(x_3) & A(x_4_half) & A(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 \\\\x_\\star & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_0) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_1_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 \\\\A(x_1) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 \\\\A(x_2_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_2) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_3) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4_half) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4) & 0.0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def lipschitz_ineq(ctx, A, p1, p2):\n",
    "    x1, u1 = ctx.get_duplet_by_point_tag(p1, op=A).expand()\n",
    "    x2, u2 = ctx.get_duplet_by_point_tag(p2, op=A).expand()\n",
    "    return (u1 - u2) ** 2 - L**2 * (x1 - x2) ** 2\n",
    "\n",
    "\n",
    "mon_sum = pf.Scalar.zero()\n",
    "for j in range(1, N_int):\n",
    "    mon_sum += monotone_lamb(f\"x_{j}_half\", f\"x_{N_int}\") * obj.interp_ineq(\n",
    "        f\"x_{j}_half\", f\"x_{N_int}\"\n",
    "    )\n",
    "mon_sum += monotone_lamb(f\"x_{N_int}\", \"x_star\") * obj.interp_ineq(\n",
    "    f\"x_{N_int}\", \"x_star\"\n",
    ")\n",
    "\n",
    "lip_sum = pf.Scalar.zero()\n",
    "for i in range(0, N_int - 1):\n",
    "    lip_sum += lipschitz_lamb(f\"x_{i}\", f\"x_{i + 1}_half\") * lipschitz_ineq(\n",
    "        ctx, obj, f\"x_{i}\", f\"x_{i + 1}_half\"\n",
    "    )\n",
    "lip_sum += lipschitz_lamb(f\"x_{N_int - 1}\", f\"x_{N_int}\") * lipschitz_ineq(\n",
    "    ctx, obj, f\"x_{N_int - 1}\", f\"x_{N_int}\"\n",
    ")\n",
    "\n",
    "tau = sp.Rational(4, N_int**2)\n",
    "perf = obj(ctx[f\"x_{N_int}\"]) ** 2\n",
    "ic = (ctx[\"x_0\"] - ctx[\"x_star\"]) ** 2\n",
    "diff = perf - tau * ic - mon_sum - lip_sum + S_guess\n",
    "residual = pm.eval_scalar(diff).inner_prod_coords\n",
    "print(\"Proof valid:\", np.allclose(residual, 0, atol=1e-10))\n",
    "print(\"max residual:\", float(np.max(np.abs(residual))))\n",
    "pf.pprint_labeled_matrix(\n",
    "    residual,\n",
    "    [str(v) for v in ctx.basis_vectors()],\n",
    "    [str(v) for v in ctx.basis_vectors()],\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f0b4bb5c",
   "metadata": {},
   "source": [
    "## Partial-Sum Lyapunov Construction and Rank Profile\n",
    "\n",
    "Define partial sums `V_k` by accumulating the active proof-certificate terms in algorithmic order. For `k=0,\\ldots,N-2`,\n",
    "\n",
    "$$V_{k+1}-V_k = -\\lambda^{Lip}_{x_k,x_{k+1/2}} g^{Lip}(x_k,x_{k+1/2}) - \\lambda^{mon}_{x_{k+1/2},x_N} g^{mon}(x_{k+1/2},x_N).$$\n",
    "\n",
    "At the final step, add the terminal Lipschitz and monotonicity terms together with the rank-one slack and performance term:\n",
    "\n",
    "$$V_N-V_{N-1} = -2g^{Lip}(x_{N-1},x_N)-\\frac{4}{N}g^{mon}(x_N,x_\\star)+S+\\|A(x_N)\\|^2.$$\n",
    "\n",
    "For `N=4`, this produces constant interior rank `4`, and the final partial sum collapses to the rank-one boundary `V_N = (4/N^2)||x_0-x_\\star||^2`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "4f57788b",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.601417Z",
     "iopub.status.busy": "2026-05-13T16:14:20.601108Z",
     "iopub.status.idle": "2026-05-13T16:14:20.607885Z",
     "shell.execute_reply": "2026-05-13T16:14:20.607057Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rank tolerance: 1e-06\n",
      "boundary identity: V_N = (4/N^2) * ||x_0 - x_star||^2\n",
      "boundary residual max abs: 6.938893903907228e-17\n"
     ]
    }
   ],
   "source": [
    "b3 = json.loads((REPO_ROOT / \"examples/dual_feg/state/dual_feg_b3.json\").read_text())\n",
    "rank_tolerance = b3[\"rank_tolerance\"]\n",
    "print(\"rank tolerance:\", rank_tolerance)\n",
    "print(\"boundary identity:\", b3[\"boundary_identity\"])\n",
    "print(\"boundary residual max abs:\", b3[\"boundary_residual_max_abs\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "8203aab0",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.610034Z",
     "iopub.status.busy": "2026-05-13T16:14:20.609824Z",
     "iopub.status.idle": "2026-05-13T16:14:20.656201Z",
     "shell.execute_reply": "2026-05-13T16:14:20.655337Z"
    }
   },
   "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 lip_expr(p1, p2):\n",
    "    return lipschitz_ineq(ctx, obj, p1, p2)\n",
    "\n",
    "\n",
    "lyap = [pf.Scalar.zero()]\n",
    "partial_sum = pf.Scalar.zero()\n",
    "S_guess = (\n",
    "    sp.Rational(2, N_int) * (ctx[\"x_0\"] - ctx[\"x_star\"]) - obj(ctx[f\"x_{N_int}\"])\n",
    ") ** 2\n",
    "perf = obj(ctx[f\"x_{N_int}\"]) ** 2\n",
    "\n",
    "for step in range(N_int):\n",
    "    if step < N_int - 1:\n",
    "        lip_pair = (f\"x_{step}\", f\"x_{step + 1}_half\")\n",
    "        partial_sum -= lipschitz_lamb(*lip_pair) * lip_expr(*lip_pair)\n",
    "\n",
    "        mon_pair = (f\"x_{step + 1}_half\", f\"x_{N_int}\")\n",
    "        partial_sum -= monotone_lamb(*mon_pair) * obj.interp_ineq(*mon_pair)\n",
    "    else:\n",
    "        lip_pair = (f\"x_{N_int - 1}\", f\"x_{N_int}\")\n",
    "        partial_sum -= lipschitz_lamb(*lip_pair) * lip_expr(*lip_pair)\n",
    "\n",
    "        mon_pair = (f\"x_{N_int}\", \"x_star\")\n",
    "        partial_sum -= monotone_lamb(*mon_pair) * obj.interp_ineq(*mon_pair)\n",
    "\n",
    "        partial_sum += S_guess + perf\n",
    "\n",
    "    lyap.append(partial_sum)\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": 11,
   "id": "6b869783",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.658384Z",
     "iopub.status.busy": "2026-05-13T16:14:20.658141Z",
     "iopub.status.idle": "2026-05-13T16:14:20.666986Z",
     "shell.execute_reply": "2026-05-13T16:14:20.665669Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "lyap[4] rank: 1\n",
      "Coverage check: lyap[N] is rank-one boundary term: True\n",
      "Boundary residual max abs: 6.938893903907228e-17\n"
     ]
    }
   ],
   "source": [
    "M_final = pm.eval_scalar(lyap[N_int]).inner_prod_coords.astype(float)\n",
    "rank_final = int(np.linalg.matrix_rank(M_final, tol=rank_tolerance))\n",
    "tau = sp.Rational(4, N_int**2)\n",
    "boundary = tau * (ctx[\"x_0\"] - ctx[\"x_star\"]) ** 2\n",
    "boundary_residual = pm.eval_scalar(lyap[N_int] - boundary).inner_prod_coords.astype(\n",
    "    float\n",
    ")\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(\"Boundary residual max abs:\", float(np.max(np.abs(boundary_residual))))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "609e6fae",
   "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 for the interior Lyapunov matrices. The useful interior basis has four vectors: a shifted auxiliary vector `q_k=z_k+A(x_N)`, the terminal residual `A(x_N)`, the anchor gap `x_0-x_N`, and the moving gap `x_k-x_N`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "a253820c",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.670590Z",
     "iopub.status.busy": "2026-05-13T16:14:20.670203Z",
     "iopub.status.idle": "2026-05-13T16:14:20.678488Z",
     "shell.execute_reply": "2026-05-13T16:14:20.677326Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "stored rank profile: [0, 4, 4, 4, 1]\n",
      "basis templates: ['q_k = z_k + A(x_N)', 'A(x_N)', 'x_0 - x_N', 'x_k - x_N']\n",
      "coefficient formula verified: True\n"
     ]
    }
   ],
   "source": [
    "b4 = json.loads((REPO_ROOT / \"examples/dual_feg/state/dual_feg_b4.json\").read_text())\n",
    "print(\"stored rank profile:\", b4[\"rank_profile\"])\n",
    "print(\"basis templates:\", b4[\"basis_templates\"])\n",
    "print(\"coefficient formula verified:\", b4[\"coeff_formula_verified\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c087ad07",
   "metadata": {},
   "source": [
    "### Candidate-vector scan\n",
    "\n",
    "The scan includes the selected basis candidates together with primitive nearby vectors such as `z_k`, `x_k`, and `A(x_k)`. The printed list shows which candidates lie in the column space of each interior `V_k`.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "1f9aaf74",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.680872Z",
     "iopub.status.busy": "2026-05-13T16:14:20.680506Z",
     "iopub.status.idle": "2026-05-13T16:14:20.687366Z",
     "shell.execute_reply": "2026-05-13T16:14:20.686571Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: ['q_1=z_1+A(x_N)', 'A(x_N)', 'x_0-x_N', 'x_1-x_N', 'z_1', 'x_1', 'A(x_1)']\n",
      "k=2: ['q_2=z_2+A(x_N)', 'A(x_N)', 'x_0-x_N', 'x_2-x_N', 'z_2', 'x_2', 'A(x_2)']\n",
      "k=3: ['q_3=z_3+A(x_N)', 'A(x_N)', 'x_0-x_N', 'x_3-x_N', 'z_3', 'x_3', 'A(x_3)']\n"
     ]
    }
   ],
   "source": [
    "candidate_labels = {}\n",
    "candidate_vectors = {}\n",
    "for k in range(1, N_int):\n",
    "    pairs = [\n",
    "        (f\"q_{k}=z_{k}+A(x_N)\", ctx[f\"z_{k}\"] + obj(ctx[f\"x_{N_int}\"])),\n",
    "        (\"A(x_N)\", obj(ctx[f\"x_{N_int}\"])),\n",
    "        (\"x_0-x_N\", ctx[\"x_0\"] - ctx[f\"x_{N_int}\"]),\n",
    "        (f\"x_{k}-x_N\", ctx[f\"x_{k}\"] - ctx[f\"x_{N_int}\"]),\n",
    "        (f\"z_{k}\", ctx[f\"z_{k}\"]),\n",
    "        (f\"x_{k}\", ctx[f\"x_{k}\"]),\n",
    "        (f\"A(x_{k})\", obj(ctx[f\"x_{k}\"])),\n",
    "    ]\n",
    "    candidate_labels[k] = [label for label, _ in pairs]\n",
    "    candidate_vectors[k] = [vector for _, vector in pairs]\n",
    "    print(f\"k={k}: {candidate_labels[k]}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "8f43b49a",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.689697Z",
     "iopub.status.busy": "2026-05-13T16:14:20.689469Z",
     "iopub.status.idle": "2026-05-13T16:14:20.765922Z",
     "shell.execute_reply": "2026-05-13T16:14:20.764166Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_1 column-space candidates:\n",
      "   q_1=z_1+A(x_N)\n",
      "   A(x_N)\n",
      "   x_0-x_N\n",
      "   x_1-x_N\n",
      "   z_1\n",
      "V_2 column-space candidates:\n",
      "   q_2=z_2+A(x_N)\n",
      "   A(x_N)\n",
      "   x_0-x_N\n",
      "   x_2-x_N\n",
      "   z_2\n",
      "V_3 column-space candidates:\n",
      "   q_3=z_3+A(x_N)\n",
      "   A(x_N)\n",
      "   x_0-x_N\n",
      "   x_3-x_N\n",
      "   z_3\n",
      "   A(x_3)\n"
     ]
    }
   ],
   "source": [
    "from pepflow.lyapunov_utils import vectors_in_column_space, decompose_rankr_symmetric\n",
    "\n",
    "for k in range(1, N_int):\n",
    "    in_col = vectors_in_column_space(\n",
    "        lyap[k],\n",
    "        candidate_vectors[k],\n",
    "        pep_context=ctx,\n",
    "        resolve_parameters={\"L\": sp.S(1), \"R\": sp.S(1)},\n",
    "        rtol=1e-6,\n",
    "        atol=1e-6,\n",
    "    )\n",
    "    print(f\"V_{k} column-space candidates:\")\n",
    "    for label, vector in zip(candidate_labels[k], candidate_vectors[k]):\n",
    "        if vector in in_col:\n",
    "            print(\"  \", label)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bba39ee5",
   "metadata": {},
   "source": [
    "### Selected basis pattern\n",
    "\n",
    "For each interior `k=1,\\ldots,N-1`, use\n",
    "\n",
    "$$\\mathcal B_k = \\{q_k, A(x_N), x_0-x_N, x_k-x_N\\},\\qquad q_k=z_k+A(x_N).$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "42a36015",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.768915Z",
     "iopub.status.busy": "2026-05-13T16:14:20.768538Z",
     "iopub.status.idle": "2026-05-13T16:14:20.776527Z",
     "shell.execute_reply": "2026-05-13T16:14:20.775180Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1: rank 4 basis ['q_1=z_1+A(x_N)', 'A(x_N)', 'x_0-x_N', 'x_1-x_N']\n",
      "k=2: rank 4 basis ['q_2=z_2+A(x_N)', 'A(x_N)', 'x_0-x_N', 'x_2-x_N']\n",
      "k=3: rank 4 basis ['q_3=z_3+A(x_N)', 'A(x_N)', 'x_0-x_N', 'x_3-x_N']\n"
     ]
    }
   ],
   "source": [
    "def V_k_basis(k):\n",
    "    return [\n",
    "        ctx[f\"z_{k}\"] + obj(ctx[f\"x_{N_int}\"]),\n",
    "        obj(ctx[f\"x_{N_int}\"]),\n",
    "        ctx[\"x_0\"] - ctx[f\"x_{N_int}\"],\n",
    "        ctx[f\"x_{k}\"] - ctx[f\"x_{N_int}\"],\n",
    "    ]\n",
    "\n",
    "\n",
    "def V_k_basis_labels(k):\n",
    "    return [f\"q_{k}=z_{k}+A(x_N)\", \"A(x_N)\", \"x_0-x_N\", f\"x_{k}-x_N\"]\n",
    "\n",
    "\n",
    "for k in range(1, N_int):\n",
    "    basis = V_k_basis(k)\n",
    "    W = np.stack([pm.eval_vector(v).coords.astype(float) for v in basis], axis=1)\n",
    "    rank = int(np.linalg.matrix_rank(W, tol=1e-7))\n",
    "    print(f\"k={k}: rank {rank} basis {V_k_basis_labels(k)}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f7a52710",
   "metadata": {},
   "source": [
    "### Coefficient matrices\n",
    "\n",
    "In the basis order `[q_k, A(x_N), x_0-x_N, x_k-x_N]`, the coefficient matrix has the pattern\n",
    "\n",
    "$$C_{00}=2,\\quad C_{03}=C_{30}=-\\frac{2}{N-k},\\quad C_{11}=-2,\\quad C_{12}=C_{21}=\\frac{2}{N},$$\n",
    "\n",
    "with all other entries equal to zero.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "63400e62",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.778914Z",
     "iopub.status.busy": "2026-05-13T16:14:20.778694Z",
     "iopub.status.idle": "2026-05-13T16:14:20.858507Z",
     "shell.execute_reply": "2026-05-13T16:14:20.857744Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "k=1 formula residual: 3.774758283725532e-15\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & q_1=z_1+A(x_N) & A(x_N) & x_0-x_N & x_1-x_N \\\\\n",
       "        \\hline\n",
       "        q_1=z_1+A(x_N) & 2.0 & 0.0 & 0.0 & -0.667 \\\\A(x_N) & 0.0 & -2.0 & 0.5 & 0.0 \\\\x_0-x_N & 0.0 & 0.5 & 0.0 & 0.0 \\\\x_1-x_N & -0.667 & 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": [
      "k=2 formula residual: 1.1546319456101628e-14\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & q_2=z_2+A(x_N) & A(x_N) & x_0-x_N & x_2-x_N \\\\\n",
       "        \\hline\n",
       "        q_2=z_2+A(x_N) & 2.0 & 0.0 & 0.0 & -1.0 \\\\A(x_N) & 0.0 & -2.0 & 0.5 & 0.0 \\\\x_0-x_N & 0.0 & 0.5 & 0.0 & 0.0 \\\\x_2-x_N & -1.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": [
      "k=3 formula residual: 1.9984014443252818e-15\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & q_3=z_3+A(x_N) & A(x_N) & x_0-x_N & x_3-x_N \\\\\n",
       "        \\hline\n",
       "        q_3=z_3+A(x_N) & 2.0 & 0.0 & 0.0 & -2.0 \\\\A(x_N) & 0.0 & -2.0 & 0.5 & 0.0 \\\\x_0-x_N & 0.0 & 0.5 & 0.0 & 0.0 \\\\x_3-x_N & -2.0 & 0.0 & 0.0 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def coeff_pattern(k, N=N_int):\n",
    "    C = sp.zeros(4, 4)\n",
    "    C[0, 0] = sp.S(2)\n",
    "    C[0, 3] = C[3, 0] = -sp.Rational(2, N - k)\n",
    "    C[1, 1] = -sp.S(2)\n",
    "    C[1, 2] = C[2, 1] = sp.Rational(2, N)\n",
    "    return C\n",
    "\n",
    "\n",
    "for k in range(1, N_int):\n",
    "    labels = V_k_basis_labels(k)\n",
    "    C = decompose_rankr_symmetric(\n",
    "        lyap[k],\n",
    "        V_k_basis(k),\n",
    "        pep_context=ctx,\n",
    "        resolve_parameters={\"L\": sp.S(1), \"R\": sp.S(1)},\n",
    "    )\n",
    "    C_formula = np.array(coeff_pattern(k, N_int), dtype=float)\n",
    "    print(f\"k={k} formula residual:\", float(np.max(np.abs(C - C_formula))))\n",
    "    pf.pprint_labeled_matrix(C_formula, labels, labels, precision=3)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7bf70c16",
   "metadata": {},
   "source": [
    "### Block 4 conclusion\n",
    "\n",
    "The current closed-form interior Lyapunov candidate is `V_k = B_k^T C_k B_k` with `B_k=[q_k, A(x_N), x_0-x_N, x_k-x_N]` and the sparse coefficient pattern above. Block 5 will symbolically verify the step, base, and boundary identities for this formula.\n",
    "\n",
    "## Symbolic Step, Base, and Boundary Identities\n",
    "\n",
    "To be completed in later blocks.\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eed6271b",
   "metadata": {},
   "source": [
    "## Symbolic Step Recursion Verification\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5f214775",
   "metadata": {},
   "source": [
    "For $0\\le k\\le N-2$, verify\n",
    "\n",
    "$$V_{k+1}-V_{k}=-\\frac{2}{(N-k)^{2}}G(x_{k},x_{k+1/2})-\\frac{4}{(N-k-1)(N-k)}M(x_{k+1/2},x_{N}).$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "ff06807f",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:20.861133Z",
     "iopub.status.busy": "2026-05-13T16:14:20.860919Z",
     "iopub.status.idle": "2026-05-13T16:14:21.177377Z",
     "shell.execute_reply": "2026-05-13T16:14:21.176750Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Step identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & q_k & A(x_N) & x_0-x_N & x_k-x_N & A(x_k) & A(x_{k+1/2}) \\\\\n",
       "        \\hline\n",
       "        q_k & 0 & 0 & 0 & 0 & 0 & 0 \\\\A(x_N) & 0 & 0 & 0 & 0 & 0 & 0 \\\\x_0-x_N & 0 & 0 & 0 & 0 & 0 & 0 \\\\x_k-x_N & 0 & 0 & 0 & 0 & 0 & 0 \\\\A(x_k) & 0 & 0 & 0 & 0 & 0 & 0 \\\\A(x_{k+1/2}) & 0 & 0 & 0 & 0 & 0 & 0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def simplify_symbolic_matrix(matrix):\n",
    "    return sp.Matrix(matrix).applyfunc(lambda entry: sp.factor(sp.nsimplify(entry)))\n",
    "\n",
    "\n",
    "ctx_step = pf.PEPContext(\"dual_feg_symbolic_step\").set_as_current()\n",
    "N_par = pf.Parameter(\"N\")\n",
    "k_par = pf.Parameter(\"k\")\n",
    "N_sym = sp.Symbol(\"N\", positive=True, integer=True)\n",
    "k_sym = sp.Symbol(\"k\", positive=True, integer=True)\n",
    "\n",
    "q_k = pf.Vector(is_basis=True, tags=[\"q_k\"])\n",
    "a_N = pf.Vector(is_basis=True, tags=[\"A(x_N)\"])\n",
    "c_0N = pf.Vector(is_basis=True, tags=[\"x_0-x_N\"])\n",
    "d_k = pf.Vector(is_basis=True, tags=[\"x_k-x_N\"])\n",
    "a_k = pf.Vector(is_basis=True, tags=[\"A(x_k)\"])\n",
    "a_half = pf.Vector(is_basis=True, tags=[\"A(x_{k+1/2})\"])\n",
    "\n",
    "rho = (N_par - k_par - 1) / (N_par - k_par)\n",
    "d_half = d_k - q_k + a_N - a_k\n",
    "q_next = rho * (q_k - a_N) - (1 / (N_par - k_par)) * a_half + a_N\n",
    "d_next = d_half - rho * (a_half - a_k)\n",
    "\n",
    "\n",
    "def V_symbolic(q_vec, d_vec, kk):\n",
    "    return (\n",
    "        2 * (q_vec**2)\n",
    "        - (4 / (N_par - kk)) * (q_vec * d_vec)\n",
    "        - 2 * (a_N**2)\n",
    "        + (4 / N_par) * (a_N * c_0N)\n",
    "    )\n",
    "\n",
    "\n",
    "lip_residual = (a_half - a_k) ** 2 - ((d_half - d_k) ** 2)\n",
    "mon_residual = -(d_half * (a_half - a_N))\n",
    "LHS = V_symbolic(q_next, d_next, k_par + 1) - V_symbolic(q_k, d_k, k_par)\n",
    "RHS = (\n",
    "    -(2 / (N_par - k_par) ** 2) * lip_residual\n",
    "    - (4 / ((N_par - k_par - 1) * (N_par - k_par))) * mon_residual\n",
    ")\n",
    "diff = LHS - RHS\n",
    "pm_step = pf.ExpressionManager(ctx_step, resolve_parameters={\"N\": N_sym, \"k\": k_sym})\n",
    "step_residual = simplify_symbolic_matrix(pm_step.eval_scalar(diff).inner_prod_coords)\n",
    "print(\"Step identity zero:\", step_residual == sp.zeros(*step_residual.shape))\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(step_residual.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_step.basis_vectors()],\n",
    "    [str(v) for v in ctx_step.basis_vectors()],\n",
    "    precision=None,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8e7c71d5",
   "metadata": {},
   "source": [
    "## Base Case Symbolic Verification\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "95e35113",
   "metadata": {},
   "source": [
    "For $k=0$, verify\n",
    "\n",
    "$$V_{1}-V_{0}=-\\frac{2}{N^{2}}G(x_{0},x_{1/2})-\\frac{4}{(N-1)N}M(x_{1/2},x_{N}).$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "94cfe5ac",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:21.180195Z",
     "iopub.status.busy": "2026-05-13T16:14:21.179979Z",
     "iopub.status.idle": "2026-05-13T16:14:21.258408Z",
     "shell.execute_reply": "2026-05-13T16:14:21.257670Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Base identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & A(x_N) & x_0-x_N & x_N-x_\\star & A(x_0) & A(x_{1/2}) \\\\\n",
       "        \\hline\n",
       "        A(x_N) & 0 & 0 & 0 & 0 & 0 \\\\x_0-x_N & 0 & 0 & 0 & 0 & 0 \\\\x_N-x_\\star & 0 & 0 & 0 & 0 & 0 \\\\A(x_0) & 0 & 0 & 0 & 0 & 0 \\\\A(x_{1/2}) & 0 & 0 & 0 & 0 & 0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ctx_base = pf.PEPContext(\"dual_feg_symbolic_base\").set_as_current()\n",
    "N_par = pf.Parameter(\"N\")\n",
    "N_sym = sp.Symbol(\"N\", positive=True, integer=True)\n",
    "\n",
    "a_N = pf.Vector(is_basis=True, tags=[\"A(x_N)\"])\n",
    "c_0N = pf.Vector(is_basis=True, tags=[\"x_0-x_N\"])\n",
    "q_0 = a_N\n",
    "d_0 = c_0N\n",
    "x_star_gap = pf.Vector(is_basis=True, tags=[\"x_N-x_star\"])\n",
    "a_0 = pf.Vector(is_basis=True, tags=[\"A(x_0)\"])\n",
    "a_half = pf.Vector(is_basis=True, tags=[\"A(x_{1/2})\"])\n",
    "\n",
    "rho = (N_par - 1) / N_par\n",
    "d_half = d_0 - q_0 + a_N - a_0\n",
    "q_1 = rho * (q_0 - a_N) - (1 / N_par) * a_half + a_N\n",
    "d_1 = d_half - rho * (a_half - a_0)\n",
    "\n",
    "\n",
    "def V_base(q_vec, d_vec, kk):\n",
    "    return (\n",
    "        2 * (q_vec**2)\n",
    "        - (4 / (N_par - kk)) * (q_vec * d_vec)\n",
    "        - 2 * (a_N**2)\n",
    "        + (4 / N_par) * (a_N * c_0N)\n",
    "    )\n",
    "\n",
    "\n",
    "lip_residual = (a_half - a_0) ** 2 - ((d_half - d_0) ** 2)\n",
    "mon_residual = -(d_half * (a_half - a_N))\n",
    "LHS = V_base(q_1, d_1, 1) - pf.Scalar.zero()\n",
    "RHS = -(2 / N_par**2) * lip_residual - (4 / ((N_par - 1) * N_par)) * mon_residual\n",
    "diff = LHS - RHS\n",
    "pm_base = pf.ExpressionManager(ctx_base, resolve_parameters={\"N\": N_sym})\n",
    "base_residual = simplify_symbolic_matrix(pm_base.eval_scalar(diff).inner_prod_coords)\n",
    "print(\"Base identity zero:\", base_residual == sp.zeros(*base_residual.shape))\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(base_residual.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_base.basis_vectors()],\n",
    "    [str(v) for v in ctx_base.basis_vectors()],\n",
    "    precision=None,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "385be289",
   "metadata": {},
   "source": [
    "### Boundary Identity Symbolic Verification\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6fa6d9e8",
   "metadata": {},
   "source": [
    "Verify\n",
    "\n",
    "$$V_{N}-V_{N-1}=-2G(x_{N-1},x_{N})-\\frac{4}{N}M(x_{N},x_{\\star})+\\left\\|\\frac{2}{N}(x_{0}-x_{\\star})-A(x_{N})\\right\\|^{2}+\\|A(x_{N})\\|^{2}.$$\n",
    "\n",
    "The residual $\\mathrm{LHS}-\\mathrm{RHS}$ should simplify to zero.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "7d44b50a",
   "metadata": {
    "execution": {
     "iopub.execute_input": "2026-05-13T16:14:21.261419Z",
     "iopub.status.busy": "2026-05-13T16:14:21.261148Z",
     "iopub.status.idle": "2026-05-13T16:14:21.275366Z",
     "shell.execute_reply": "2026-05-13T16:14:21.274456Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Boundary identity zero: True\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & q_{N-1} & A(x_N) & x_0-x_N & x_{N-1}-x_N & x_0-x_\\star \\\\\n",
       "        \\hline\n",
       "        q_{N-1} & 0 & 0 & 0 & 0 & 0 \\\\A(x_N) & 0 & 0 & 0 & 0 & 0 \\\\x_0-x_N & 0 & 0 & 0 & 0 & 0 \\\\x_{N-1}-x_N & 0 & 0 & 0 & 0 & 0 \\\\x_0-x_\\star & 0 & 0 & 0 & 0 & 0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ctx_boundary = pf.PEPContext(\"dual_feg_symbolic_boundary\").set_as_current()\n",
    "N_par = pf.Parameter(\"N\")\n",
    "N_sym = sp.Symbol(\"N\", positive=True, integer=True)\n",
    "\n",
    "q_prev = pf.Vector(is_basis=True, tags=[\"q_{N-1}\"])\n",
    "a_N = pf.Vector(is_basis=True, tags=[\"A(x_N)\"])\n",
    "c_0N = pf.Vector(is_basis=True, tags=[\"x_0-x_N\"])\n",
    "d_prev = pf.Vector(is_basis=True, tags=[\"x_{N-1}-x_N\"])\n",
    "c_0star = pf.Vector(is_basis=True, tags=[\"x_0-x_star\"])\n",
    "\n",
    "V_prev = (\n",
    "    2 * (q_prev**2) - 4 * (q_prev * d_prev) - 2 * (a_N**2) + (4 / N_par) * (a_N * c_0N)\n",
    ")\n",
    "V_N = (4 / N_par**2) * (c_0star**2)\n",
    "lip_residual = (q_prev - d_prev) ** 2 - (d_prev**2)\n",
    "mon_residual = -((c_0star - c_0N) * a_N)\n",
    "S_terminal = ((2 / N_par) * c_0star - a_N) ** 2\n",
    "perf = a_N**2\n",
    "LHS = V_N - V_prev\n",
    "RHS = -2 * lip_residual - (4 / N_par) * mon_residual + S_terminal + perf\n",
    "diff = LHS - RHS\n",
    "pm_boundary = pf.ExpressionManager(ctx_boundary, resolve_parameters={\"N\": N_sym})\n",
    "boundary_residual = simplify_symbolic_matrix(\n",
    "    pm_boundary.eval_scalar(diff).inner_prod_coords\n",
    ")\n",
    "print(\n",
    "    \"Boundary identity zero:\", boundary_residual == sp.zeros(*boundary_residual.shape)\n",
    ")\n",
    "pf.pprint_labeled_matrix(\n",
    "    np.array(boundary_residual.tolist(), dtype=object),\n",
    "    [str(v) for v in ctx_boundary.basis_vectors()],\n",
    "    [str(v) for v in ctx_boundary.basis_vectors()],\n",
    "    precision=None,\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
