{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "3e23641c",
   "metadata": {},
   "source": [
    "# Proximal Gradient Method Example"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8baf79c6",
   "metadata": {},
   "source": [
    "## Import the required libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1988d6fd",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pepflow as pf\n",
    "import numpy as np\n",
    "import sympy as sp\n",
    "import matplotlib.pyplot as plt\n",
    "import itertools\n",
    "from IPython.display import display"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d4737f98-b1d1-4866-a39e-04539b329da4",
   "metadata": {},
   "source": [
    "## Define the functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "a8c5be8d",
   "metadata": {},
   "outputs": [],
   "source": [
    "L = pf.Parameter(\"L\")\n",
    "f = pf.SmoothConvexFunction(is_basis=True, tags=[\"f\"], L=L)\n",
    "g = pf.ConvexFunction(is_basis=True, tags=[\"g\"])\n",
    "h = (f + g).add_tag(\"h\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3d7b57bb",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with PGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ad6701aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_pgm(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_pgm = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    h.set_stationary_point(\"x_star\")\n",
    "    for i in range(N):\n",
    "        y = x - 1 / L * f.grad(x)\n",
    "        y.add_tag(f\"y_{i + 1}\")\n",
    "        x = g.prox(y, 1 / L, tag=f\"x_{i + 1}\")\n",
    "    return ctx_pgm"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9f1f0159",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of PGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "425348ad",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f701c3f3620>"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiwAAAGdCAYAAAAxCSikAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjMsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvZiW1igAAAAlwSFlzAAAPYQAAD2EBqD+naQAARTBJREFUeJzt3QmYjeX/x/HPWMZuENnXsmUPSWWLklZFWSq0/7UpadFGqSglLX4tKvwqS5vq14ISpRJFQiREZKcYS81Yzv/63k9nnBkzzHLmLHPer+t6nHWe88wzY87n3Pf3vu84n8/nEwAAQATLF+4DAAAAOBYCCwAAiHgEFgAAEPEILAAAIOIRWAAAQMQjsAAAgIhHYAEAABGPwAIAACJeAeUBhw4d0saNG1WiRAnFxcWF+3AAAEAm2Ny1u3fvVqVKlZQvX768H1gsrFStWjXchwEAALJh/fr1qlKlSt4PLNay4v+GS5YsGe7DAQAAmZCYmOgaHPzv43k+sPi7gSysEFgAAIgumSnnoOgWAABEPAILAACIeAQWAAAQ8fJEDQsA5JUhngcOHNDBgwfDfShA0OTPn18FChTI8bQjBBYAiADJycnatGmT9u3bF+5DAYKuaNGiqlixouLj47O9DwILAETA5Jdr1qxxn0RtAi37o84kmMgrrYbJycnatm2b+x2vXbv2MSeIywiBBQDCzP6gW2ix+SjskyiQlxQpUkQFCxbU77//7n7XCxcunK39UHQLABEiu588gVj43aaF5Sis7m3OHGnTJqliRalNGyseyvE5BwAAWZStyDNmzBjVqFHDNeu0atVK8+fPz/C5Y8eOVZs2bVS6dGm3derU6Yjn9+vXz/XXBm7nnHOOwum996QaNaQOHaTevb1Lu233AwCACA8sU6ZM0cCBAzVkyBAtXLhQTZo0UefOnbV169Z0nz979mz16tVLs2bN0ty5c10f7dlnn60NGzakep4FFKuQ92+TJk1SuFgo6d5d+uOP1PfbIdv9hBYAACI8sIwaNUrXXXedrrrqKp100kl68cUXXZHYa6+9lu7z33zzTd14441q2rSp6tWrp1deecUVl82cOTPV8woVKqQKFSqkbNYaE65uoAEDrLL5yMf89912m/c8AABixahRo3TvvfdGR2Cx6t4FCxa4bp2UHeTL525b60lm2BwD+/fvV5kyZY5oiTn++ONVt25d9e/fXzt27MhwH0lJSW6Fx8AtWKxmxd+ykqCdekAPa6yuTRVa1q/3ngcAQKxYunSpGjZsGB2BZfv27W4GxvLly6e6325v3rw5U/u4++673TwDgaHHuoP++9//ulaXxx9/XF9++aW6dOmS4WyPw4cPV0JCQspm3UzBYgW2fgdUQA9riK7VqzpO2zN8HgAAed3SaAosOTVixAhNnjxZU6dOTTUOu2fPnrrwwgvVqFEjde3aVR999JG+//571+qSnsGDB2vXrl0p23pr8ggSGw3kt1fFtVq13PWGWprh8wAAuat9+/a6zfrjI2Q/md1fsF8v2DJ7fDYB3MqVK11pR1QElrJly7qZGLds2ZLqfrttdSdH8+STT7rAMmPGDDVu3Pioz61Vq5Z7rVWrVqX7uNW7lCxZMtUWLDZ0uUoVyb/kwVJ5abKRlrhLu98adOx5AACPlQXY+8N5550X0W/E7733noYNGxaWY4pma9asyfHU+iENLHagzZs3T1Uw6y+gbd26dYZf98QTT7hfkGnTpqlFixbHfJ0//vjD1bDYyQk1m2flmWeUEk6WqFFKYPGHmNGjmY8FAAK9+uqruuWWW/TVV19p48aNEXtyrH6yRIkS4T6MqOwOatCgQViPIctdQjak2eZWmTBhgpYvX+4KZPfu3etGDZk+ffq4Lhs/q0l54IEH3Cgim7vFal1s27Nnj3vcLu+880599913Wrt2rQs/F110kU488UQ3XDocLrlEeucdqXLl1IHFWl7sfnscAKCUv+M25YW9H1gLy/jx449o7bj11lt11113ucBgLfJDhw5N9Rz7QHvGGWeoVKlSOu6443T++edr9erV6Z5iq3m059gAjEBWUnDllVe6ub2sFvKZZ55JmdvL3l/Sa3mxD932odrec6z1vlq1anr00UezfExHYytw33zzza7m0noP7D3Rulj87Puw82MDT6xcwl7TyiL87L1ztH1SDmAjbwPPYWbOsb1X23t08eLFXYPAU089FTX1K9kKLD169HDdOw8++KA7YYsWLXI/VH8h7rp169w8Kn4vvPCCG13UvXt3d4L8m+3DWBPi4sWLXQ1LnTp1dM0117hWnDlz5rhfnnCxUGK/33eM8wJLyyJLteY3H2EFANJ46623XG2DjfK84oor3AfUwDdkYx9yixUrpnnz5rmA8PDDD+uzzz5L9WZqH4h/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999/r+uvvz7ch4EsILBkR4sW3uUPP2TrywEgryhcuLAef/xx/fXXX4oGycnJ7rJcuXIqaq3liBoEluwgsACIQFZWN3u2NGmSdxmKMrtOnTqpQoUKGj58eIbPGTp0qJr6W6b/Zd0x1i2TtrvjscceU/ny5VWqVCk9/PDDOnDggO68806VKVNGVapU0bhx41LtZ/369brsssvc8+05F110kdauXXvEfh999FFVqlRJdevWTbdLaOfOnbrhhhvca1sIs9WIP/roI/fYjh071KtXL1WuXNmFnEaNGmmSneRM+vXXX12L0y+//JLq/qefflonnHCCu37w4EFdc801qlmzpooUKeKO85lnnslyt1bTpk3d+Q78vq699loX0EqWLKkzzzxTP/30U8rjdr1Dhw4qUaKEe7x58+b6IUI/jBNYchJYFi6k8BZARHjvPXsDkzp0kHr39i7ttt2fm/Lnz+9CxnPPPac//vgjR/v64osvtHHjRn311VcaNWqUhgwZovPPP1+lS5fWvHnz9H//938uVPhfZ//+/ercubN7s50zZ46++eYbFS9eXOecc05KS4qZOXOmVqxYoc8++ywlhAQ6dOiQunTp4r7+jTfe0LJly1w3l31v/pWG7Y38448/1tKlS11X0pVXXqn58+dn6vuqU6eOWrRooTfffDPV/Xa7t/2w/j0GC2Rvv/22e/0HH3xQ9957r956660cndNLL71UW7du1aeffqoFCxbo5JNPVseOHfXnn3+6xy+//HL3utZFZo/fc889KliwoCKSLw/YtWuXz74VuwyJAwd8vuLFfT47fUuXhuY1AeRZf//9t2/ZsmXuMjvefdfni4vz/iQFbnafbfZ4bujbt6/voosuctdPPfVU39VXX+2uT5061f1N9hsyZIivSZMmqb726aef9lWvXj3Vvuz2wYMHU+6rW7eur02bNim3Dxw44CtWrJhv0qRJ7vbrr7/unnPo0KGU5yQlJfmKFCnimz59esp+y5cv7+4PZK9lx2Dsufny5fOtWLEi09/7eeed57vjjjtSbrdr1843YMCADJ9vr3XCCSek3LbXsnO0fPnyDL/mpptu8nXr1i3d8532e/Cz82zn28yZM8dXsmRJ3z///JPqOXYcL730krteokQJ3/jx433h+h3Pyvs3LSzZYan75JO96xHadAYgNli3z4ABXkRJy3+fDV7J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pDzniSee0LPPPqsXX3xR8+bNU7Fixdw+//nnH0V1HQvdQgAABEWcz5o3ssBaVFq2bKnnn3/e3T506JALIbfccovuueeeY379wYMHXUuLfX2fPn1c60qlSpV0xx13aNCgQe45u3btUvny5TV+/Hj17NnzmPtMTExUQkKC+7qSJUsqYkyYIPXrJ51yijRvXriPBgCAiJKV9+8stbAkJydrwYIFrssmZQf58rnb1nqSGfv27dP+/ftVpkwZd3vNmjXavHlzqn3awVswymifSUlJ7psM3CKSv4XFhjbv3BnuowEAIGplKbBs377dtZBY60cgu22hIzPuvvtu16LiDyj+r8vKPocPH+5CjX+zFp6IVKWKVKeONUOxejMAANEySmjEiBGaPHmypk6d6gp2s2vw4MGu+ci/rV+/XhHrzDO9y5kzw30kAADERmApW7as8ufPry1btqS6325XqFDhqF/75JNPusAyY8YMNW7cOOV+/9dlZZ+FChVyfV2BW8TydwsRWAAACE1giY+PV/PmzTUz4M3Xim7tduvWrTP8OhsFNGzYME2bNk0tWrRI9VjNmjVdMAncp9Wk2Giho+0zarRv713+/LOlsHAfDQAAsdElZEOabW6VCRMmaPny5erfv7/27t2rq666yj1uI3+sy8bv8ccf1wMPPKDXXnvNzd1idSm27dmzxz0eFxen2267TY888og+/PBDLVmyxO3D6ly6du2qqGdrCjVt6l1n1lsAALKlQFa/oEePHtq2bZub6M2CR9OmTV3Lib9odt26dW7kkN8LL7zgRhd179491X5sHpehQ4e663fddZcLPddff7127typM844w+0zJ3UuEcUKjBctkmbMkHr1CvfRAACQ9+dhiUQROw+Ln3V3WWixmpyNG71J5QAAiHGJuTUPC7LpjDOkokVtDLe0eDGnEQCALCKwhEKhQoeHN0+bFpKXBAAgLyGwhMo553iXBBYAALKMwBLqwPL119Lu3SF7WQAA8gICS6iccIJ04onSgQMMbwYAIIsILKFEtxAAANlCYAlXYIn+0eQAAIQMgSXU0/THx0tr10q//hrSlwYAIJoRWEKpWDGpbVvvOqOFAADINAJLuLqFPvkk5C8NAEC0IrCE2vnne5ezZtmcxCF/eQAAohGBJdTq1pVq15b27/cWQwQAAMdEYAmHCy/0Lj/8MCwvDwBAtCGw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jb1RLVq0yLUwvfPOO+6PtNX3xEJosSXkBwwYoM8++0yFCxdWLAr8FGn1OxZgqlevrrfeekvXXHONYuHDirWwPPbYY+62tbDY34IXX3wxJgPLq6++6n4nrMUtVrz11lt68803NXHiRNcVbn8P7QOsnYNQ/w4QWHBUN998sz766CM3asqKUGONfZI88cQT3fXmzZu7T5jPPPOMK0DN66xeaevWrTr55JNT7rNPV/a78PzzzyspKcl94oolpUqVUp06dbRq1SrFgooVKx4RzuvXr693331Xseb33393xdfvvfeeYsmdd97pWll69uzpbtsoMTsXNpKUwIKIYKNibrnlFtcFMnv27JgrtDvaJ057o44FHTt2dF1igWx0gDUN33333TEXVvwFyKtXr9aVV16pWGDdwGmnM7BaBmtlijVWeGx1PP7i01ixb98+5cuXutzV/u/b38JQo4XlKH+YAj9FrVmzxjWFWdFptWrVFAvdQNYE+MEHH7i++s2bN7v7bSibjcWPBYMHD3bNv/bz3r17tzsfFt6mT5+uWGA/97Q1Szak3ebjiJVapkGDBrnRYvYGvXHjRrdSrf2x7tWrl2LB7bff7oourUvosssucyNDXn75ZbfFEntztsBiLQqxNKzd2O+/1azY30HrErJpHUaNGuUKj0POVmvGkWbNmmWz4hyx9e3bNyZOV3rfu23jxo3zxYqrr77aV716dV98fLyvXLlyvo4dO/pmzJjhi2Xt2rXzDRgwwBcrevTo4atYsaL7HahcubK7vWrVKl8s+d///udr2LChr1ChQr569er5Xn75ZV+smT59uvv7t2LFCl+sSUxMdP/nq1Wr5itcuLCvVq1avvvuu8+XlJQU8mOJs39CH5MAAAAyj3lYAABAxCOwAACAiEdgAQAAEY/AAgAAIh6BBQAARDwCCwAAiHgEFgAAEPEILAAAIOIRWAAAQMQjsAAAgIhHYAEAABGPwAIAABTp/h9jgWPHCxHq3gAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N = 8\n",
    "R = pf.Parameter(\"R\")\n",
    "L_value = 1\n",
    "R_value = 1\n",
    "\n",
    "ctx_plt = make_ctx_pgm(ctx_name=\"ctx_plt\", N=N, stepsize=1 / L)\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(h(x_k) - h(ctx_plt[\"x_star\"]))\n",
    "    result = pb_plt.solve(resolve_parameters={\"L\": L_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",
    "    L_value / (4 * cont_iters),\n",
    "    \"r-\",\n",
    "    label=\"Analytical bound $\\\\frac{L}{4k}$\",\n",
    ")\n",
    "plt.scatter(iters, opt_values, color=\"blue\", marker=\"o\", label=\"Numerical values\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "99362ee7",
   "metadata": {},
   "source": [
    "## Verification of convergence of PGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d0e252cb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.1250001921749368\n"
     ]
    }
   ],
   "source": [
    "N = sp.S(2)\n",
    "L_value = sp.S(1)\n",
    "R_value = sp.S(1)\n",
    "\n",
    "ctx_prf = make_ctx_pgm(ctx_name=\"ctx_prf\", N=N, stepsize=1 / L)\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(h(ctx_prf[f\"x_{N}\"]) - h(ctx_prf[\"x_star\"]))\n",
    "\n",
    "result = pb_prf.solve(resolve_parameters={\"L\": L_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(f)\n",
    "# Dual variables associated with the interpolations conditions of g with no relaxation\n",
    "gamm_dense = result.get_scalar_constraint_dual_value_in_numpy(g)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "45f2b9ef",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dash app running on http://127.0.0.1:8050/\n"
     ]
    }
   ],
   "source": [
    "pf.launch_primal_interactive(\n",
    "    pb_prf, ctx_prf, resolve_parameters={\"L\": L_value, \"R\": R_value}\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5530c49c",
   "metadata": {},
   "source": [
    "### Solve the problem again with the found relaxation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "24cf9cd5-0c2a-4d93-9d81-08071fe6fd4f",
   "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": "code",
   "execution_count": 8,
   "id": "1a22704f-f133-4903-b04b-939ec64ef680",
   "metadata": {},
   "outputs": [],
   "source": [
    "relaxed_constraints = []\n",
    "\n",
    "for tag_i in lamb_dense.row_names:\n",
    "    i = tag_to_index(tag_i)\n",
    "    if i == N + 1:\n",
    "        continue\n",
    "    for tag_j in lamb_dense.col_names:\n",
    "        j = tag_to_index(tag_j)\n",
    "        if i < N and i + 1 == j:\n",
    "            continue\n",
    "        relaxed_constraints.append(f\"f:{tag_i},{tag_j}\")\n",
    "\n",
    "for tag_i in gamm_dense.row_names:\n",
    "    i = tag_to_index(tag_i)\n",
    "    if i == N + 1:\n",
    "        continue\n",
    "    for tag_j in gamm_dense.col_names:\n",
    "        j = tag_to_index(tag_j)\n",
    "        if i < N and i + 1 == j:\n",
    "            continue\n",
    "        relaxed_constraints.append(f\"g:{tag_i},{tag_j}\")\n",
    "\n",
    "pb_prf.set_relaxed_constraints(relaxed_constraints)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f7b841a1",
   "metadata": {},
   "source": [
    "- Solve the PEP problem again with the relaxed constraints and store the results.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "412b66dd",
   "metadata": {},
   "outputs": [],
   "source": [
    "result = pb_prf.solve(resolve_parameters={\"L\": L_value, \"R\": R_value})\n",
    "\n",
    "# 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 f\n",
    "lamb_sol = result.get_scalar_constraint_dual_value_in_numpy(f)\n",
    "# Dual variable associated with the interpolations conditions of g\n",
    "gamm_sol = result.get_scalar_constraint_dual_value_in_numpy(g)\n",
    "# Dual variable associated with the Gram matrix G\n",
    "S_sol = result.get_gram_dual_matrix()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9d1baf45",
   "metadata": {},
   "source": [
    "### Verify closed form expression of $\\lambda$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e6907f42",
   "metadata": {},
   "source": [
    "- Print the values of $\\lambda$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "7326d189",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_0 & x_1 & x_2 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.25 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.667 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.25 & 0.417 & 0.333 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "lamb_sol.pprint()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "3075a90c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_0 & x_1 & x_2 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.25 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.667 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.25 & 0.417 & 0.333 & 0.0 \\\\\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 i == N + 1:  # Additional constraint 1 (between x_★)\n",
    "        if j == 0:\n",
    "            return lamb(\"x_0\", \"x_1\")\n",
    "        elif j < N:\n",
    "            return lamb(f\"x_{j}\", f\"x_{j + 1}\") - lamb(f\"x_{j - 1}\", f\"x_{j}\")\n",
    "        elif j == N:\n",
    "            return 1 - lamb(f\"x_{N - 1}\", f\"x_{N}\")\n",
    "    if i < N and i + 1 == j:  # Additional constraint 2 (consecutive)\n",
    "        return j / (2 * N + 1 - j)\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": "74a2e3e0",
   "metadata": {},
   "source": [
    "- Check whether our candidate of $\\lambda$ matches with solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "74d4368a",
   "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-4),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "55f644fd",
   "metadata": {},
   "source": [
    "### Verify closed form expression of $\\gamma$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ff225273",
   "metadata": {},
   "source": [
    "- Print the values of $\\gamma$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "436dc521",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccc}\n",
       "         & x_1 & x_2 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.0 & 0.333 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.333 & 0.667 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "gamm_sol.pprint()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "ef2ec4ae",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccc}\n",
       "         & x_1 & x_2 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.0 & 0.333 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.333 & 0.667 & 0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def gamm(tag_i, tag_j, N=N):\n",
    "    i = tag_to_index(tag_i)\n",
    "    j = tag_to_index(tag_j)\n",
    "    if i == N + 1:  # Additional constraint 1 (between x_★)\n",
    "        if 1 <= j <= N:\n",
    "            return 2 * N / ((2 * N + 1 - j) * (2 * N - j))\n",
    "    if i < N and i + 1 == j:  # Additional constraint 2 (consecutive)\n",
    "        return (j - 1) / (2 * N + 1 - j)\n",
    "    return 0\n",
    "\n",
    "\n",
    "gamm_cand = pf.pprint_labeled_matrix(\n",
    "    gamm, gamm_sol.row_names, gamm_sol.col_names, return_matrix=True\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "86b343c9",
   "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(gamm_cand, gamm_sol.matrix, atol=1e-4),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "db25c45d",
   "metadata": {},
   "source": [
    "### Verify closed form expression $S$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fb4c18c1",
   "metadata": {},
   "source": [
    "- Create an ExpressionManager to translate $x_i$, $f(x_i)$, and $\\nabla f(x_i)$ into a basis representation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "05ddf829",
   "metadata": {},
   "outputs": [],
   "source": [
    "pm = pf.ExpressionManager(ctx_prf, resolve_parameters={\"L\": L_value, \"R\": R_value})"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "98ba117b",
   "metadata": {},
   "source": [
    "- Print the values of $S$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "d17dee1d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_\\star) & \\nabla f(x_0) & \\nabla g(x_1) & \\nabla f(x_1) & \\nabla g(x_2) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.125 & -0.125 & 0.0 & -0.125 & -0.167 & -0.208 & -0.333 & -0.167 \\\\x_\\star & -0.125 & 0.125 & -0.0 & 0.125 & 0.167 & 0.208 & 0.333 & 0.167 \\\\\\nabla f(x_\\star) & 0.0 & -0.0 & 0.5 & -0.125 & 0.0 & -0.208 & -0.0 & -0.167 \\\\\\nabla f(x_0) & -0.125 & 0.125 & -0.125 & 0.25 & 0.167 & 0.208 & 0.333 & 0.167 \\\\\\nabla g(x_1) & -0.167 & 0.167 & 0.0 & 0.167 & 0.333 & 0.333 & 0.333 & 0.167 \\\\\\nabla f(x_1) & -0.208 & 0.208 & -0.208 & 0.208 & 0.333 & 0.667 & 0.5 & 0.167 \\\\\\nabla g(x_2) & -0.333 & 0.333 & -0.0 & 0.333 & 0.333 & 0.5 & 1.0 & 0.5 \\\\\\nabla f(x_2) & -0.167 & 0.167 & -0.167 & 0.167 & 0.167 & 0.167 & 0.5 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "S_sol.pprint()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f43a1265",
   "metadata": {},
   "source": [
    "- Subtract the decomposed closed-form expressions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "c6fad11c",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = ctx_prf.tracked_point(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "7c4c0e27",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_\\star) & \\nabla f(x_0) & \\nabla g(x_1) & \\nabla f(x_1) & \\nabla g(x_2) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.125 & -0.125 & 0.0 & -0.125 & -0.167 & -0.208 & -0.333 & -0.167 \\\\x_\\star & -0.125 & 0.125 & -0.0 & 0.125 & 0.167 & 0.208 & 0.333 & 0.167 \\\\\\nabla f(x_\\star) & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & -0.0 \\\\\\nabla f(x_0) & -0.125 & 0.125 & 0.0 & 0.156 & 0.167 & 0.177 & 0.333 & 0.167 \\\\\\nabla g(x_1) & -0.167 & 0.167 & 0.0 & 0.167 & 0.333 & 0.333 & 0.333 & 0.167 \\\\\\nabla f(x_1) & -0.208 & 0.208 & 0.0 & 0.177 & 0.333 & 0.573 & 0.5 & 0.083 \\\\\\nabla g(x_2) & -0.333 & 0.333 & -0.0 & 0.333 & 0.333 & 0.5 & 1.0 & 0.5 \\\\\\nabla f(x_2) & -0.167 & 0.167 & -0.0 & 0.167 & 0.167 & 0.083 & 0.5 & 0.417 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "S_guess1 = sum(\n",
    "    N\n",
    "    / ((2 * N + 1 - i) * (2 * N - i) * L)\n",
    "    * (\n",
    "        i / (2 * N) * f.grad(x[i])\n",
    "        + (2 * N - i) / (2 * N) * f.grad(x[i - 1])\n",
    "        - f.grad(x[N + 1])\n",
    "    )\n",
    "    ** 2\n",
    "    for i in range(1, N + 1)\n",
    ")\n",
    "\n",
    "remainder1 = S_sol.matrix - pm.eval_scalar(S_guess1).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder1, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "3915864b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_\\star) & \\nabla f(x_0) & \\nabla g(x_1) & \\nabla f(x_1) & \\nabla g(x_2) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.078 & -0.078 & 0.0 & -0.069 & -0.111 & -0.075 & -0.2 & -0.167 \\\\x_\\star & -0.078 & 0.078 & -0.0 & 0.069 & 0.111 & 0.075 & 0.2 & 0.167 \\\\\\nabla f(x_\\star) & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & -0.0 \\\\\\nabla f(x_0) & -0.069 & 0.069 & 0.0 & 0.067 & 0.078 & 0.044 & 0.2 & 0.167 \\\\\\nabla g(x_1) & -0.111 & 0.111 & 0.0 & 0.078 & 0.244 & 0.2 & 0.2 & 0.167 \\\\\\nabla f(x_1) & -0.075 & 0.075 & 0.0 & 0.044 & 0.2 & 0.173 & 0.1 & 0.083 \\\\\\nabla g(x_2) & -0.2 & 0.2 & -0.0 & 0.2 & 0.2 & 0.1 & 0.6 & 0.5 \\\\\\nabla f(x_2) & -0.167 & 0.167 & -0.0 & 0.167 & 0.167 & 0.083 & 0.5 & 0.417 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "S_guess2 = sum(\n",
    "    (\n",
    "        (i - 1) / (2 * N - i) * (2 * N + 2) / (2 * N + 1)\n",
    "        + 1 / (2 * N - i) ** 2 * 2 * N / (2 * N + 1)\n",
    "    )\n",
    "    / (2 * L)\n",
    "    * (f.grad(x[i - 1]) + g.grad(x[i]) - L / (2 * N + 1 - i) * (x[i - 1] - x[N + 1]))\n",
    "    ** 2\n",
    "    for i in range(1, N + 1)\n",
    ")\n",
    "\n",
    "remainder2 = remainder1 - pm.eval_scalar(S_guess2).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder2, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "7c78a359",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_\\star) & \\nabla f(x_0) & \\nabla g(x_1) & \\nabla f(x_1) & \\nabla g(x_2) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_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 \\\\\\nabla f(x_\\star) & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & -0.0 \\\\\\nabla f(x_0) & 0.0 & -0.0 & 0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla g(x_1) & 0.0 & -0.0 & 0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla f(x_1) & 0.0 & -0.0 & 0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla g(x_2) & -0.0 & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & -0.0 & -0.0 \\\\\\nabla f(x_2) & -0.0 & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & -0.0 & -0.0 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "S_guess3 = sum(\n",
    "    i\n",
    "    / (2 * N + 1 - i)\n",
    "    * 2\n",
    "    * N\n",
    "    / (2 * N + 1)\n",
    "    / (2 * L)\n",
    "    * (\n",
    "        (2 * N + 1) / (2 * N) * (f.grad(x[i]) + g.grad(x[i]))\n",
    "        - 1 / (2 * N) * (f.grad(x[i - 1]) + g.grad(x[i]))\n",
    "        - L / (2 * N - i) * (x[i] - x[N + 1])\n",
    "    )\n",
    "    ** 2\n",
    "    for i in range(1, N + 1)\n",
    ")\n",
    "\n",
    "remainder3 = remainder2 - pm.eval_scalar(S_guess3).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder3, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e35888e9",
   "metadata": {},
   "source": [
    "- Check whether our candidate of $S$ matches with solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "b444e21f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_\\star) & \\nabla f(x_0) & \\nabla g(x_1) & \\nabla f(x_1) & \\nabla g(x_2) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.125 & -0.125 & 0.0 & -0.125 & -0.167 & -0.208 & -0.333 & -0.167 \\\\x_\\star & -0.125 & 0.125 & 0.0 & 0.125 & 0.167 & 0.208 & 0.333 & 0.167 \\\\\\nabla f(x_\\star) & 0.0 & 0.0 & 0.5 & -0.125 & 0.0 & -0.208 & 0.0 & -0.167 \\\\\\nabla f(x_0) & -0.125 & 0.125 & -0.125 & 0.25 & 0.167 & 0.208 & 0.333 & 0.167 \\\\\\nabla g(x_1) & -0.167 & 0.167 & 0.0 & 0.167 & 0.333 & 0.333 & 0.333 & 0.167 \\\\\\nabla f(x_1) & -0.208 & 0.208 & -0.208 & 0.208 & 0.333 & 0.667 & 0.5 & 0.167 \\\\\\nabla g(x_2) & -0.333 & 0.333 & 0.0 & 0.333 & 0.333 & 0.5 & 1.0 & 0.5 \\\\\\nabla f(x_2) & -0.167 & 0.167 & -0.167 & 0.167 & 0.167 & 0.167 & 0.5 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "S_guess = S_guess1 + S_guess2 + S_guess3\n",
    "S_guess_eval = pm.eval_scalar(S_guess).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(S_guess_eval, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "60bf29ef",
   "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-4),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c0cb963e-6d97-4743-b47b-813709a3aa04",
   "metadata": {},
   "source": [
    "### Symbolic calculation"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a4e885aa-893f-448a-a62c-0dffa3a59218",
   "metadata": {},
   "source": [
    "- Assemble the RHS of the proof."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "adc58ca9-8642-4981-af3f-07d63db19498",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/4*(f(x_1)-f(x_0)+\\nabla f(x_1)*(x_0-(x_1))+1/2*L*\\|\\nabla f(x_0)-\\nabla f(x_1)\\|^2)+2/3*(f(x_2)-f(x_1)+\\nabla f(x_2)*(x_1-(x_2))+1/2*L*\\|\\nabla f(x_1)-\\nabla f(x_2)\\|^2)+1/4*(f(x_0)-f(x_\\star)+\\nabla f(x_0)*(x_\\star-x_0)+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_0)\\|^2)+5/12*(f(x_1)-f(x_\\star)+\\nabla f(x_1)*(x_\\star-(x_1))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_1)\\|^2)+1/3*(f(x_2)-f(x_\\star)+\\nabla f(x_2)*(x_\\star-(x_2))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_2)\\|^2)$"
      ],
      "text/plain": [
       "0+1/4*(f(x_1)-f(x_0)+grad_f(x_1)*(x_0-(x_1))+1/2*L*|grad_f(x_0)-grad_f(x_1)|^2)+2/3*(f(x_2)-f(x_1)+grad_f(x_2)*(x_1-(x_2))+1/2*L*|grad_f(x_1)-grad_f(x_2)|^2)+1/4*(f(x_0)-f(x_star)+grad_f(x_0)*(x_star-x_0)+1/2*L*|grad_f(x_star)-grad_f(x_0)|^2)+5/12*(f(x_1)-f(x_star)+grad_f(x_1)*(x_star-(x_1))+1/2*L*|grad_f(x_star)-grad_f(x_1)|^2)+1/3*(f(x_2)-f(x_star)+grad_f(x_2)*(x_star-(x_2))+1/2*L*|grad_f(x_star)-grad_f(x_2)|^2)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "interp_scalar_sum = pf.Scalar.zero()\n",
    "for x_i, x_j in itertools.product(ctx_prf.tracked_point(f), ctx_prf.tracked_point(f)):\n",
    "    if lamb(x_i.tag, x_j.tag) != 0:\n",
    "        interp_scalar_sum += lamb(x_i.tag, x_j.tag) * f.interp_ineq(x_i.tag, x_j.tag)\n",
    "\n",
    "display(interp_scalar_sum)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "74701ad1-da36-432d-9d2e-9cfdc8529ecf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/4*(f(x_1)-f(x_0)+\\nabla f(x_1)*(x_0-(x_1))+1/2*L*\\|\\nabla f(x_0)-\\nabla f(x_1)\\|^2)+2/3*(f(x_2)-f(x_1)+\\nabla f(x_2)*(x_1-(x_2))+1/2*L*\\|\\nabla f(x_1)-\\nabla f(x_2)\\|^2)+1/4*(f(x_0)-f(x_\\star)+\\nabla f(x_0)*(x_\\star-x_0)+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_0)\\|^2)+5/12*(f(x_1)-f(x_\\star)+\\nabla f(x_1)*(x_\\star-(x_1))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_1)\\|^2)+1/3*(f(x_2)-f(x_\\star)+\\nabla f(x_2)*(x_\\star-(x_2))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_2)\\|^2)+1/3*(g(x_2)-g(x_1)+\\nabla g(x_2)*(x_1-(x_2)))+1/3*(g(x_1)-g(x_\\star)+\\nabla g(x_1)*(x_\\star-(x_1)))+2/3*(g(x_2)-g(x_\\star)+\\nabla g(x_2)*(x_\\star-(x_2)))$"
      ],
      "text/plain": [
       "0+1/4*(f(x_1)-f(x_0)+grad_f(x_1)*(x_0-(x_1))+1/2*L*|grad_f(x_0)-grad_f(x_1)|^2)+2/3*(f(x_2)-f(x_1)+grad_f(x_2)*(x_1-(x_2))+1/2*L*|grad_f(x_1)-grad_f(x_2)|^2)+1/4*(f(x_0)-f(x_star)+grad_f(x_0)*(x_star-x_0)+1/2*L*|grad_f(x_star)-grad_f(x_0)|^2)+5/12*(f(x_1)-f(x_star)+grad_f(x_1)*(x_star-(x_1))+1/2*L*|grad_f(x_star)-grad_f(x_1)|^2)+1/3*(f(x_2)-f(x_star)+grad_f(x_2)*(x_star-(x_2))+1/2*L*|grad_f(x_star)-grad_f(x_2)|^2)+1/3*(g(x_2)-g(x_1)+grad_g(x_2)*(x_1-(x_2)))+1/3*(g(x_1)-g(x_star)+grad_g(x_1)*(x_star-(x_1)))+2/3*(g(x_2)-g(x_star)+grad_g(x_2)*(x_star-(x_2)))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for x_i, x_j in itertools.product(ctx_prf.tracked_point(g), ctx_prf.tracked_point(g)):\n",
    "    if gamm(x_i.tag, x_j.tag) != 0:\n",
    "        interp_scalar_sum += gamm(x_i.tag, x_j.tag) * g.interp_ineq(x_i.tag, x_j.tag)\n",
    "\n",
    "display(interp_scalar_sum)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "bd819c81-30f0-4e20-ad6f-6be3c6cd6dfc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/4*(f(x_1)-f(x_0)+\\nabla f(x_1)*(x_0-(x_1))+1/2*L*\\|\\nabla f(x_0)-\\nabla f(x_1)\\|^2)+2/3*(f(x_2)-f(x_1)+\\nabla f(x_2)*(x_1-(x_2))+1/2*L*\\|\\nabla f(x_1)-\\nabla f(x_2)\\|^2)+1/4*(f(x_0)-f(x_\\star)+\\nabla f(x_0)*(x_\\star-x_0)+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_0)\\|^2)+5/12*(f(x_1)-f(x_\\star)+\\nabla f(x_1)*(x_\\star-(x_1))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_1)\\|^2)+1/3*(f(x_2)-f(x_\\star)+\\nabla f(x_2)*(x_\\star-(x_2))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_2)\\|^2)+1/3*(g(x_2)-g(x_1)+\\nabla g(x_2)*(x_1-(x_2)))+1/3*(g(x_1)-g(x_\\star)+\\nabla g(x_1)*(x_\\star-(x_1)))+2/3*(g(x_2)-g(x_\\star)+\\nabla g(x_2)*(x_\\star-(x_2)))-(0+2/12*L*\\|1/4*\\nabla f(x_1)+3/4*\\nabla f(x_0)-\\nabla f(x_\\star)\\|^2+2/6*L*\\|1/2*\\nabla f(x_2)+1/2*\\nabla f(x_1)-\\nabla f(x_\\star)\\|^2+0+4/45/2*L*\\|\\nabla f(x_0)+\\nabla g(x_1)-L/4*(x_0-x_\\star)\\|^2+4/5/2*L*\\|\\nabla f(x_1)+\\nabla g(x_2)-L/3*(x_1-x_\\star)\\|^2+0+1/5/2*L*\\|5/4*(\\nabla f(x_1)+\\nabla g(x_1))-1/4*(\\nabla f(x_0)+\\nabla g(x_1))-L/3*(x_1-x_\\star)\\|^2+8/15/2*L*\\|5/4*(\\nabla f(x_2)+\\nabla g(x_2))-1/4*(\\nabla f(x_1)+\\nabla g(x_2))-L/2*(x_2-x_\\star)\\|^2)$"
      ],
      "text/plain": [
       "0+1/4*(f(x_1)-f(x_0)+grad_f(x_1)*(x_0-(x_1))+1/2*L*|grad_f(x_0)-grad_f(x_1)|^2)+2/3*(f(x_2)-f(x_1)+grad_f(x_2)*(x_1-(x_2))+1/2*L*|grad_f(x_1)-grad_f(x_2)|^2)+1/4*(f(x_0)-f(x_star)+grad_f(x_0)*(x_star-x_0)+1/2*L*|grad_f(x_star)-grad_f(x_0)|^2)+5/12*(f(x_1)-f(x_star)+grad_f(x_1)*(x_star-(x_1))+1/2*L*|grad_f(x_star)-grad_f(x_1)|^2)+1/3*(f(x_2)-f(x_star)+grad_f(x_2)*(x_star-(x_2))+1/2*L*|grad_f(x_star)-grad_f(x_2)|^2)+1/3*(g(x_2)-g(x_1)+grad_g(x_2)*(x_1-(x_2)))+1/3*(g(x_1)-g(x_star)+grad_g(x_1)*(x_star-(x_1)))+2/3*(g(x_2)-g(x_star)+grad_g(x_2)*(x_star-(x_2)))-(0+2/12*L*|1/4*grad_f(x_1)+3/4*grad_f(x_0)-grad_f(x_star)|^2+2/6*L*|1/2*grad_f(x_2)+1/2*grad_f(x_1)-grad_f(x_star)|^2+0+4/45/2*L*|grad_f(x_0)+grad_g(x_1)-L/4*(x_0-x_star)|^2+4/5/2*L*|grad_f(x_1)+grad_g(x_2)-L/3*(x_1-x_star)|^2+0+1/5/2*L*|5/4*(grad_f(x_1)+grad_g(x_1))-1/4*(grad_f(x_0)+grad_g(x_1))-L/3*(x_1-x_star)|^2+8/15/2*L*|5/4*(grad_f(x_2)+grad_g(x_2))-1/4*(grad_f(x_1)+grad_g(x_2))-L/2*(x_2-x_star)|^2)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "RHS = interp_scalar_sum - S_guess\n",
    "display(RHS)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "23459a7d-a899-46dc-9333-e2b347cf09a7",
   "metadata": {},
   "source": [
    "- Assemble the LHS of the proof"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "a95f8b25-8ef3-4ed1-b907-3fbe8d5375fc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_2)+g(x_2)-(f(x_\\star)+g(x_\\star))-L/8*\\|x_0-x_\\star\\|^2$"
      ],
      "text/plain": [
       "f(x_2)+g(x_2)-(f(x_star)+g(x_star))-L/8*|x_0-x_star|^2"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "LHS = h(x[N]) - h(ctx_prf[\"x_star\"]) - L / (4 * N) * (x[0] - ctx_prf[\"x_star\"]) ** 2\n",
    "display(LHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "f501f6a9-77b5-4711-8a10-4a168d0dd82d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_2)+g(x_2)-(f(x_\\star)+g(x_\\star))-L/8*\\|x_0-x_\\star\\|^2-(0+1/4*(f(x_1)-f(x_0)+\\nabla f(x_1)*(x_0-(x_1))+1/2*L*\\|\\nabla f(x_0)-\\nabla f(x_1)\\|^2)+2/3*(f(x_2)-f(x_1)+\\nabla f(x_2)*(x_1-(x_2))+1/2*L*\\|\\nabla f(x_1)-\\nabla f(x_2)\\|^2)+1/4*(f(x_0)-f(x_\\star)+\\nabla f(x_0)*(x_\\star-x_0)+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_0)\\|^2)+5/12*(f(x_1)-f(x_\\star)+\\nabla f(x_1)*(x_\\star-(x_1))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_1)\\|^2)+1/3*(f(x_2)-f(x_\\star)+\\nabla f(x_2)*(x_\\star-(x_2))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_2)\\|^2)+1/3*(g(x_2)-g(x_1)+\\nabla g(x_2)*(x_1-(x_2)))+1/3*(g(x_1)-g(x_\\star)+\\nabla g(x_1)*(x_\\star-(x_1)))+2/3*(g(x_2)-g(x_\\star)+\\nabla g(x_2)*(x_\\star-(x_2)))-(0+2/12*L*\\|1/4*\\nabla f(x_1)+3/4*\\nabla f(x_0)-\\nabla f(x_\\star)\\|^2+2/6*L*\\|1/2*\\nabla f(x_2)+1/2*\\nabla f(x_1)-\\nabla f(x_\\star)\\|^2+0+4/45/2*L*\\|\\nabla f(x_0)+\\nabla g(x_1)-L/4*(x_0-x_\\star)\\|^2+4/5/2*L*\\|\\nabla f(x_1)+\\nabla g(x_2)-L/3*(x_1-x_\\star)\\|^2+0+1/5/2*L*\\|5/4*(\\nabla f(x_1)+\\nabla g(x_1))-1/4*(\\nabla f(x_0)+\\nabla g(x_1))-L/3*(x_1-x_\\star)\\|^2+8/15/2*L*\\|5/4*(\\nabla f(x_2)+\\nabla g(x_2))-1/4*(\\nabla f(x_1)+\\nabla g(x_2))-L/2*(x_2-x_\\star)\\|^2))$"
      ],
      "text/plain": [
       "f(x_2)+g(x_2)-(f(x_star)+g(x_star))-L/8*|x_0-x_star|^2-(0+1/4*(f(x_1)-f(x_0)+grad_f(x_1)*(x_0-(x_1))+1/2*L*|grad_f(x_0)-grad_f(x_1)|^2)+2/3*(f(x_2)-f(x_1)+grad_f(x_2)*(x_1-(x_2))+1/2*L*|grad_f(x_1)-grad_f(x_2)|^2)+1/4*(f(x_0)-f(x_star)+grad_f(x_0)*(x_star-x_0)+1/2*L*|grad_f(x_star)-grad_f(x_0)|^2)+5/12*(f(x_1)-f(x_star)+grad_f(x_1)*(x_star-(x_1))+1/2*L*|grad_f(x_star)-grad_f(x_1)|^2)+1/3*(f(x_2)-f(x_star)+grad_f(x_2)*(x_star-(x_2))+1/2*L*|grad_f(x_star)-grad_f(x_2)|^2)+1/3*(g(x_2)-g(x_1)+grad_g(x_2)*(x_1-(x_2)))+1/3*(g(x_1)-g(x_star)+grad_g(x_1)*(x_star-(x_1)))+2/3*(g(x_2)-g(x_star)+grad_g(x_2)*(x_star-(x_2)))-(0+2/12*L*|1/4*grad_f(x_1)+3/4*grad_f(x_0)-grad_f(x_star)|^2+2/6*L*|1/2*grad_f(x_2)+1/2*grad_f(x_1)-grad_f(x_star)|^2+0+4/45/2*L*|grad_f(x_0)+grad_g(x_1)-L/4*(x_0-x_star)|^2+4/5/2*L*|grad_f(x_1)+grad_g(x_2)-L/3*(x_1-x_star)|^2+0+1/5/2*L*|5/4*(grad_f(x_1)+grad_g(x_1))-1/4*(grad_f(x_0)+grad_g(x_1))-L/3*(x_1-x_star)|^2+8/15/2*L*|5/4*(grad_f(x_2)+grad_g(x_2))-1/4*(grad_f(x_1)+grad_g(x_2))-L/2*(x_2-x_star)|^2))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "diff = LHS - RHS\n",
    "display(diff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "70da9aad-963c-412f-8de4-200794afd2ad",
   "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",
    "    diff.repr_by_basis(ctx_prf, sympy_mode=True, resolve_parameters={\"L\": sp.S(\"L\")})\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "32920422",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "594c0bc6",
   "metadata": {},
   "source": [
    "#### With verified closed form expression, we have verified the below for fixed $N$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "04bafdc8",
   "metadata": {},
   "source": [
    "\\begin{align*}\n",
    "    &f(x_N) + g(x_N) - f(x_\\star) - g(x_\\star) - \\frac{L}{4N} \\|x_0 - x_\\star\\|^2   \\\\\n",
    "    &= \\sum_{i=1}^{N} \\frac{i}{2N + 1 - i} \n",
    "\\left( \n",
    "    f(x_i) - f(x_{i-1}) \n",
    "    + \\langle \\nabla f(x_{i-1}), x_{i-1} - x_i \\rangle \n",
    "    + \\frac{1}{2L} \\|\\nabla f(x_i) - \\nabla f(x_{i-1})\\|^2 \n",
    "\\right)\n",
    "  \\\\ \n",
    "    &\\quad + \\frac{1}{N+1} \n",
    "\\left( \n",
    "    f(x_N) - f(x_\\star) \n",
    "    + \\langle \\nabla f(x_N), x_\\star - x_N \\rangle \n",
    "    + \\frac{1}{2L} \\|\\nabla f(x_N) - \\nabla f(x_\\star)\\|^2 \n",
    "\\right)\n",
    " \\\\\n",
    "    &\\quad + \\sum_{i=1}^{N-1} \n",
    "\\frac{2N + 1}{(2N + 1 - i)(2N - i)} \n",
    "\\left(\n",
    "    f(x_i) - f(x_\\star)\n",
    "    + \\langle \\nabla f(x_i), x_\\star - x_i \\rangle\n",
    "    + \\frac{1}{2L} \\|\\nabla f(x_i) - \\nabla f(x_\\star)\\|^2\n",
    "\\right)\n",
    " \\\\\n",
    "     &\\quad + \\frac{1}{2N}\n",
    "\\left(\n",
    "    f(x_0) - f(x_\\star)\n",
    "    + \\langle \\nabla f(x_0), x_\\star - x_0 \\rangle\n",
    "    + \\frac{1}{2L} \\|\\nabla f(x_0) - \\nabla f(x_\\star)\\|^2\n",
    "\\right)\n",
    " \\\\\n",
    "     &\\quad\n",
    "     + \\sum_{i=1}^{N}\n",
    "\\frac{i - 1}{2N + 1 - i}\n",
    "\\left(\n",
    "    g(x_i) - g(x_{i-1})\n",
    "    + \\langle \\widetilde{\\nabla} g(x_i), x_{i-1} - x_i \\rangle\n",
    "\\right)\n",
    "      \\\\\n",
    "     &\\quad\n",
    "     + \\sum_{i=1}^{N}\n",
    "\\frac{2N}{(2N + 1 - i)(2N - i)}\n",
    "\\left(\n",
    "    g(x_i) - g(x_\\star)\n",
    "    + \\langle \\widetilde{\\nabla} g(x_i), x_\\star - x_i \\rangle\n",
    "\\right)\n",
    "      \\\\\n",
    "     &\\quad - \\sum_{i=1}^{N} \\frac{2N}{(2N+1-i)(2N-i)} \\frac{1}{2L} \n",
    "\\left\\| \n",
    "\\nabla f(x_*) - \\frac{i}{2N} \\nabla f(x_i) - \\frac{2N-i}{2N} \\nabla f(x_{i-1})\n",
    "\\right\\|^2 \\\\\n",
    "&\\quad - \\sum_{i=1}^{N} \\frac{i}{2N+1-i} \\frac{2N}{2N+1} \\frac{1}{2L} \n",
    "\\left\\|\n",
    "\\frac{2N+1}{2N} (\\nabla f(x_i) + \\widetilde{\\nabla} g(x_i)) - \\frac{1}{2N} (\\nabla f(x_{i-1}) + \\widetilde{\\nabla} g(x_i)) - \\frac{L}{2N-i} (x_i - x_*)\n",
    "\\right\\|^2 \\\\\n",
    "&\\quad - \\sum_{i=1}^{N} \\left(\n",
    "\\frac{i-1}{2N-i} \\frac{2N+2}{2N+1} + \\frac{1}{(2N-i)^2} \\frac{2N}{2N+1}\n",
    "\\right) \\frac{1}{2L} \n",
    "\\left\\|\n",
    "\\nabla f(x_{i-1}) + \\widetilde{\\nabla} g(x_i) - \\frac{L}{2N+1-i} (x_{i-1}-x_*)\n",
    "\\right\\|^2\n",
    "\\end{align*}"
   ]
  }
 ],
 "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
