{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "1c269791",
   "metadata": {},
   "source": [
    "# Accelerated Proximal Point Method (APPM) Example\n",
    "\n",
    "This code tests the Accelerated Proximal Point Method which is the exact optimal method that \n",
    "reduces the fixed-point residual $||x - J_{\\alpha A}x||$ with respect to the initial distance to the solution \n",
    "for a maximal monotone operator A. It was introduced in \"Accelerated proximal point method for \n",
    "maximally monotone operators\" by Donghwan Kim (2021). This method is equivalent to what \n",
    "later came to be known as the Optimized Halpern Method, which was studied in \"On the \n",
    "Convergence Rate of the Halpern Iteration\" by Felix Lieder (2021). The details of the \n",
    "equivalence can be found in Exercise 12.10 of Large-Scale Convex Optimization: \n",
    "Algorithms & Analyses via Monotone Operators by Ernest K. Ryu and Wotao Yin (2022)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e624b576",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "5a10dfb8",
   "metadata": {},
   "source": [
    "## Import the required libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd77a040",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pepflow as pf\n",
    "import numpy as np\n",
    "import sympy as sp\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython.display import display\n",
    "from itertools import combinations"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d0a0719f",
   "metadata": {},
   "source": [
    "## Define the operators"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c0b7f1a0",
   "metadata": {},
   "outputs": [],
   "source": [
    "A = pf.MonotoneOperator(is_basis=True, tags=[\"A\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ac3c22c6",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with APPM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "245f666e",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_appm(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_appm = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    y = x.add_tag(\"y_0\")\n",
    "    A.set_zero_point(\"x_star\")\n",
    "    for i in range(N):\n",
    "        x = A.resolvent(y, stepsize, tag=f\"x_{i + 1}\")\n",
    "        y = (\n",
    "            sp.S(i + 1) / sp.S(i + 2) * (sp.S(2) * x - y)\n",
    "            + sp.S(1) / sp.S(i + 2) * ctx_appm[\"x_0\"]\n",
    "        ).add_tag(f\"y_{i + 1}\")\n",
    "\n",
    "    return ctx_appm"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fa6b4a78",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of APPM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "252e5596",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x1292bc590>"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiMAAAGdCAYAAADAAnMpAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjMsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvZiW1igAAAAlwSFlzAAAPYQAAD2EBqD+naQAARmpJREFUeJzt3Xl8TGf///HXJCRBFkJESIi9lMauqSoqrZ9bfSltFS2lG9WW5u5CW3S1tVV6Vxeq6H3fli663KWWuqu01dqVUmssRYLSRIKEmfP749yZGJLIRGZOknk/H4/zODNnzjnzmXlg3q5zXdexGYZhICIiImIRP6sLEBEREd+mMCIiIiKWUhgRERERSymMiIiIiKUURkRERMRSCiMiIiJiKYURERERsZTCiIiIiFiqjNUFFITD4eDIkSOEhIRgs9msLkdEREQKwDAMTp8+TfXq1fHzy7v9o0SEkSNHjhATE2N1GSIiIlIIhw4dIjo6Os/XS0QYCQkJAcwPExoaanE1IiIiUhBpaWnExMQ4f8fzUiLCSPalmdDQUIURERGREuZKXSzUgVVEREQspTAiIiIillIYEREREUuViD4jIiLFld1u5/z581aXIWIJf39/ypQpc9XTbiiMiIgUUnp6On/88QeGYVhdiohlypcvT1RUFAEBAYU+h8KIiEgh2O12/vjjD8qXL09ERIQmZBSfYxgGWVlZHD9+nKSkJOrXr5/vxGb5URgRESmE8+fPYxgGERERlCtXzupyRCxRrlw5ypYty4EDB8jKyiIoKKhQ51EHVhGRq6AWEfF1hW0NuZjPtozY7bB6NRw9ClFR0L49+PtbXZWIiIjvcTvOrFq1iu7du1O9enVsNhtffPHFFY9ZuXIlLVq0IDAwkHr16jF79uxClFp0Fi6E2Fjo1An69TPXsbHmdhEREfEut8NIRkYGcXFxTJs2rUD7JyUl0a1bNzp16sTmzZsZMWIEDzzwAEuXLnW72KKwcCHccQf88Yfr9sOHze0KJCIiIt7l9mWarl270rVr1wLv/95771G7dm3eeOMNABo1asQPP/zAm2++SZcuXdx9+6tit8Pw4ZDbKDzDAJsNRoyAHj10yUZEREqn22+/nZUrV9K5c2c+/fRTq8sBvNCBdc2aNSQkJLhs69KlC2vWrMnzmMzMTNLS0lyWorB6tWuLyKP8g3cZQkN+B8xAcuiQuZ+IiEhpNHz4cD766COry3Dh8TCSnJxMZGSky7bIyEjS0tI4e/ZsrseMHz+esLAw5xITE1MktRw96vq8H3MZwvs0Yke++4mIiJQWHTt2JCQkxOoyXBTLob2jRo0iNTXVuRw6dKhIzhsV5fr8KOaGKI7mu5+IiLivY8eOjBgxoticp6DnK+r3K2rFvb7C8HgYqVatGikpKS7bUlJSCA0NzXOioMDAQEJDQ12WotC+PURHm31DAI5QHYDqHAHM7TEx5n4iIqXdmjVr8Pf3p1u3blaXAuT9I7tw4UJefvll7xckXuPxMBIfH8+KFStcti1fvpz4+HhPv/Vl/P1h6lTzsc2WE0aiOOoMKFOmqPOqiPiGmTNn8thjj7Fq1SqOHDlidTl5Cg8PL3aXFaRouR1G0tPT2bx5M5s3bwbMobubN2/m4MGDgHmJZcCAAc79hwwZwr59+3j66af5/fffeeedd/j444954okniuYTuKlXL/j0U6hRI+cyTXWOEB1tbu/Vy5KyRES8Kj09nQULFjB06FC6det22fxPHTt25PHHH+fpp58mPDycatWq8cILL7jss2TJEm688UYqVqxI5cqVue2229i7d2+u7/fRRx9RuXJlMjMzXbb37NmTe++9l/vuu4/vv/+eqVOnYrPZsNls7N+/31nLxS0mDoeDSZMmUa9ePQIDA6lZsyavvvqq2zXl58KFCzz66KOEhYVRpUoVRo8e7XJDxMzMTB5//HGqVq1KUFAQN954I+vWrXO+Hhsby5QpU1zO2axZM5fvsCDfcUZGBgMGDCA4OJioqCjnyNTSxu0wsn79epo3b07z5s0BSExMpHnz5owZMwaAo0ePOoMJQO3atVm0aBHLly8nLi6ON954gw8++MDrw3ov1qsX7N8PIyaaLSPt6xwhKUlBRER8x8cff8w111xDw4YNueeee/jwww8vu/vwnDlzqFChAr/88guTJk3ipZdeYvny5c7XMzIySExMZP369axYsQI/Pz9uv/12HA7HZe935513Yrfb+eqrr5zbjh07xqJFixg8eDBTp04lPj6eBx98kKNHj3L06NE8By+MGjWKCRMmMHr0aLZv387cuXOdAyXcqSk/c+bMoUyZMqxdu5apU6cyefJkPvjgA+frTz/9NJ999hlz5sxh48aN1KtXjy5dunDy5Em33ye/7/ipp57i+++/58svv2TZsmWsXLmSjRs3uvUel0pISODOO+9k8eLFREdH5zu61WuMEiA1NdUAjNTU1KI98a+/GgYYRkRE0Z5XREq9s2fPGtu3bzfOnj1rbnA4DCM93ZrF4XC7/htuuMGYMmWKYRiGcf78eaNKlSrGd99953y9Q4cOxo033uhyTOvWrY1nnnkmz3MeP37cAIytW7c6zzF8+HDn60OHDjW6du3qfP7GG28YderUMRz/q//S/S+uJXt7WlqaERgYaMyYMaNAn/PSmvJ7n4tfb9SokbMuwzCMZ555xmjUqJFhGIaRnp5ulC1b1vj3v//tfD0rK8uoXr26MWnSJMMwDKNWrVrGm2++6XLeuLg4Y+zYsS7vk993fPr0aSMgIMD4+OOPna//+eefRrly5fKt39su+7twkYL+fhfL0TRekz1s5vhxyMqythYRKdnOnIHgYGuWM2fcKnXnzp2sXbuWvn37AlCmTBn69OnDzJkzXfa77rrrXJ5HRUVx7Ngx5/Pdu3fTt29f6tSpQ2hoKLGxsQAureMXe/DBB1m2bBmHDx8GYPbs2dx3331u3Wxwx44dZGZm0rlz51xfd7emvFx//fUudcXHx7N7927sdjt79+7l/PnztGvXzvl62bJladOmDTt27MjtdHnK7zveu3cvWVlZtG3b1vl6eHg4DRs2dOs9SgKfvVEeAJUrQ9mycP48JCdDzZpWVyQi4nEzZ87kwoULVK9e3bnNMAwCAwN5++23CQsLA8wf2IvZbDaXyx3du3enVq1azJgxg+rVq+NwOGjSpAlZefznrnnz5sTFxfHRRx9x66238ttvv7Fo0SK3as9rFGZha/IUPz+/yy57nT9//rL9rvQdu8sbd5G+9HMVBd9uGbHZIPsvo2Y6E5GrUb48pKdbs5QvX+AyL1y4wEcffcQbb7zhHIywefNmtmzZQvXq1Zk3b16BzvPnn3+yc+dOnn/+eTp37kyjRo04derUFY974IEHmD17NrNmzSIhIcGlX0hAQAB2uz3f4+vXr0+5cuUuG6V5NTXl5pdffnF5/vPPP1O/fn38/f2pW7cuAQEB/Pjjj87Xz58/z7p162jcuDEAERERHL3odyUtLY2kpCS3aqhbty5ly5Z1qeXUqVPs2rUrz2MMw/D44gm+3TIC5qWaAwegGA9rE5ESwGaDChWsruKKvv76a06dOsX999/vbAHJ1rt3b2bOnMmQIUOueJ5KlSpRuXJlpk+fTlRUFAcPHmTkyJFXPK5fv348+eSTzJgx47IpyWNjY/nll1/Yv38/wcHBhIeH4+fn+n/moKAgnnnmGZ5++mkCAgJo164dx48f57fffmPQoEGFqik3Bw8eJDExkYcffpiNGzfyj3/8wzmSpUKFCgwdOpSnnnqK8PBwatasyaRJkzhz5gz3338/ADfffDOzZ8+me/fuVKxYkTFjxuDv5rwRwcHB3H///Tz11FNUrlyZqlWr8txzz132nRTUJ598wvLlyzl16hRjx46lSZMmRbJvUVAYyW4ZURgRER8wc+ZMEhISLgsiYIaRSZMm8euvv17xPH5+fsyfP5/HH3+cJk2a0LBhQ9566y06duyY73FhYWH07t2bRYsW0bNnT5fXnnzySQYOHEjjxo05e/YsSUlJzj4fFxs9ejRlypRhzJgxHDlyhKioKIYMGVLomnIzYMAAzp49S5s2bfD392f48OE89NBDztcnTJiAw+Hg3nvv5fTp07Rq1YqlS5dSqVIlwBzxk5SUxG233UZYWBgvv/yy2y0jAK+99hrp6el0796dkJAQ/v73v5Oamur2ecAc0XTnnXeyefNmFi9e7AwYX375Jd99953LUOS89vUUm+GpNpcilJaWRlhYGKmpqUU2G6vTo4/CtGnw3HPwyitFe24RKbXOnTtHUlIStWvXJigoyOpySpTOnTtz7bXX8tZbb1ldis9xOBwMGTKEMWPGEB0dDcDLL79MdHQ0gwYNuuK+ucnv70JBf799u88IqGVERMRLTp06xeeff87KlSsZNmyY1eWUWps2baJjx47Uq1ePDz/8kN69e5OSkoLD4eCJJ55g6NChLuHi119/JS4ujrS0NHr06MH06dPz3NdTdJlGYURExCuaN2/OqVOnmDhxYqkcnlocZGRk0LdvXxYvXkz58uVp06YNPXr0IDIykkmTJrFu3ToyMzO55ZZb6N27N2AOly5Xrhxdu3blhRde4JZbbslzX0/RZZqlS+H//T+47jrYsqVozy0ipZYu00hxtHDhQr755htmzJjhHK69Z88eauYxdcXZs2epXr06sbGx/Otf/+Laa691+z11maYoqGVERERKiexLLgBbt27l+uuvzzOIAGzbto34+HgcDgdlylh3sURhJDuMnDihWVhFRKRECwgI4MiRIzgcDkaP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FixYQNWqVQkNDS30ea677jr++OMPdu3alWvryI8//kiPHj245557ALN/x65du2jcuLFb79O/f3+efvpp+vbty759+7j77rtd3uOGG27gkUcecW67UitFREQERy9KmWlpaSQlJTmft2jRguTkZMqUKePSWfhSDRo0oEGDBjzxxBP07duXWbNmlegwoss0F1MnVhHxgvbtzSvDeU3rkT0pdPv2nq+ladOm9O/fn7feestle8eOHTl+/DiTJk1i7969TJs2jW+++eaq369///5UqVKFHj16sHr1apKSkli5ciWPP/64W51pO3TowE033UTv3r1Zvnw5SUlJfPPNNyxZsgSA+vXrs3z5cn766Sd27NjBww8/TEpKitv19urVi9OnTzN06FA6depE9ewbq/7vPdavX8/SpUvZtWsXo0ePZt26dfme7+abb+af//wnq1evZuvWrQwcONDZ6RbMkU7x8fH07NmTZcuWsX//fn766Seee+451q9fz9mzZ3n00UdZuXIlBw4c4Mcff2TdunU0KuH/gVYYuZjCiIh4gb8/ZI8AvTSQZD+fMsXczxteeukl52WUbI0aNeKdd95h2rRpxMXFsXbtWp588smrfq/y5cuzatUqatasSa9evWjUqBH3338/586dc7ul5LPPPqN169b07duXxo0b8/TTT2P/30Qtzz//PC1atKBLly507NiRatWqFWom1JCQELp3786WLVvo37+/y2sPP/wwvXr1ok+fPrRt25Y///zTpZUkN6NGjaJDhw7cdtttdOvWjZ49ezqHCoN5SWvx4sXcdNNNDBo0iAYNGnD33Xdz4MABIiMj8ff3588//2TAgAE0aNCAu+66i65du/Liiy+6/dmKE5tx6cWrYigtLY2wsDBSU1Ovqlnvil55BUaPhsGDYeZMz72PiJR4586dIykpidq1axMUFFSocyxcCMOHu3ZmjYkxg0ivXkVTp4in5fd3oaC/3+ozcrHsETWa+ExEvKBXL+jRwxw1c/So2UekfXvvtYiIFBcKIxfTZRoR8TJ/f+jY0eoqRKylPiMXuziMFP+rVyIiIqWCwsjFsi/TpKebQ3xFRETE4xRGLla+PFSubD5WvxERKYASMAZAxKOK4u+AwsilsltH1G9ERPKRPTeEOzN6ipRGZ86cAS6fjdcd6sB6qZgY2LJFYURE8lWmTBnKly/P8ePH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pe6pJSKMnD59GkDTwouIiJRAp0+fJiwsLM/XS8RoGofDwZEjRwgJCcFmsxXZedPS0oiJieHQoUM+O0rH178DX//8oO/A1z8/6Dvw9c8PnvsODMPg9OnTVK9e3WXaj0uViJYRPz8/oqOjPXb+0NBQn/0DmM3XvwNf//yg78DXPz/oO/D1zw+e+Q7yaxHJpg6sIiIiYimFEREREbGUT4eRwMBAxo4d69N3CPb178DXPz/oO/D1zw/6Dnz984P130GJ6MAqIiIipZdPt4yIiIiI9RRGRERExFIKIyIiImIphRERERGxlE+GkVWrVtG9e3eqV6+OzWbjiy++sLokrxo/fjytW7cmJCSEqlWr0rNnT3bu3Gl1WV717rvvct111zkn+ImPj+ebb76xuizLTJgwAZvNxogRI6wuxWteeOEFbDaby3LNNddYXZZXHT58mHvuuYfKlStTrlw5mjZtyvr1660uy2tiY2Mv+zNgs9kYNmyY1aV5hd1uZ/To0dSuXZty5cpRt25dXn755SveR8YTSsQMrEUtIyODuLg4Bg8eTK9evawux+u+//57hg0bRuvWrblw4QLPPvsst956K9u3b6dChQpWl+cV0dHRTJgwgfr162MYBnPmzKFHjx5s2rSJa6+91uryvGrdunW8//77XHfddVaX4nXXXnst3377rfN5mTK+80/iqVOnaNeuHZ06deKbb74hIiKC3bt3U6lSJatL85p169Zht9udz7dt28Ytt9zCnXfeaWFV3jNx4kTeffdd5syZw7XXXsv69esZNGgQYWFhPP74416txXf+5l2ka9eudO3a1eoyLLNkyRKX57Nnz6Zq1aps2LCBm266yaKqvKt79+4uz1999VXeffddfv75Z58KI+np6fTv358ZM2bwyiuvWF2O15UpU4Zq1apZXYYlJk6cSExMDLNmzXJuq127toUVeV9ERITL8wkTJlC3bl06dOhgUUXe9dNPP9GjRw+6desGmC1F8+bNY+3atV6vxScv04ir1NRUAMLDwy2uxBp2u5358+eTkZFBfHy81eV41bBhw+jWrRsJCQlWl2KJ3bt3U716derUqUP//v05ePCg1SV5zVdffUWrVq248847qVq1Ks2bN2fGjBlWl2WZrKws/vWvfzF48OAivSFrcXbDDTewYsUKdu3aBcCWLVv44YcfLPnPuk+2jEgOh8PBiBEjaNeuHU2aNLG6HK/aunUr8fHxnDt3juDgYD7//HMaN25sdVleM3/+fDZu3Mi6deusLsUSbdu2Zfbs2TRs2JCjR4/y4osv0r59e7Zt20ZISIjV5Xncvn37ePfdd0lMTOTZZ59l3bp1PP744wQEBDBw4ECry/O6L774gr/++ov77rvP6lK8ZuTIkaSlpXHNNdfg7++P3W7n1VdfpX///t4vxvBxgPH5559bXYZlhgwZYtSqVcs4dOiQ1aV4XWZmprF7925j/fr1xsiRI40qVaoYv/32m9VlecXBgweNqlWrGlu2bHFu69ChgzF8+HDrirLYqVOnjNDQUOODDz6wuhSvKFu2rBEfH++y7bHHHjOuv/56iyqy1q233mrcdtttVpfhVfPmzTOio6ONefPmGb/++qvx0UcfGeHh4cbs2bO9XotaRnzYo48+ytdff82qVauIjo62uhyvCwgIoF69egC0bNmSdevWMXXqVN5//32LK/O8DRs2cOzYMVq0aOHcZrfbWbVqFW+//TaZmZn4+/tbWKH3VaxYkQYNGrBnzx6rS/GKqKioy1oCGzVqxGeffWZRRdY5cOAA3377LQsXLrS6FK966qmnGDlyJHfffTcATZs25cCBA4wfP97rrWMKIz7IMAwee+wxPv/8c1auXOlzndby4nA4yMzMtLoMr+jcuTNbt2512TZo0CCuueYannnmGZ8LImB25t27dy/33nuv1aV4Rbt27S4b0r9r1y5q1aplUUXWmTVrFlWrVnV25PQVZ86cwc/Pteuov78/DofD67X4ZBhJT093+d9PUlISmzdvJjw8nJo1a1pYmXcMGzaMuXPn8uWXXxISEkJycjIAYWFhlCtXzuLqvGPUqFF07dqVmjVrcvr0aebOncvKlStZunSp1aV5RUhIyGV9hCpUqEDlypV9pu/Qk08+Sffu3alVqxZHjhxh7Nix+Pv707dvX6tL84onnniCG264gXHjxnHXXXexdu1apk+fzvTp060uzascDgezZs1i4MCBPjW0G8xRha+++io1a9bk2muvZdOmTUyePJnBgwd7vxivXxgqBr777jsDuGwZOHCg1aV5RW6fHTBmzZpldWleM3jwYKNWrVpGQECAERERYXTu3NlYtmyZ1WVZytf6jPTp08eIiooyAgICjBo1ahh9+vQx9uzZY3VZXvWf//zHaNKkiREYGGhcc801xvTp060uyeuWLl1qAMbOnTutLsXr0tLSjOHDhxs1a9Y0goKCjDp16hjPPfeckZmZ6fVabIZhwVRrIiIiIv+jeUZERETEUgojIiIiYimFEREREbGUwoiIiIhYSmFERERELKUwIiIiIpZSGBERERFLKYyIiIiIpRRGRERExFIKIyIiImIphRERERGxlMKIiIiIWOr/A8CF2Hzuwgp7AAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N = 8\n",
    "alpha = pf.Parameter(\"alpha\")\n",
    "R = pf.Parameter(\"R\")\n",
    "alpha_value = 1\n",
    "R_value = 1\n",
    "\n",
    "ctx_plt = make_ctx_appm(ctx_name=\"ctx_plt\", N=N, stepsize=alpha)\n",
    "pb_plt = pf.PEPBuilder(ctx_plt)\n",
    "pb_plt.add_initial_constraint(\n",
    "    ((ctx_plt[\"x_0\"] - ctx_plt[\"x_star\"]) ** 2).le(R, name=\"initial_condition\")\n",
    ")\n",
    "\n",
    "opt_values = []\n",
    "for k in range(1, N):\n",
    "    x_k = ctx_plt[f\"x_{k}\"]\n",
    "    pb_plt.set_performance_metric(A(x_k) ** 2)\n",
    "    result = pb_plt.solve(resolve_parameters={\"alpha\": alpha_value, \"R\": R_value})\n",
    "    opt_values.append(result.opt_value)\n",
    "\n",
    "iters = np.arange(1, N)\n",
    "cont_iters = np.arange(1, N, 0.01)\n",
    "plt.plot(\n",
    "    cont_iters,\n",
    "    1 / (alpha_value**2 * cont_iters**2),\n",
    "    \"r-\",\n",
    "    label=\"Analytical bound $\\\\frac{1}{\\\\alpha^2 k^2}$\",\n",
    ")\n",
    "plt.scatter(iters, opt_values, color=\"blue\", marker=\"o\", label=\"Numerical values\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "32135ceb",
   "metadata": {},
   "source": [
    "## Verification of convergence of APPM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "79bf9989",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.06250301959936459\n"
     ]
    }
   ],
   "source": [
    "N = sp.S(4)\n",
    "alpha_value = sp.S(1)\n",
    "R_value = sp.S(1)\n",
    "\n",
    "ctx_prf = make_ctx_appm(ctx_name=\"ctx_prf\", N=N, stepsize=alpha)\n",
    "pb_prf = pf.PEPBuilder(ctx_prf)\n",
    "pb_prf.add_initial_constraint(\n",
    "    ((ctx_prf[\"x_0\"] - ctx_prf[\"x_star\"]) ** 2).le(R, name=\"initial_condition\")\n",
    ")\n",
    "pb_prf.set_performance_metric(A(ctx_prf[f\"x_{N}\"]) ** 2)\n",
    "\n",
    "result = pb_prf.solve(resolve_parameters={\"alpha\": alpha_value, \"R\": R_value})\n",
    "print(result.opt_value)\n",
    "\n",
    "# Dual variables associated with the interpolations conditions of f with no relaxation\n",
    "lamb_dense = result.get_scalar_constraint_dual_value_in_numpy(A)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "d6e4c6fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "# pf.launch_primal_interactive(\n",
    "#     pb_prf, ctx_prf, resolve_parameters={\"alpha\": alpha_value, \"R\": R_value}\n",
    "# )"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cea37a36",
   "metadata": {},
   "source": [
    "- It turns out for APPM no further relaxation is needed. Now we store the results.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d496c94b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Dual variable associated with the initial condition\n",
    "tau_sol = result.dual_var_manager.dual_value(\"initial_condition\")\n",
    "# Dual variable associated with the interpolations conditions of A\n",
    "lamb_sol = result.get_scalar_constraint_dual_value_in_numpy(A)\n",
    "# Dual variable associated with the Gram matrix G\n",
    "S_sol = result.get_gram_dual_matrix()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "881a2cab",
   "metadata": {},
   "source": [
    "### Verify closed form expression of $\\lambda$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "91fe58e5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def tag_to_index(tag, N=N):\n",
    "    \"\"\"This is a function that takes in a tag of an iterate and returns its index.\n",
    "    We index \"x_star\" as \"N+1 where N is the last iterate.\n",
    "    \"\"\"\n",
    "    # Split the string on \"_\" and get the index\n",
    "    if (idx := tag.split(\"_\")[1]).isdigit():\n",
    "        return int(idx)\n",
    "    elif idx == \"star\":\n",
    "        return N + 1"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c413d8be",
   "metadata": {},
   "source": [
    "- Print the values of $\\lambda$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "c275e7c3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_2 & x_3 & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.25 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.75 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 1.5 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "lamb_sol.pprint()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7ad66967",
   "metadata": {},
   "source": [
    "- Consider proper candidate of closed form expression of $\\lambda$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "939fc693",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_2 & x_3 & x_4 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.25 & 0.0 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.75 & 0.0 & 0.0 \\\\x_3 & 0.0 & 0.0 & 1.5 & 0.0 \\\\x_4 & 0.0 & 0.0 & 0.0 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def lamb(tag_i, tag_j, N=N):\n",
    "    i = tag_to_index(tag_i)\n",
    "    j = tag_to_index(tag_j)\n",
    "    if j - 1 == i:\n",
    "        if j == N + 1:\n",
    "            return sp.S(2) / N  ## Between N and optimal\n",
    "        else:\n",
    "            return sp.S(2) * (sp.S(j) - sp.S(1)) * sp.S(j) / N**2  ## Consecutive\n",
    "    return 0\n",
    "\n",
    "\n",
    "lamb_cand = pf.pprint_labeled_matrix(\n",
    "    lamb, lamb_sol.row_names, lamb_sol.col_names, return_matrix=True\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "564d90bb",
   "metadata": {},
   "source": [
    "- Check whether our candidate of $\\lambda$ matches with solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "9c45d58e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Did we guess the right closed form of lambda? True\n"
     ]
    }
   ],
   "source": [
    "print(\n",
    "    \"Did we guess the right closed form of lambda?\",\n",
    "    np.allclose(lamb_cand, lamb_sol.matrix, atol=1e-3),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c3308b36",
   "metadata": {},
   "source": [
    "### Closed form expression of $S$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "7a8d4dab",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & y_0 & x_\\star & A(x_1) & A(x_2) & A(x_3) & A(x_4) \\\\\n",
       "        \\hline\n",
       "        y_0 & 0.062 & -0.062 & -0.0 & 0.0 & 0.0 & -0.25 \\\\x_\\star & -0.062 & 0.062 & 0.0 & -0.0 & -0.0 & 0.25 \\\\A(x_1) & -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 \\\\A(x_3) & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 \\\\A(x_4) & -0.25 & 0.25 & 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": [
    "S_sol.pprint()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "8fd03963",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[y_0, x_star, A(x_1), A(x_2), A(x_3), A(x_4)]"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ctx_prf.basis_vectors()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "d87aeca4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & y_0 & x_\\star & A(x_1) & A(x_2) & A(x_3) & A(x_4) \\\\\n",
       "        \\hline\n",
       "        y_0 & 0.062 & -0.062 & 0.0 & 0.0 & 0.0 & -0.25 \\\\x_\\star & -0.062 & 0.062 & 0.0 & 0.0 & 0.0 & 0.25 \\\\A(x_1) & 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 \\\\A(x_3) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\A(x_4) & -0.25 & 0.25 & 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": [
    "x_0 = ctx_prf[\"x_0\"]\n",
    "x_N = ctx_prf[f\"x_{N}\"]\n",
    "x_star = ctx_prf[\"x_star\"]\n",
    "\n",
    "S_guess = (A(x_N) - 1 / (alpha * N) * (x_0 - x_star)) ** 2\n",
    "\n",
    "pm = pf.ExpressionManager(ctx_prf, resolve_parameters={\"alpha\": sp.S(1)})\n",
    "S_guess_eval = pm.eval_scalar(S_guess).matrix\n",
    "pf.pprint_labeled_matrix(S_guess_eval, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "1a17abbe",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Did we guess the right closed form of S? True\n"
     ]
    }
   ],
   "source": [
    "print(\n",
    "    \"Did we guess the right closed form of S?\",\n",
    "    np.allclose(S_guess_eval, S_sol.matrix, atol=1e-3),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8c2d605d",
   "metadata": {},
   "source": [
    "#### Verify symbolic calculation for fixed $N$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "c0406d9e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/4/alpha*-(x_1-(x_2))*(A(x_1)-A(x_2))+3/4/alpha*-(x_2-(x_3))*(A(x_2)-A(x_3))+3/2/alpha*-(x_3-(x_4))*(A(x_3)-A(x_4))+1/2/alpha*-(x_4-x_\\star)*(A(x_4)-A(x_\\star))$"
      ],
      "text/plain": [
       "0+1/4/alpha*-(x_1-(x_2))*(A(x_1)-A(x_2))+3/4/alpha*-(x_2-(x_3))*(A(x_2)-A(x_3))+3/2/alpha*-(x_3-(x_4))*(A(x_3)-A(x_4))+1/2/alpha*-(x_4-x_star)*(A(x_4)-A(x_star))"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "interpolation_scalar_sum = 0\n",
    "for i in range(N + 2):\n",
    "    for j in range(N + 2):\n",
    "        xi = \"x_star\" if i == N + 1 else f\"x_{i}\"\n",
    "        xj = \"x_star\" if j == N + 1 else f\"x_{j}\"\n",
    "        if lamb(xi, xj) != 0:\n",
    "            interpolation_scalar_sum += lamb(xi, xj) / alpha * A.interp_ineq(xi, xj)\n",
    "\n",
    "interpolation_scalar_sum"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "3c6186dc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/4/alpha*-(x_1-(x_2))*(A(x_1)-A(x_2))+3/4/alpha*-(x_2-(x_3))*(A(x_2)-A(x_3))+3/2/alpha*-(x_3-(x_4))*(A(x_3)-A(x_4))+1/2/alpha*-(x_4-x_\\star)*(A(x_4)-A(x_\\star))-\\|A(x_4)-1/alpha*4*(y_0-x_\\star)\\|^2$"
      ],
      "text/plain": [
       "0+1/4/alpha*-(x_1-(x_2))*(A(x_1)-A(x_2))+3/4/alpha*-(x_2-(x_3))*(A(x_2)-A(x_3))+3/2/alpha*-(x_3-(x_4))*(A(x_3)-A(x_4))+1/2/alpha*-(x_4-x_star)*(A(x_4)-A(x_star))-|A(x_4)-1/alpha*4*(y_0-x_star)|^2"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "RHS = interpolation_scalar_sum - S_guess\n",
    "display(RHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "65123cc6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\|A(x_4)\\|^2-1/alpha^2*16*\\|y_0-x_\\star\\|^2$"
      ],
      "text/plain": [
       "|A(x_4)|^2-1/alpha**2*16*|y_0-x_star|^2"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "LHS = A(x_N) ** 2 - 1 / (alpha**2 * N**2) * (x_0 - x_star) ** 2\n",
    "display(LHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "c56a037e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\|A(x_4)\\|^2-1/alpha^2*16*\\|y_0-x_\\star\\|^2-(0+1/4/alpha*-(x_1-(x_2))*(A(x_1)-A(x_2))+3/4/alpha*-(x_2-(x_3))*(A(x_2)-A(x_3))+3/2/alpha*-(x_3-(x_4))*(A(x_3)-A(x_4))+1/2/alpha*-(x_4-x_\\star)*(A(x_4)-A(x_\\star))-\\|A(x_4)-1/alpha*4*(y_0-x_\\star)\\|^2)$"
      ],
      "text/plain": [
       "|A(x_4)|^2-1/alpha**2*16*|y_0-x_star|^2-(0+1/4/alpha*-(x_1-(x_2))*(A(x_1)-A(x_2))+3/4/alpha*-(x_2-(x_3))*(A(x_2)-A(x_3))+3/2/alpha*-(x_3-(x_4))*(A(x_3)-A(x_4))+1/2/alpha*-(x_4-x_star)*(A(x_4)-A(x_star))-|A(x_4)-1/alpha*4*(y_0-x_star)|^2)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "difference = LHS - RHS\n",
    "display(difference)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "c9f5d8f5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle 0$"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "pf.pprint_str(\n",
    "    difference.repr_by_basis(\n",
    "        ctx_prf, sympy_mode=True, resolve_parameters={\"alpha\": sp.S(\"alpha\")}\n",
    "    )\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2f25e295",
   "metadata": {},
   "source": [
    "\\begin{align*}\n",
    "    \\| \\tilde{\\mathbb{A}}(x^N) \\|^2 - \\frac{\\| x^0 - x^\\star \\|^2}{\\alpha^2 N^2}\n",
    "    &= -\\sum_{k=1}^{N-1} \\frac{2k(k+1)}{\\alpha^2 N^2} \\langle \\tilde{\\mathbb{A}}(x^{k+1}) - \\tilde{\\mathbb{A}}(x^k), x^{k+1} - x^k \\rangle \\\\&\\quad \n",
    "    - \\frac{2}{\\alpha N} \\langle \\tilde{\\mathbb{A}}(x^{N}), x^{N} - x^\\star \\rangle \\\\&\\quad \n",
    "    - \\| \\tilde{\\mathbb{A}}(x^N) - \\frac{1}{\\alpha N} ( x^0 - x^\\star)  \\| ^2.\n",
    "\\end{align*}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3ed3c248",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Compute the sum of active inequalities up to k-th iteration\n",
    "# lyap[k] = sum_{j=1}^{k+1} lambda(x_j, x_{j+1}) * A.interp_ineq(x_j, x_{j+1})\n",
    "\n",
    "lyap = []\n",
    "partial_sum = 0\n",
    "for i in np.arange(1, N):\n",
    "    xi = f\"x_{i}\"\n",
    "    xj = f\"x_{i + 1}\"\n",
    "    partial_sum += lamb(xi, xj) * A.interp_ineq(xi, xj)\n",
    "    coords_matrix = pm.eval_scalar(partial_sum).inner_prod_coords\n",
    "    lyap.append(coords_matrix.astype(float))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b0d193cc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Rank of lyap[0]: 2\n",
      "Rank of lyap[1]: 2\n",
      "Rank of lyap[2]: 2\n"
     ]
    }
   ],
   "source": [
    "# Check that rank is constantly 2\n",
    "for k in range(len(lyap)):\n",
    "    print(f\"Rank of lyap[{k}]: {np.linalg.matrix_rank(lyap[k], tol=1e-6)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "id": "44ae6a94",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Collect \"special vectors\" by tags: first add the basis vectors add the x_i\n",
    "special_tags = [v.tag for v in ctx_prf.basis_vectors()]\n",
    "for i in np.arange(1, N + 1):\n",
    "    special_tags.append(f\"x_{i}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2b32a1fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Get the numerical coords vector for each special vector\n",
    "special_vectors = [\n",
    "    np.asarray(pm.eval_vector(ctx_prf[tag]).coords, dtype=float) for tag in special_tags\n",
    "]\n",
    "\n",
    "# Add any two difference between special vectors as new special vectors\n",
    "for i, j in combinations(range(len(special_vectors)), 2):\n",
    "    diff = special_vectors[i] - special_vectors[j]\n",
    "    special_vectors.append(diff)\n",
    "    special_tags.append(f\"{special_tags[i]}-{special_tags[j]}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "7e18f92d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['y_0',\n",
       " 'x_star',\n",
       " 'A(x_1)',\n",
       " 'A(x_2)',\n",
       " 'A(x_3)',\n",
       " 'A(x_4)',\n",
       " 'x_1',\n",
       " 'x_2',\n",
       " 'x_3',\n",
       " 'x_4',\n",
       " 'y_0-x_star',\n",
       " 'y_0-A(x_1)',\n",
       " 'y_0-A(x_2)',\n",
       " 'y_0-A(x_3)',\n",
       " 'y_0-A(x_4)',\n",
       " 'y_0-x_1',\n",
       " 'y_0-x_2',\n",
       " 'y_0-x_3',\n",
       " 'y_0-x_4',\n",
       " 'x_star-A(x_1)',\n",
       " 'x_star-A(x_2)',\n",
       " 'x_star-A(x_3)',\n",
       " 'x_star-A(x_4)',\n",
       " 'x_star-x_1',\n",
       " 'x_star-x_2',\n",
       " 'x_star-x_3',\n",
       " 'x_star-x_4',\n",
       " 'A(x_1)-A(x_2)',\n",
       " 'A(x_1)-A(x_3)',\n",
       " 'A(x_1)-A(x_4)',\n",
       " 'A(x_1)-x_1',\n",
       " 'A(x_1)-x_2',\n",
       " 'A(x_1)-x_3',\n",
       " 'A(x_1)-x_4',\n",
       " 'A(x_2)-A(x_3)',\n",
       " 'A(x_2)-A(x_4)',\n",
       " 'A(x_2)-x_1',\n",
       " 'A(x_2)-x_2',\n",
       " 'A(x_2)-x_3',\n",
       " 'A(x_2)-x_4',\n",
       " 'A(x_3)-A(x_4)',\n",
       " 'A(x_3)-x_1',\n",
       " 'A(x_3)-x_2',\n",
       " 'A(x_3)-x_3',\n",
       " 'A(x_3)-x_4',\n",
       " 'A(x_4)-x_1',\n",
       " 'A(x_4)-x_2',\n",
       " 'A(x_4)-x_3',\n",
       " 'A(x_4)-x_4',\n",
       " 'x_1-x_2',\n",
       " 'x_1-x_3',\n",
       " 'x_1-x_4',\n",
       " 'x_2-x_3',\n",
       " 'x_2-x_4',\n",
       " 'x_3-x_4']"
      ]
     },
     "execution_count": 62,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "special_tags"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bbbc816e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Check if each vector in a given list is in the column space of the matrix A\n",
    "\n",
    "\n",
    "def vectors_in_column_space(A, vectors, rtol=1e-7, atol=1e-12):\n",
    "    \"\"\"\n",
    "    A: (d, d) numpy array\n",
    "    vectors: list of (d,) numpy arrays\n",
    "    returns: list of bools, same length as vectors\n",
    "    \"\"\"\n",
    "    A = np.asarray(A, dtype=float)\n",
    "    # d = A.shape[0]\n",
    "\n",
    "    # SVD of A\n",
    "    U, S, Vt = np.linalg.svd(A, full_matrices=False)\n",
    "\n",
    "    # numerical rank\n",
    "    tol = atol + rtol * S[0]\n",
    "    rank = np.sum(S > tol)\n",
    "\n",
    "    # basis for column space\n",
    "    Uc = U[:, :rank]\n",
    "\n",
    "    results = []\n",
    "    for v in vectors:\n",
    "        v = np.asarray(v, dtype=float)\n",
    "        proj = Uc @ (Uc.T @ v)\n",
    "        residual = np.linalg.norm(v - proj)\n",
    "        results.append(residual <= atol + rtol * np.linalg.norm(v))\n",
    "\n",
    "    return results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "458a814a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['A(x_1)', 'A(x_2)', 'y_0-x_1', 'y_0-x_2', 'A(x_1)-A(x_2)', 'x_1-x_2']\n"
     ]
    }
   ],
   "source": [
    "# Extract the special vectors representing lyap[k] = V_{k+1}\n",
    "V1_aligned_vectors = [\n",
    "    special_tags[idx]\n",
    "    for idx, (v, b) in enumerate(\n",
    "        zip(special_vectors, vectors_in_column_space(lyap[0], special_vectors))\n",
    "    )\n",
    "    if b\n",
    "]\n",
    "print(V1_aligned_vectors)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "50b6d9c5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['A(x_3)', 'y_0-x_3']\n"
     ]
    }
   ],
   "source": [
    "V2_aligned_vectors = [\n",
    "    special_tags[idx]\n",
    "    for idx, (v, b) in enumerate(\n",
    "        zip(special_vectors, vectors_in_column_space(lyap[1], special_vectors))\n",
    "    )\n",
    "    if b\n",
    "]\n",
    "print(V2_aligned_vectors)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "d2f50de2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['A(x_4)', 'y_0-x_4']\n"
     ]
    }
   ],
   "source": [
    "V3_aligned_vectors = [\n",
    "    special_tags[idx]\n",
    "    for idx, (v, b) in enumerate(\n",
    "        zip(special_vectors, vectors_in_column_space(lyap[2], special_vectors))\n",
    "    )\n",
    "    if b\n",
    "]\n",
    "print(V3_aligned_vectors)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f6b90414",
   "metadata": {},
   "outputs": [],
   "source": [
    "# V_{k+1} is represented using A(x_{k+2}) and y_0 - x_{k+2}\n",
    "# Let's find the corresponding coefficients"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4af77df1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def decompose_rank2_symmetric(S, v, w, *, sym_tol=1e-10, rank_tol=1e-12):\n",
    "    \"\"\"\n",
    "    Find a,b,c such that\n",
    "      S = a vv^T + b (vw^T + wv^T) + c ww^T\n",
    "    assuming S is symmetric and rank(S)=2 and v,w are linearly independent in range(S).\n",
    "\n",
    "    Returns: (a,b,c)\n",
    "    \"\"\"\n",
    "    S = np.asarray(S, dtype=float)\n",
    "    v = np.asarray(v, dtype=float).reshape(-1)\n",
    "    w = np.asarray(w, dtype=float).reshape(-1)\n",
    "    n = S.shape[0]\n",
    "    assert S.shape == (n, n), \"S must be square\"\n",
    "    assert v.shape == (n,) and w.shape == (n,), \"v,w must have shape (n,)\"\n",
    "\n",
    "    # Basic checks (optional but helpful)\n",
    "    if np.linalg.norm(S - S.T, ord=\"fro\") > sym_tol * max(\n",
    "        1.0, np.linalg.norm(S, ord=\"fro\")\n",
    "    ):\n",
    "        raise ValueError(\"S is not symmetric within tolerance.\")\n",
    "    if np.linalg.matrix_rank(S, tol=rank_tol) != 2:\n",
    "        raise ValueError(\"S does not have rank 2 (within tolerance).\")\n",
    "\n",
    "    # Pick two indices i,j so that the 2x2 matrix [v_i w_i; v_j w_j] is well-conditioned\n",
    "    pairs = [(i, j) for i in range(n) for j in range(i + 1, n)]\n",
    "    best = None\n",
    "    best_score = -np.inf\n",
    "    for i, j in pairs:\n",
    "        M = np.array([[v[i], w[i]], [v[j], w[j]]], dtype=float)\n",
    "        det = abs(np.linalg.det(M))\n",
    "        if det > best_score:\n",
    "            best_score = det\n",
    "            best = (i, j)\n",
    "\n",
    "    if best is None or best_score < 1e-14:\n",
    "        raise ValueError(\n",
    "            \"Could not find a stable pair of coordinates; v,w may be nearly dependent.\"\n",
    "        )\n",
    "\n",
    "    i, j = best\n",
    "\n",
    "    # Use equations at entries (i,i), (j,j), (i,j):\n",
    "    # S_ii = a v_i^2 + 2b v_i w_i + c w_i^2\n",
    "    # S_jj = a v_j^2 + 2b v_j w_j + c w_j^2\n",
    "    # S_ij = a v_i v_j + b (v_i w_j + w_i v_j) + c w_i w_j\n",
    "    A = np.array(\n",
    "        [\n",
    "            [v[i] * v[i], 2.0 * v[i] * w[i], w[i] * w[i]],\n",
    "            [v[j] * v[j], 2.0 * v[j] * w[j], w[j] * w[j]],\n",
    "            [v[i] * v[j], (v[i] * w[j] + w[i] * v[j]), w[i] * w[j]],\n",
    "        ],\n",
    "        dtype=float,\n",
    "    )\n",
    "    rhs = np.array([S[i, i], S[j, j], S[i, j]], dtype=float)\n",
    "\n",
    "    # Solve (robustly) for a,b,c\n",
    "    abc, *_ = np.linalg.lstsq(A, rhs, rcond=None)\n",
    "    a, b, c = abc\n",
    "\n",
    "    # Optional: validate (you can remove if you want)\n",
    "    approx = (\n",
    "        a * np.outer(v, v) + b * (np.outer(v, w) + np.outer(w, v)) + c * np.outer(w, w)\n",
    "    )\n",
    "    rel_err = np.linalg.norm(S - approx, ord=\"fro\") / max(\n",
    "        1.0, np.linalg.norm(S, ord=\"fro\")\n",
    "    )\n",
    "    if rel_err > 1e-8:\n",
    "        # try a more global solve (still only 3 unknowns)\n",
    "        # minimize over all entries: vec(S) ≈ a vec(vv^T) + b vec(vw^T+wv^T) + c vec(ww^T)\n",
    "        X = np.stack(\n",
    "            [\n",
    "                np.outer(v, v).reshape(-1),\n",
    "                (np.outer(v, w) + np.outer(w, v)).reshape(-1),\n",
    "                np.outer(w, w).reshape(-1),\n",
    "            ],\n",
    "            axis=1,\n",
    "        )\n",
    "        y = S.reshape(-1)\n",
    "        abc2, *_ = np.linalg.lstsq(X, y, rcond=None)\n",
    "        a, b, c = abc2\n",
    "        approx = (\n",
    "            a * np.outer(v, v)\n",
    "            + b * (np.outer(v, w) + np.outer(w, v))\n",
    "            + c * np.outer(w, w)\n",
    "        )\n",
    "        rel_err2 = np.linalg.norm(S - approx, ord=\"fro\") / max(\n",
    "            1.0, np.linalg.norm(S, ord=\"fro\")\n",
    "        )\n",
    "        if rel_err2 > 1e-8:\n",
    "            raise ValueError(\n",
    "                f\"Decomposition seems inconsistent / numerically unstable (rel err {rel_err2:g}).\"\n",
    "            )\n",
    "\n",
    "    return float(a), float(b), float(c)\n",
    "\n",
    "\n",
    "# Example usage:\n",
    "# a,b,c = decompose_rank2_symmetric(S, v, w)\n",
    "# S_recon = a*np.outer(v,v) + b*(np.outer(v,w)+np.outer(w,v)) + c*np.outer(w,w)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "379d7340",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.5, -0.12499999999999994, -6.938893903907228e-18)"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# lyap[0] = 0.5 * ||A(x_2)||^2 - 0.25 * <A(x_2), y_0 - x_2>\n",
    "decompose_rank2_symmetric(\n",
    "    lyap[0],\n",
    "    pm.eval_vector(ctx_prf[\"A(x_2)\"]).coords,\n",
    "    pm.eval_vector(ctx_prf[\"y_0\"]).coords - pm.eval_vector(ctx_prf[\"x_2\"]).coords,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd8f58c4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(1.1249999999999998, -0.18749999999999997, -2.7755575615628914e-17)"
      ]
     },
     "execution_count": 49,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# lyap[1] = 1.125 * ||A(x_3)||^2 - 0.375 * <A(x_3), y_0 - x_3>\n",
    "decompose_rank2_symmetric(\n",
    "    lyap[1],\n",
    "    pm.eval_vector(ctx_prf[\"A(x_3)\"]).coords,\n",
    "    pm.eval_vector(ctx_prf[\"y_0\"]).coords - pm.eval_vector(ctx_prf[\"x_3\"]).coords,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7751fb74",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(2.0000000000000004, -0.25000000000000006, 1.942890293094024e-16)"
      ]
     },
     "execution_count": 50,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# lyap[2] = 2 * ||A(x_4)||^2 - 0.5 * <A(x_4), y_0 - x_4>\n",
    "decompose_rank2_symmetric(\n",
    "    lyap[2],\n",
    "    pm.eval_vector(ctx_prf[\"A(x_4)\"]).coords,\n",
    "    pm.eval_vector(ctx_prf[\"y_0\"]).coords - pm.eval_vector(ctx_prf[\"x_4\"]).coords,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "4a394be5",
   "metadata": {},
   "outputs": [],
   "source": [
    "# V_{k+1} = 1/8 * ((k+2)^2 * ||A(x_{k+2})||^2) - (k+2) * <A(x_{k+2}), y_0 - x_{k+2}>)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e0c74a58",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "pepflow",
   "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.1"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
