{
 "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": "markdown",
   "id": "5a10dfb8",
   "metadata": {},
   "source": [
    "## Import the required libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "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 itertools import combinations\n",
    "from IPython.display import display, Math"
   ]
  },
  {
   "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 0x1275b01a0>"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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G1S7dm01bdpUzz77bJ7XWb58uc466yy3vW1z8skna+nSpbu97l133aXGjRurdevWBXbT/P7777rsssvcvi1gtWvXTm+//bb72S+//KJzzjlHTZo0cQHmoIMO0ouW+Irpp59+ci1FP/74Y57HH3zwQbVq1crdzs7O1kUXXaR99tlHVapUccf50EMPlbirqUOHDu58R76viy++2IWvjIwMHXPMMZo9e3buz+12z549VaNGDffzjh076uuvv1YyI4yE0U0DII4scJxxxu5//9h0R/Z4LANJ+fLlXYB45JFH9HMZ/wD73//+p5UrV+rjjz/W6NGjNXLkSJ144omqVauWvvzyS11++eUuMIT388cff6h3797ui/STTz7RZ599purVq+v//u//cltAzNSpUzV//nx98MEHuQEjUk5Ojk444QT3/Oeff15z5851XU/23sKXtbcv6XfeeUdz5sxx3TsXXHCBZsyYUaz3tf/++6tTp076z3/+k+dxu3+uJcc/j8HC1ssvv+z2P2LECN1000166aWXynROzzzzTK1du1bvvfeevvnmGx166KE69thj9atNPyHpvPPOc/u1biv7+dChQ1WxYkUltVASyMrKCtmh2jpm5s4Nhex01KwZu30ASBlbt24NzZ07161LaufOUKhpU+9XTkFLWloo1KyZt1209e/fP3TyySe72127dg0NHDjQ3X7ttdfc79mwkSNHhtq3b5/nuQ8++GBo7733zvNadj87Ozv3sdatW4e6d+8e8V53hqpVqxZ68cUX3f1///vfbpucnJzcbbZv3x6qUqVKaMqUKbmv26BBA/d4JNuXHYOxbcuVKxeaP39+sd97nz59Qn/7299y7/fo0SM0aNCgQre3fbVq1Sr3vu3LztG8efMKfc5VV10VOv300ws83/nfQ1j79u3d+TaffPJJKCMjI7Rt27Y829hxPPnkk+52jRo1Qs8991woGf4tFPf7m5aR/DUjWVnSxo3+pUMAKe+TT4ruEbZIYuVrtl0sWd3I+PHjNW/evFK/xoEHHqhy5XZ9lViXiXWJhFlLRZ06ddxf+uEuhoULF7qWEWsRscW6aqwlY9GiRbnPs9dIT08vdL/ffvutax2wFoyCWBfKHXfc4V7HXt/2M2XKFGVmZhb7vZ199tmu++iLL77IbRWxVooDDjggd5uxY8e6FhjrUrF9jBs3rkT7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YWT06NG65JJLNGDAALVt21ZPPPGEqlatqmeeeabQ52RnZ+u8887TbbfdppYtW5b1mFPXGWdI1apJCxZIX3zh99EAAJB4YWTHjh365ptv1KtXr10vUK6cu//5558X+rzbb79d9evX10UXXVSs/Wzfvl0bNmzIswRC9eq7ZmR97jm/jwYAgMQLI+vXr3etHA0aNMjzuN1fvXp1gc/59NNP9fTTT+upp54q9n5GjRqlmjVr5i7N7IJyQdG/v7eeOFHautXvowEAILlH02zcuFEXXHCBCyJ169Yt9vOGDRumrKys3GX58uUKjKOPlpo3l7KypDff9PtoAACIuQol2dgCRfny5bVmzZo8j9v9hjYaJJ9Fixa5wtWTTjop97GcnBxvxxUqaP78+WrVqtVuz6tUqZJbAsmG9/brJ915p9dV07ev30cEAEDitIykp6erY8eOmjp1ap5wYfe7deu22/YHHHCAvv/+e3377be5y1//+lf17NnT3Q5U90tpumref18KUqsQACCQStQyYmxYb//+/dWpUycddthhGjNmjDZv3uxG15h+/fqpSZMmru7D5iFp165dnufvtddebp3/cUTYd1+vu2baNOmf/5Ruu43TAwBIWSUOI3379tW6des0YsQIV7TaoUMHTZ48ObeoNTMz042wQRldfrkXRqzw95ZbpIoVOaUAgJSUFgol/uxaNrTXRtVYMWtGRoYCYccOqWlTad06adIk6dRT/T4iAABi8v1NE0aiSk+XwvOyPPmk30cDAEDMEEYS2SWXSGlp0pQp0uLFfh8NAAAxQRhJZDZ1/vHHe7fHjfP7aAAAiAnCSDIUshq79o/VkQAAkGIII4nuxBOlJk28QlabIh4AgBRDGEl0FSpIV17p3X7wQSnxBz8BAFAihJFkcNllUpUq0qxZ0scf+300AABEFWEkGdSps2uK+NGj/T4aAACiijCSLAYP9tZvvSUtWOD30QAAEDWEkWTRurXUp49XM/LQQ34fDQAAUUMYSSZDhnjrZ5+Vfv3V76MBACAqCCPJpGdPqX17acsW6fHH/T4aAACigjCSTGxq+Btu2DXMd/Nmv48IAIAyI4wkm759pVatpF9+4QJ6AICUQBhJxknQhg3zbt9/v7Rtm99HBABAmRBGktEFF0jNm0urV3vXrAEAIIkRRpJRerp0443e7Xvv5QJ6AICkRhhJVgMHSg0bSpmZ0r//7ffRAABQaoSRZFW58q6RNXfcIW3f7vcRAQBQKoSRZHb55VKjRtKyZdK4cX4fDQAApUIYSWZVq0ojRni377xT2rTJ7yMCAKDECCPJ7qKLvHlH1q7lmjUAgKREGEl2FSt6NSPmvvu8ydAAAEgihJFUmZXVrlmzYYM31BcAgCRCGEkF5cpJd93l3X74YWnpUr+PCACAYiOMpIq//EU65hhviO/f/+730QAAUGyEkVS6ou+YMV4rycsvSx9/7PcRAQBQLISRVHLQQdKll3q3Bw+WsrP9PiIAAPaIMJJqbr9dqllTmjVLeu45v48GAIA9Ioykmnr1pJEjvds33SRlZfl9RAAAFIkwkoquukpq3dqbCM0CCQAACYwwkorS06XHH/du23rGDL+PCACAQhFGUlXPnlK/flIo5BW17tzp9xEBAFAgwkgq+8c/pNq1pdmzuW4NACBhEUZSvZjVrldj7Oq+y5b5fUQAAOyGMJLqBgyQuneXtmzxrvBr3TYAACQQwkiqsxlZ//lPqUoVaepU6Ykn/D4iAADyIIwEwf77S6NGebdvuEFavNjvIwIAIBdhJCiuuUY66ihp82Zp4EApJ8fvIwIAwCGMBKm75tlnpWrVpI8+cqNr7NI106ZJL77orbmUDQDAD4SRIGnZ0hvuKyn770PVp/EsNx3Jued605K0aCFNmuT3QQIAgoYwEjSXXaaVnU9W+Z079NDas1VNm3J/tGKFdMYZBBIAQHwRRgImOydNvX9+Wj+riVrrJz2kQbk/C4/6HTyYLhsAQPwQRgLmk0+kOavq6Dz9RzlK00V6Rn01IU8gWb7c2w4AgHggjATMqlXe+mP10J26xd1+SpeojeYWuB0AALFGGAmYRo123b5dI/Q/9VQNbdJrOlUZyipwOwAAYokwEjA2M3zTplJampStCuqricpUM1c/Ml79VU45atbM2w4AgHggjARM+fK7LuBrgWS96ul0vaptqqRT9IaGaZTGjPG2AwAgHggjAXTaadIrr0hNmnj3v1ZnXanH3O070obrtCrv+XuAAIBAIYwEOJAsXSp9+KH0wgtSvw8HKufSy5Rmw2n69pW+/97vQwQABEQFvw8A/rGumKOPjnjg8Ieln+Z7c8P36SN98YXUuLGPRwgACAJaRrBLero3/Wrr1t5kIyed5F1YDwCAGCKMIK9ataR335Xq1ZNmzpTOOYfpWAEAMUUYQcEX1HvzTalyZemtt6Srr941VzwAAFFGGEHBunaVnn/eG//7xBPSTTdxpgAAMUEYQeFOP10aN867fc893gIAQJQRRlC0iy+W/vEP7/awYdJj3nwkAAD4GkbGjh2rFi1aqHLlyurSpYtmzJhR6LZPPfWUunfvrlq1armlV69eRW6PBPS3v0m3eBfV01VXSc8+6/cRAQCCHEYmTpyoIUOGaOTIkZo5c6bat2+v3r17a+3atQVuP23aNJ1zzjn68MMP9fnnn6tZs2Y6/vjjtWLFimgcP+Ll9tula67xbg8caCmTcw8AiIq0UKhkwySsJaRz58569NFH3f2cHLuwWjNdc801Gjp06B6fn52d7VpI7Pn9+vUr1j43bNigmjVrKisrSxkZGSU5XEST/a8yaJD0yCPe/bFjpSuv5BwDAMr0/V2ilpEdO3bom2++cV0tuS9Qrpy7b60exbFlyxb98ccfql27dkl2jURgI2vsKntDhuzqsnn4Yb+PCgAQpOng169f71o2GjRokOdxu//jjz8W6zVuvPFGNW7cOE+gyW/79u1uiUxWSKBAYgWtFStK997rtZRkZXk1JfYzAAASeTTNPffcowkTJui1115zxa+FGTVqlGvWCS/WDYQEYqFj1Chp+HDv/ogR3sRo2dl+HxkAINXDSN26dVW+fHmtWbMmz+N2v2HDhkU+9x//+IcLI++//74OPvjgIrcdNmyY618KL8vtOilIvEBiRa3WTWO3bcjv2WdL27b5fWQAgFQOI+np6erYsaOmTp2a+5gVsNr9bt26Ffq8++67T3fccYcmT56sTp067XE/lSpVcoUukQsSlI2wmTDB67Z55RXphBOk337z+6gAAKncTWPDem3ukPHjx2vevHm64oortHnzZg0YMMD93EbIWMtG2L333qvhw4frmWeecXOTrF692i2bNm2K7juBf846S3rvPalGDRvLbUOupPnz+UQAALEJI3379nVdLiNGjFCHDh307bffuhaPcFFrZmamVq1albv9448/7kbhnHHGGWrUqFHuYq+BFHLssdInn0jNm0sLFniB5IMP/D4qAEAqzjPiB+YZSSJWT3TaadL06VL58tKYMd4QYEbaAEDgbIjFPCPAHlkL2f/+Z/113ugaqym55BJp61ZOHgCgQIQRRF+lStJzz1nlstci8vTT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trtvo6nsXNiIy3i3jhFGAsgahK655hrXLTFt2rTAFa0V1WVlX8JBcOyxx7puqkhWRW/NtTfeeGPggki4mHfRokW64IILFATWNZt/SL/VDuy9994KGivitbqZcCFnUGzZskXlyuUtHbV/+/a7MN4CGUbsl07kXz9LlizRt99+6wo4mzdvriAULb7wwgt64403XN/46tWr3eM2nMvGmgfBsGHDXJGWfd4bN25058OC2ZQpUxQE9rnnrxGyYd0230RQaoeuv/56N6rKvnxXrlzprlhqv4jPOeccBcF1113nChjvvvtunXXWWW4Exbhx49wSJPbFa2HEWgKCNLTb2P//ViNivwdtFKFNbTB69GhXxBt3oQD68MMPXa1I/qV///6hICjovdvy7LPPhoJi4MCBrlYmPT09VK9evdCxxx4bev/990NBFrSakb59+4YaNWrk/h9o0qSJu79w4cJQkLz11luhdu3ahSpVqhQ64IADQuPGjQsFzZQpU9zvv/nz54eCZsOGDe7ffPPmzUOVK1cOtWzZMnTzzTe7erp4S7P/xD8CAQAAeJhnBAAA+IowAgAAfEUYAQAAviKMAAAAXxFGAACArwgjAADAV4QRAADgK8IIAADwFWEEAAD4ijACAAB8RRgBAAC+IowAAAD56f8BwIXYfLVn4W0AAAAASUVORK5CYII=",
      "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.06250301959937361\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": "markdown",
   "id": "5e00cf3f",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "2b32a1fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "special_vectors = ctx_prf.basis_vectors()\n",
    "\n",
    "y_0 = ctx_prf[f\"y_{0}\"]\n",
    "x = [y_0] + [ctx_prf[f\"x_{i}\"] for i in range(1, N + 1)]\n",
    "special_vectors += x[1:]\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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "7e18f92d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[y_0, x_star, A(x_1), A(x_2), A(x_3), A(x_4), x_1, x_2, x_3, x_4, y_0-x_star, y_0-A(x_1), y_0-A(x_2), y_0-A(x_3), y_0-A(x_4), y_0-(x_1), y_0-(x_2), y_0-(x_3), y_0-(x_4), x_star-A(x_1), x_star-A(x_2), x_star-A(x_3), x_star-A(x_4), x_star-(x_1), x_star-(x_2), x_star-(x_3), x_star-(x_4), A(x_1)-A(x_2), A(x_1)-A(x_3), A(x_1)-A(x_4), A(x_1)-(x_1), A(x_1)-(x_2), A(x_1)-(x_3), A(x_1)-(x_4), A(x_2)-A(x_3), A(x_2)-A(x_4), A(x_2)-(x_1), A(x_2)-(x_2), A(x_2)-(x_3), A(x_2)-(x_4), A(x_3)-A(x_4), A(x_3)-(x_1), A(x_3)-(x_2), A(x_3)-(x_3), A(x_3)-(x_4), A(x_4)-(x_1), A(x_4)-(x_2), A(x_4)-(x_3), A(x_4)-(x_4), x_1-(x_2), x_1-(x_3), x_1-(x_4), x_2-(x_3), x_2-(x_4), x_3-(x_4)]\n"
     ]
    }
   ],
   "source": [
    "print(special_vectors)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ccd3a2c6",
   "metadata": {},
   "source": [
    "## Identify the vectors composing the Lyapunov function"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "60f505ad",
   "metadata": {},
   "source": [
    "Compute the sum of active inequalities up to k-th iteration"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "08b40c13",
   "metadata": {},
   "outputs": [],
   "source": [
    "lyap = [pf.Scalar.zero()]\n",
    "partial_sum = 0\n",
    "for j in np.arange(1, N):\n",
    "    partial_sum += lamb(f\"x_{j}\", f\"x_{j + 1}\") * A.interp_ineq(f\"x_{j}\", f\"x_{j + 1}\")\n",
    "    lyap.append(partial_sum)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "34bab1c8",
   "metadata": {},
   "source": [
    "Check that the ranks of V_k are constantly 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "d58cc17c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Rank of lyap[0]: 0\n",
      "Rank of lyap[1]: 2\n",
      "Rank of lyap[2]: 2\n",
      "Rank of lyap[3]: 2\n"
     ]
    }
   ],
   "source": [
    "for k in range(len(lyap)):\n",
    "    lyap_numeric_k = pm.eval_scalar(lyap[k]).inner_prod_coords.astype(float)\n",
    "    print(f\"Rank of lyap[{k}]: {np.linalg.matrix_rank(lyap_numeric_k, tol=1e-6)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "bbbc816e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Collect the vectors in the column space of matrix A from a given list\n",
    "\n",
    "\n",
    "def vectors_in_column_space(V, vectors, rtol=1e-7, atol=1e-12):\n",
    "    \"\"\"\n",
    "    V: quadratic form of vectors that corresponds to a matrix\n",
    "    vectors: list of vectors\n",
    "    returns: list of vectors that are in col(V) up to numerical tolerance\n",
    "    \"\"\"\n",
    "    V_coords = np.asarray(pm.eval_scalar(V).inner_prod_coords, dtype=float)\n",
    "\n",
    "    # SVD of A\n",
    "    U, S, _ = np.linalg.svd(V_coords, 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_coords = np.asarray(pm.eval_vector(v).coords, dtype=float)\n",
    "        proj = Uc @ (Uc.T @ v_coords)\n",
    "        residual = np.linalg.norm(v_coords - proj)\n",
    "        if residual <= atol + rtol * np.linalg.norm(v_coords):\n",
    "            results.append(v)\n",
    "    return results"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5bc18ca0",
   "metadata": {},
   "source": [
    "#### Extract the special vectors representing lyap[k] = V_{k+1}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "458a814a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "V_1: []\n",
      "V_2: [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",
      "V_3: [A(x_3), y_0-(x_3)]\n",
      "V_4: [A(x_4), y_0-(x_4)]\n"
     ]
    }
   ],
   "source": [
    "for k in range(len(lyap)):\n",
    "    print(f\"V_{k + 1}:\", vectors_in_column_space(lyap[k], special_vectors))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d0864e17",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "92198e08",
   "metadata": {},
   "source": [
    "### Numerically guessing the coefficients of $V_k$\n",
    "- V_{k} is represented using A(x_{k}) and y_0 - x_{k}\n",
    "- Let's find the corresponding coefficients"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "4af77df1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def decompose_rank2_symmetric(pm, S, v, w, *, sym_tol=1e-10):\n",
    "    S = np.asarray(pm.eval_scalar(S).inner_prod_coords, dtype=float)\n",
    "    v = np.asarray(pm.eval_vector(v).coords, dtype=float).ravel()\n",
    "    w = np.asarray(pm.eval_vector(w).coords, dtype=float).ravel()\n",
    "\n",
    "    n = S.shape[0]\n",
    "    if S.shape != (n, n):\n",
    "        raise ValueError(\"S must be square.\")\n",
    "    if v.shape != (n,) or w.shape != (n,):\n",
    "        raise ValueError(\"v,w must have shape (n,).\")\n",
    "\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",
    "\n",
    "    # Solve 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",
    "\n",
    "    (a, b, c), *_ = np.linalg.lstsq(X, y, rcond=None)\n",
    "    return float(a), float(b), float(c)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "1db91883",
   "metadata": {},
   "outputs": [],
   "source": [
    "precision = 3"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c71e43e0",
   "metadata": {},
   "source": [
    "$ V_{1} = \\frac{1}{2} * ||A(x_{2})||^2 - \\frac{1}{4} * \\langle A(x_{2}), y_0 - x_{2} \\rangle $"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "379d7340",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0. 0. 0.]\n"
     ]
    }
   ],
   "source": [
    "lyap_0_coeff = decompose_rank2_symmetric(pm, lyap[0], A(x[2]), y_0 - x[2])\n",
    "print(np.round(lyap_0_coeff, precision))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6c7d1be0",
   "metadata": {},
   "source": [
    "$ V_{2} = \\frac{9}{8} * ||A(x_{3})||^2 - \\frac{3}{8} * \\langle A(x_{3}), y_0 - x_{3} \\rangle $"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "bd8f58c4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[ 0.045 -0.045  0.045]\n"
     ]
    }
   ],
   "source": [
    "lyap_1_coeff = decompose_rank2_symmetric(pm, lyap[1], A(x[3]), y_0 - x[3])\n",
    "print(np.round(lyap_1_coeff, precision))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "13865f6c",
   "metadata": {},
   "source": [
    "$ V_{3} = 2 * ||A(x_{4})||^2 - \\frac{1}{2} * \\langle A(x_{3}), y_0 - x_{3} \\rangle $"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "7751fb74",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[ 0.061 -0.061  0.061]\n"
     ]
    }
   ],
   "source": [
    "lyap_2_coeff = decompose_rank2_symmetric(pm, lyap[2], A(x[4]), y_0 - x[4])\n",
    "print(np.round(lyap_2_coeff, precision))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e1c3e393",
   "metadata": {},
   "source": [
    "$$ V_{k} = \\frac{1}{8} \\Big(k^2 ||A(x_{k})||^2 - k \\langle A(x_{k}), y_0 - x_{k} \\rangle \\Big) $$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "480f6786",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2499dddc",
   "metadata": {},
   "source": [
    "### Finding the coefficient of Lyapunov function (symbolic)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "65b7d2e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "ctx_appm_lyap = pf.PEPContext(\"appm_lyap_finder\").set_as_current()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "9221b21f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle a_k*Ax_{k1}*Ax_{k1}+b_k*(y_0-y_k+Ax_{k1})*Ax_{k1}-(a_k1*Ax_{k2}*Ax_{k2}+b_k1*(y_0-(1/(k+2)*y_0+(k+1)/(k+2)*(y_k-2*Ax_{k1}))+Ax_{k2})*Ax_{k2})-2*(k+1)*(k+2)/N^2*(Ax_{k1}-Ax_{k2})*(y_k-(1/(k+2)*y_0+(k+1)/(k+2)*(y_k-2*Ax_{k1}))-Ax_{k1}+Ax_{k2})$"
      ],
      "text/plain": [
       "a_k*Ax_{k1}*Ax_{k1}+b_k*(y_0-y_k+Ax_{k1})*Ax_{k1}-(a_k1*Ax_{k2}*Ax_{k2}+b_k1*(y_0-(1/(k+2)*y_0+(k+1)/(k+2)*(y_k-2*Ax_{k1}))+Ax_{k2})*Ax_{k2})-2*(k+1)*(k+2)/N**2*(Ax_{k1}-Ax_{k2})*(y_k-(1/(k+2)*y_0+(k+1)/(k+2)*(y_k-2*Ax_{k1}))-Ax_{k1}+Ax_{k2})"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y_0 = pf.Vector(is_basis=True, tags=[\"y_0\"])\n",
    "y_k = pf.Vector(is_basis=True, tags=[\"y_k\"])\n",
    "Ax_k1 = pf.Vector(is_basis=True, tags=[\"Ax_{k1}\"])\n",
    "Ax_k2 = pf.Vector(is_basis=True, tags=[\"Ax_{k2}\"])\n",
    "k = pf.Parameter(\"k\")\n",
    "alpha = 1\n",
    "\n",
    "y_k1 = 1 / (k + 2) * y_0 + (k + 1) / (k + 2) * (y_k - 2 * Ax_k1)\n",
    "\n",
    "a_k = pf.Parameter(\"a_k\")\n",
    "a_k1 = pf.Parameter(\"a_k1\")\n",
    "b_k = pf.Parameter(\"b_k\")\n",
    "b_k1 = pf.Parameter(\"b_k1\")\n",
    "N = pf.Parameter(\"N\")\n",
    "\n",
    "V_k = a_k * Ax_k1 * Ax_k1 + b_k * (y_0 - y_k + Ax_k1) * Ax_k1\n",
    "V_k1 = a_k1 * Ax_k2 * Ax_k2 + b_k1 * (y_0 - y_k1 + Ax_k2) * Ax_k2\n",
    "\n",
    "diff = (\n",
    "    V_k\n",
    "    - V_k1\n",
    "    - 2 * (k + 1) * (k + 2) / N**2 * (Ax_k1 - Ax_k2) * (y_k - y_k1 - Ax_k1 + Ax_k2)\n",
    ")\n",
    "diff"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "1ec61868",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['y_0', 'y_k', 'Ax_{k1}', 'Ax_{k2}']"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "basis = ctx_appm_lyap.basis_vectors()\n",
    "row_index = [str(v) for v in basis]\n",
    "row_index"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "08303e8c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & y_0 & y_k & Ax_{k1} & Ax_{k2} \\\\\n",
       "        \\hline\n",
       "        y_0 & 0 & 0 & 0.5*b_k + 0.5*(2*k + 2)/N^2 & -0.5*b_k1*(1.0 - 1.0/(k + 2)) - 0.5*(2*k + 2)/N^2 \\\\y_k & 0 & 0 & -0.5*b_k - 0.5*(k + 2)*(2*k + 2)*(-1.0*(k + 1)/(k + 2) + 1.0)/N^2 & 0.5*b_k1*(k + 1)/(k + 2) + 0.5*(k + 2)*(2*k + 2)*(-1.0*(k + 1)/(k + 2) + 1.0)/N^2 \\\\Ax_{k1} & 0.5*b_k + 0.5*(2*k + 2)/N^2 & -0.5*b_k - 0.5*(k + 2)*(2*k + 2)*(-1.0*(k + 1)/(k + 2) + 1.0)/N^2 & 1.0*a_k + 1.0*b_k - 1.0*(k + 2)*(2*k + 2)*(2.0*(k + 1)/(k + 2) - 1.0)/N^2 & -1.0*b_k1*(k + 1)/(k + 2) + 0.5*(k + 2)*(2*k + 2)*(2.0*(k + 1)/(k + 2) - 1.0)/N^2 - 0.5*(k + 2)*(2*k + 2)/N^2 \\\\Ax_{k2} & -0.5*b_k1*(1.0 - 1.0/(k + 2)) - 0.5*(2*k + 2)/N^2 & 0.5*b_k1*(k + 1)/(k + 2) + 0.5*(k + 2)*(2*k + 2)*(-1.0*(k + 1)/(k + 2) + 1.0)/N^2 & -1.0*b_k1*(k + 1)/(k + 2) + 0.5*(k + 2)*(2*k + 2)*(2.0*(k + 1)/(k + 2) - 1.0)/N^2 - 0.5*(k + 2)*(2*k + 2)/N^2 & -1.0*a_k1 - 1.0*b_k1 + 1.0*(k + 2)*(2*k + 2)/N^2 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "pm_lyap = pf.ExpressionManager(\n",
    "    ctx_appm_lyap,\n",
    "    resolve_parameters={\n",
    "        \"a_k\": sp.Symbol(\"a_k\"),\n",
    "        \"a_k1\": sp.Symbol(\"a_k1\"),\n",
    "        \"b_k\": sp.Symbol(\"b_k\"),\n",
    "        \"b_k1\": sp.Symbol(\"b_k1\"),\n",
    "        \"N\": sp.Symbol(\"N\"),\n",
    "        \"k\": sp.Symbol(\"k\"),\n",
    "    },\n",
    ")\n",
    "\n",
    "diff_matrix = pm_lyap.eval_scalar(diff).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(diff_matrix, row_index, row_index, precision=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "6c8a88f5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Solutions:\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\left\\{ a_{k} : \\frac{2 \\left(k + 1\\right)^{2}}{N^{2}}, \\  a_{k1} : \\frac{2 \\left(k + 2\\right)^{2}}{N^{2}}, \\  b_{k} : - \\frac{2 \\left(k + 1\\right)}{N^{2}}, \\  b_{k1} : - \\frac{2 \\left(k + 2\\right)}{N^{2}}\\right\\}$"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import sympy as sp\n",
    "\n",
    "diff_matrix_sympify = sp.Matrix(diff_matrix)\n",
    "\n",
    "a_k, a_k1, b_k, b_k1, N, k = sp.symbols(\"a_k a_k1 b_k b_k1 N k\")\n",
    "\n",
    "unknowns = (a_k, a_k1, b_k, b_k1)\n",
    "\n",
    "eqs = list(diff_matrix_sympify)\n",
    "eqs = [e for e in eqs]\n",
    "\n",
    "sol = sp.linsolve(eqs, unknowns)\n",
    "sol_simplify = sp.factor(sp.factor(sp.nsimplify(sol)))\n",
    "\n",
    "print(\"Solutions:\")\n",
    "sol_dict = dict(zip(unknowns, next(iter(sol_simplify))))\n",
    "display(Math(sp.latex(sol_dict)))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7cd5c1ff",
   "metadata": {},
   "source": [
    "Double check we found the right solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "d667ca19",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Verification:\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\left[\\begin{matrix}0 & 0 & 0 & 0\\\\0 & 0 & 0 & 0\\\\0 & 0 & 0 & 0\\\\0 & 0 & 0 & 0\\end{matrix}\\right]$"
      ],
      "text/plain": [
       "Matrix([\n",
       "[0, 0, 0, 0],\n",
       "[0, 0, 0, 0],\n",
       "[0, 0, 0, 0],\n",
       "[0, 0, 0, 0]])"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "if sol:\n",
    "    print(\"\\nVerification:\")\n",
    "    display(sp.simplify(diff_matrix_sympify.subs(sol_dict)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8a49d6e0",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "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
}
