{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "88a7ab35-6e71-42c7-940c-b6a7d48480fa",
   "metadata": {},
   "source": [
    "# Optimal Gradient Method Example\n",
    "\n",
    "\n",
    "This code tests Optimized Gradient Method (OGM) which is the exact optimal method\n",
    "that reduces function value respect to initial distance to solution for L-smooth\n",
    "convex functions. Introduced in \"Optimized first-order methods for smooth convex\n",
    "minimization\" by Donghwan Kim, Jeffrey A Fessler (2016)."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "715ea76a-6c5b-46cc-b2ee-ebff20d907b5",
   "metadata": {},
   "source": [
    "## Import the required libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7d69b7b5-f8e8-4ff3-ab69-dff43e5948bf",
   "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",
    "import functools\n",
    "from IPython.display import display"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "99a2034b",
   "metadata": {},
   "source": [
    "## Define the functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "47ccad84",
   "metadata": {},
   "outputs": [],
   "source": [
    "L = pf.Parameter(\"L\")\n",
    "f = pf.SmoothConvexFunction(is_basis=True, tags=[\"f\"], L=L)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b4d33270-5aa6-49a8-bf67-4a23bb57ed25",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with OGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "7dd141b3-c393-4dfe-947d-7b32ff442645",
   "metadata": {},
   "outputs": [],
   "source": [
    "@functools.cache\n",
    "def theta(i, N):\n",
    "    if i == -1:\n",
    "        return 0\n",
    "    if i == N:\n",
    "        return 1 / sp.S(2) * (sp.S(1) + sp.sqrt(8 * theta(N - 1, N) ** 2 + sp.S(1)))\n",
    "    return 1 / sp.S(2) * (sp.S(1) + sp.sqrt(4 * theta(i - 1, N) ** 2 + sp.S(1)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "8bd6c27d-fba0-40bc-b2b0-4d1eac5df268",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_ogm(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_ogm = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    z = x\n",
    "    f.set_stationary_point(\"x_star\")\n",
    "    for i in range(N):\n",
    "        y = x - stepsize * f.grad(x)\n",
    "        z = z - 2 * stepsize * theta(i, N) * f.grad(x)\n",
    "        z.add_tag(f\"z_{i + 1}\")\n",
    "        x = (1 - 1 / theta(i + 1, N)) * y + 1 / theta(i + 1, N) * z\n",
    "        x.add_tag(f\"x_{i + 1}\")\n",
    "    return ctx_ogm"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "651aec8b-c801-4381-8d8e-d161ed60ee65",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of OGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "c00faeef-27d0-463d-a6be-2d67af44ca5a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x71cfdb935210>"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiwAAAGdCAYAAAAxCSikAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjMsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvZiW1igAAAAlwSFlzAAAPYQAAD2EBqD+naQAAOB5JREFUeJzt3Xl8FPX9x/H3ZkMOjgS5cpBI5AY5IqdBMIj5GS36I4IWEbmKKMhpKgjI0eqvBhEkFGlRakWtCKJALSpIwUAKKBBAzoIcSoQcSGnCERPYnd8fSxY2JCEbckyS1/Px2AfszHdmPjOPSfad78x812IYhiEAAAAT8yjvAgAAAG6GwAIAAEyPwAIAAEyPwAIAAEyPwAIAAEyPwAIAAEyPwAIAAEyPwAIAAEzPs7wLKAl2u12nT59WrVq1ZLFYyrscAABQBIZh6Pz58woODpaHR+F9KJUisJw+fVqhoaHlXQYAACiG5ORkhYSEFNqmUgSWWrVqSXLssJ+fXzlXAwAAiiIzM1OhoaHOz/HCVIrAknsZyM/Pj8ACAEAFU5TbObjpFgAAmB6BBQAAmB6BBQAAmF6luIcFAMzEMAxduXJFNputvEsByp3VapWnp+ctDztCYAGAEpSTk6OUlBRdunSpvEsBTKN69eoKCgqSl5dXsddBYLleVpbk61t67QFUana7XSdOnJDValVwcLC8vLwYzBJVmmEYysnJ0ZkzZ3TixAk1a9bspgPEFYTAkmvxYmn2bGnjRqkog9AlJ0u9ekmTJkkjRpR+fQBMLycnR3a7XaGhoapevXp5lwOYgq+vr6pVq6Yff/xROTk58vHxKdZ6uOlWcvSUzJ4tHT0q9ezpCCOFSU52tDt61LFcVlZZVAmggijuX5BAZVUSPxP8VEmOyzobN0qNG0vHjztDi80mJSRIH33k+Ndm07Wwcvy4o/3GjVwWAgCglHFJKFdoqCOVXA0jKzu/qvEeC/RTyrVDFBJ0RfPtr6pv2tWwkpBQtMtHAADgltDDcr2roWVlwEg9lrZQP6W4Hp5TKR56LG2hVgaMJKwAAFCGCCx52IJDNd5jgQxJeQ+PcfX9BI8FsgUTVgAAlcsbb7yhqVOnlncZ+SKw5JGYqKuXgfI/NIY8lJziqcTEsq0LAIDStn//frVp06a8y8gXgSWPlJSSbQcAQEVBYKlAgoJKth0A3JS7QyNU4KEUevbsqQkTJphmPUVdX0lvr6SVRH2GYej7779Xy5YtS6aoEkZgyaNHD8fTQBbZ851vkV2hQVfUo0cZFwagclq8WGrX7ubjP+VKTna0X7y4VMrZtm2brFarevfuXSrrd1dBH8QrV67UK6+8UvYFVWInTpy45eHzSxOBJQ/r6WTNt4+VpBtCS+77ePtYWU8X8ZcLABTEhINWvvPOOxo7dqw2b96s06dPl/j6S0qdOnVUq1at8i6jUtm/f7/uvPPO8i6jQASW6139ZdA3bZE+CRithkGugSUkyK5PAkarb9qiov1yAYDCFDBoZb7KYNDKCxcuaPny5Ro1apR69+6tJUuWuMzv2bOnxo0bp0mTJqlOnToKDAzU7373O5c2a9euVffu3VW7dm3VrVtXDz/8sI4dO5bv9t5//33VrVtX2dnZLtNjYmI0aNAgDR06VJs2bdL8+fNlsVhksVj0ww8/OGu5vufFbrdr9uzZatq0qby9vXX77bfrD3/4g9s1FebKlSsaM2aM/P39Va9ePU2fPl2GYTjnZ2dna9y4cWrQoIF8fHzUvXt37dixwzk/LCxM8fHxLusMDw93OYZFOcYXL17U4MGDVbNmTQUFBWnu3Llu70t+zHz/ikRguSbPL4O+O6bqh2RPff21tHSp9PXX0olkT/XdMbVov1wAoChyB60s7PdK3rBSSuNAffzxx2rZsqVatGihp556Sn/9619dPpAl6b333lONGjX07bffavbs2Xr55Ze1fv165/yLFy8qNjZWO3fu1IYNG+Th4aFHH31UdvuNl9kff/xx2Ww2ffbZZ85p6enp+vzzz/Wb3/xG8+fPV0REhEaMGKGUlBSlpKQotID9njJlimbNmqXp06fr4MGDWrp0qQICAtyuqTDvvfeePD09tX37ds2fP19vvPGG/vKXvzjnT5o0SZ9++qnee+897dq1S02bNlV0dLT+85//uL2dwo7xxIkTtWnTJv3973/XV199pYSEBO3atcutbeRn//79WrBggcLCwhQWFqbHH3/8ltdZooxKICMjw5BkZGRkFG8Fly4ZRtOmhiEZRuPGhnHyZOHtT550tJMcy126VLztAqhUsrKyjIMHDxpZWVnuL3z975Xrfw8VNL0UdOvWzYiPjzcMwzAuX75s1KtXz/j666+d8yMjI43u3bu7LNO5c2fjxRdfLHCdZ86cMSQZ+/btc65j/PjxzvmjRo0yHnroIef7uXPnGo0bNzbsdnu+7a+vJXd6Zmam4e3tbSxevLhI+5m3psK2c/38Vq1aOesyDMN48cUXjVatWhmGYRgXLlwwqlWrZnz44YfO+Tk5OUZwcLAxe/ZswzAMo1GjRsa8efNc1tu+fXtj5syZLtsp7BifP3/e8PLyMj7++GPn/LNnzxq+vr6F1l/eCvrZcOfzmx4WydGtOmmS1LRp0f5yyf2LqGlTx3J8lxCAW5VfT8vWrWXSsyJJhw8f1vbt2zVgwABJkqenp/r376933nnHpV27du1c3gcFBSk9Pd35/vvvv9eAAQPUuHFj+fn5KSwsTJJ08uTJfLc7YsQIffXVVzp16pQkacmSJRo6dKgsFkuRaz906JCys7N1//335zvf3ZoKcvfdd7vUFRERoe+//142m03Hjh3T5cuXdc899zjnV6tWTV26dNGhQ4fc2k5hx/jYsWPKyclR165dnfPr1KmjFi1auLWNiojvEso1YoT01FNFDx+hodLevYQVACUnz3eaKffDrwy+u+ydd97RlStXFBwc7JxmGIa8vb315ptvyt/fX5LjQ/h6FovF5dLKI488okaNGmnx4sUKDg6W3W5XmzZtlJOTk+9277rrLrVv317vv/++HnjgAR04cECff/65W7X73uT3sLs1lRYPD48bLrFdvnz5hnY3O8bucif8lYS8+1hS6GG5nrvhg7ACoKSFhkoffOA67YMPSjWsXLlyRe+//77mzp2rPXv2OF/fffedgoOD9dFHHxVpPWfPntXhw4c1bdo03X///WrVqpXOnTt30+WefvppLVmyRO+++66ioqJc7lPx8vKSzWYrdPlmzZrJ19dXGzZsKLGa8vPtt9+6vP/mm2/UrFkzWa1WNWnSRF5eXtqyZYtz/uXLl7Vjxw61bt1aklS/fn2lXDfqaGZmpk6cOOFWDU2aNFG1atVcajl37pyOHDlS4DKGYZTpq7TQwwIAZpKcLA0a5Dpt0KBS7WFZs2aNzp07p+HDhzt7UnL169dP77zzjkaOHHnT9dx2222qW7eu3n77bQUFBenkyZOaPHnyTZd78skn9cILL2jx4sV6//33XeaFhYXp22+/1Q8//KCaNWuqTp068vBw/Vvbx8dHL774oiZNmiQvLy/dc889OnPmjA4cOKBhw4YVq6b8nDx5UrGxsXr22We1a9cuLViwwPmETo0aNTRq1ChNnDhRderU0e23367Zs2fr0qVLGj58uCSpV69eWrJkiR555BHVrl1bM2bMkNVqdauGmjVravjw4Zo4caLq1q2rBg0a6KWXXrrhmJSFFStWaP369Tp37pxmzpxZ6k8Y0cMCAGaR92mgLVvK5KnEd955R1FRUTeEFckRWHbu3Km9e/fedD0eHh5atmyZkpKS1KZNGz3//PN6/fXXb7qcv7+/+vXrp5o1ayomJsZl3gsvvCCr1arWrVurfv36Bd53Mn36dP32t7/VjBkz1KpVK/Xv31/p6enFrik/gwcPVlZWlrp06aLRo0dr/PjxeuaZZ5zzZ82apX79+mnQoEHq0KGDjh49qnXr1um2226T5HiSKTIyUg8//LB69+6tmJgYNWnSxO06Xn/9dfXo0UOPPPKIoqKi1L17d3Xs2LHA9snJyerZs6dat26tdu3aacWKFS7zc3tFch+fzttLkpOToxEjRqh169aKiIhwPvX0+OOP6+2339ZLL72kL774wu39cJfFKM3+mzKSmZkpf39/ZWRkyM/Pr7zLAVBF/fLLLzpx4oTuuOMO+fj4uLdwQY8ul9EjzeXt/vvv15133qk//vGP5V1KpZOSkqK0tDSFh4crNTVVHTt21JEjR1SjRg1J0p/+9Cd5enrq+++/l9Vq1UMPPaTIyEjn8i+99JKaN2+uIUOGaNq0aQoODtZzzz0nyTH+zciRIzVjxgyFhIQUWENBPxvufH4Xq4dl4cKFCgsLk4+Pj7p27art27cX2PbAgQPq16+fwsLCZLFYbhg0R5Li4uLUuXNn1apVSw0aNFBMTIwOHz5cnNIAoOIpLJQUZZyWCuzcuXNatWqVEhISNHr06PIup1IKCgpSeHi4JCkwMFD16tVzGRvmueeeU0ZGhv74xz/qkUcecQkrGRkZ2rx5s4YMGSJJuuOOO3T8+HFJjrDy/PPPa9SoUYWGlZLidmBZvny5YmNjNXPmTO3atUvt27dXdHS0y2Nt17t06ZIaN26sWbNmKTAwMN82mzZt0ujRo/XNN99o/fr1unz5sh544AFdvHjR3fIAoGIpSg9KJQ4td911l4YOHarXXnutSjyaW96SkpJks9lcbmxetGiR/P39NW7cOP3jH/9QYmKic94///lPHT16VOHh4QoPD9e0adNUp04dSdKcOXO0Y8cOvfXWW/r0009Lv3h3B3/p0qWLMXr0aOd7m81mBAcHG3FxcTddNr9Bc/KTnp5uSDI2bdpUpJpueeA4ACgBbg8cx6CVKENnz541WrdubWzZssVleu5geLkD2F0/ON706dON999/3/n+wQcfNNasWeP2tst84LicnBwlJSUpKirKOc3Dw0NRUVHatm1biYWojIwMSXKmuLyys7OVmZnp8gKACodBK1FGsrOzFRMTo8mTJ6tbt24u83LHacm96fb6cVv++9//ytvbW5LjfpPdu3frvvvuk2EYqlevnvbs2SNJeuKJJ0p9H9wKLD///LNsNpvz+xlyBQQEKDU1tUQKstvtmjBhgu65554CH5GKi4uTv7+/81XQd0sAgOmNGOEYhLKov8dyB60cMaJ060KlYRiGhg4dql69emlQ3kfmb6Jp06bO+1RffvlljRs3TtWrV9fRo0f12GOP6e9//7vOnj2revXqlUbpLkz3WPPo0aO1f/9+LVu2rMA2U6ZMUUZGhvOVXEmu5QKoohi0EqVoy5YtWr58uVavXu28F2Xfvn1FWvbJJ59UQkKCmjVrpuzsbL344ouSpN27d+tXv/qVTp8+rd27d+uuu+4qzV2Q5ObAcfXq1ZPValVaWprL9LS0tAJvqHXHmDFjtGbNGm3evLnQO469vb2dXVQAAKBg3bt3L/bQ/vXq1dPOnTtvmL57924999xzOnz4sD7++OMiDSx4q9zqYfHy8lLHjh1dhj+22+3asGGDIiIiil2EYRgaM2aMVq1apY0bN+qOO+4o9roAAEDpSk5OVmhoqB599FEtXbq01Ee5lYoxNH9sbKyGDBmiTp06qUuXLoqPj9fFixc1bNgwSY6RABs2bKi4uDhJjht1Dx486Pz/qVOntGfPHtWsWVNNmzaV5LgMtHTpUv39739XrVq1nPfD+Pv73/RLrQAAQNn629/+Jslxj8uFCxfKZJtuB5b+/fvrzJkzmjFjhlJTUxUeHq61a9c6b8Q9efKky3canD592uXa1pw5czRnzhxFRkYqISFBkvTnP/9ZktSzZ0+Xbb377rsaOnSouyUCAIBKhqH5AaCE3NLQ/EAlVm5D8wMAAJQlAgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAoMIKCwtTfHx8ia2vZ8+emjBhQomtLz9Dhw5VTExMqW6jMiKwAEAVN3ToUFksFs2aNctl+urVq2WxWMqpqqLZsWOHnnnmmfIuA2WAwAIAJmOzSQkJ0kcfOf612Up/mz4+Pnrttdd07ty50t9YCcjJyZEk1a9fX9WrVy/nalAWCCwAYCIrV0phYdJ990lPPun4NyzMMb00RUVFKTAwUHFxcQW2+d3vfqfw8HCXafHx8QoLC3O+z73c8eqrryogIEC1a9fWyy+/rCtXrmjixImqU6eOQkJC9O6777qsJzk5Wb/+9a9Vu3Zt1alTR3369NEPP/xww3r/8Ic/KDg4WC1atJB04yWh//73v3r22WcVEBAgHx8ftWnTRmvWrJEknT17VgMGDFDDhg1VvXp1tW3bVh999FGRj9GRI0dksVj073//22X6vHnz1KRJE0mSzWbT8OHDdccdd8jX11ctWrTQ/PnzC11vfpe1wsPD9bvf/c5lv55++mnVr19ffn5+6tWrl7777jvn/O+++0733XefatWqJT8/P3Xs2FE7d+4s8r5VBAQWADCJlSulxx6TfvrJdfqpU47ppRlarFarXn31VS1YsEA/5S3ATRs3btTp06e1efNmvfHGG5o5c6Yefvhh3Xbbbfr22281cuRIPfvss87tXL58WdHR0apVq5YSExO1ZcsW1axZUw8++KCzJ0WSNmzYoMOHD2v9+vXOEHI9u92uhx56SFu2bNHf/vY3HTx4ULNmzZLVapXk+AK+jh076vPPP9f+/fv1zDPPaNCgQdq+fXuR9qt58+bq1KmTPvzwQ5fpH374oZ588klnDSEhIVqxYoUOHjyoGTNmaOrUqfr444+LdSxzPf7440pPT9eXX36ppKQkdejQQffff7/+85//SJIGDhyokJAQ7dixQ0lJSZo8ebKqVat2S9s0HaMSyMjIMCQZGRkZ5V0KgCosKyvLOHjwoJGVleX2sleuGEZIiGFI+b8sFsMIDXW0K2lDhgwx+vTpYxiGYdx9993Gb37zG8MwDGPVqlXG9R8TM2fONNq3b++y7Lx584xGjRq5rKtRo0aGzWZzTmvRooXRo0eP6/b1ilGjRg3jo48+MgzDMD744AOjRYsWht1ud7bJzs42fH19jXXr1jnXGxAQYGRnZ7tsv1GjRsa8efMMwzCMdevWGR4eHsbhw4eLvO+9e/c2fvvb3zrfR0ZGGuPHjy+w/bx584wmTZo43x8+fNiQZBw6dKjAZUaPHm3069fP+f764513H3K1b9/emDlzpmEYhpGYmGj4+fkZv/zyi0ubJk2aGG+99ZZhGIZRq1YtY8mSJQXWUN4K+tlw5/ObHhYAMIHExBt7Vq5nGFJysqNdaXrttdf03nvv6dChQ8Vex5133ikPj2sfLwEBAWrbtq3zvdVqVd26dZWeni7JcTnj6NGjqlWrlmrWrKmaNWuqTp06+uWXX3Ts2DHncm3btpWXl1eB292zZ49CQkLUvHnzfOfbbDa98soratu2rerUqaOaNWtq3bp1OnnyZJH37YknntAPP/ygb775RpKjd6VDhw5q2bKls83ChQvVsWNH1a9fXzVr1tTbb7/t1jby+u6773ThwgXVrVvXeXxq1qypEydOOI9PbGysnn76aUVFRWnWrFkux62y8CzvAgAAUkpKybYrrnvvvVfR0dGaMmWKhg4d6jLPw8NDhmG4TLt8+fIN68h7KcJiseQ7zW63S5IuXLigjh073nCpRXLcVJurRo0ahdbu6+tb6PzXX39d8+fPV3x8vNq2basaNWpowoQJLpedbiYwMFC9evXS0qVLdffdd2vp0qUaNWqUc/6yZcv0wgsvaO7cuYqIiFCtWrX0+uuv69tvvy1wnTc7rhcuXFBQUJASEhJuWLZ27dqSHPcXPfnkk/r888/15ZdfaubMmVq2bJkeffTRIu+b2RFYAMAEgoJKtt2tmDVrlsLDw503tuaqX7++UlNTZRiG83HnPXv23PL2OnTooOXLl6tBgwby8/Mr9nratWunn376SUeOHMm3l2XLli3q06ePnnrqKUmO+02OHDmi1q1bu7WdgQMHatKkSRowYICOHz+uJ554wmUb3bp103PPPeecdrPejvr16yvluiSamZmpEydOON936NBBqamp8vT0dLnBOa/mzZurefPmev755zVgwAC9++67lSqwcEkIAEygRw8pJEQqaNgTi0UKDXW0K21t27bVwIED9cc//tFles+ePXXmzBnNnj1bx44d08KFC/Xll1/e8vYGDhyoevXqqU+fPkpMTNSJEyeUkJCgcePGuXUDcGRkpO69917169dP69ev14kTJ/Tll19q7dq1kqRmzZpp/fr12rp1qw4dOqRnn31WaWlpbtfbt29fnT9/XqNGjdJ9992n4OBg57xmzZpp586dWrdunY4cOaLp06drx44dha6vV69e+uCDD5SYmKh9+/ZpyJAhzhuFJccTXBEREYqJidFXX32lH374QVu3btVLL72knTt3KisrS2PGjFFCQoJ+/PFHbdmyRTt27FCrVq3c3jczI7AAgAlYrVLu0695Q0vu+/h4R7uy8PLLLzsv2eRq1aqV/vSnP2nhwoVq3769tm/frhdeeOGWt1W9enVt3rxZt99+u/r27atWrVpp+PDh+uWXX9zucfn000/VuXNnDRgwQK1bt9akSZNkuzqQzbRp09ShQwdFR0erZ8+eCgwMLNaIs7Vq1dIjjzyi7777TgMHDnSZ9+yzz6pv377q37+/unbtqrNnz7r0tuRnypQpioyM1MMPP6zevXsrJibG+Zi05Lh89sUXX+jee+/VsGHD1Lx5cz3xxBP68ccfFRAQIKvVqrNnz2rw4MFq3ry5fv3rX+uhhx7S73//e7f3zcwsRt4LZxVQZmam/P39lZGRcUvdiQBwK3755RedOHFCd9xxh3x8fIq1jpUrpfHjXW/ADQ11hJW+fUumTqCsFfSz4c7nN/ewAICJ9O0r9enjeBooJcVxz0qPHmXXswKYFYEFAEzGapV69izvKgBz4R4WAABgegQWAABgegQWAABgegQWAChhleDhS6BElcTPBIEFAEpI7vDzly5dKudKAHPJ/Zm4lW+Q5ikhACghVqtVtWvXdn6pX/Xq1Z1D2ANVkWEYunTpktLT01W7dm2XEXzdRWABgBIUGBgoSc7QAsDxJY25PxvFRWABgBJksVgUFBSkBg0a5PtNxkBVU61atVvqWclFYAGAUmC1WkvklzQAB266BQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApkdgAQAApleswLJw4UKFhYXJx8dHXbt21fbt2wtse+DAAfXr109hYWGyWCyKj4+/5XUCAICqxe3Asnz5csXGxmrmzJnatWuX2rdvr+joaKWnp+fb/tKlS2rcuLFmzZqlwMDAElknAACoWiyGYRjuLNC1a1d17txZb775piTJbrcrNDRUY8eO1eTJkwtdNiws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5TVlyhRlZGQ4X8n8NQoz8vWVJk2SmjYt2mWe3NDStKljOb5LCACc3PouoXr16slqtSotLc1lelpamgIDA/NdJjAwsND29erVk6enp1q3bu3SplWrVvrXv/6V7zq9vb3l7e3tTulA+RgxQnrqqaKHj9BQae9ewgoA5OFWD4uXl5c6duyoDRs2OKfZ7XZt2LBBERER+S4TERHh0l6S1q9f72zv5eWlzp076/Dhwy5tjhw5okaNGrlTHmBO7oYPwgoA3MDtb2uOjY3VkCFD1KlTJ3Xp0kXx8fG6ePGihg0bJkkaPHiwGjZsqLi4OEnS+PHjFRkZqblz56p3795atmyZdu7cqbffftu5zokTJ6p///669957dd9992nt2rX6xz/+oYSEhJLZSwAAUKG5HVj69++vM2fOaMaMGUpNTVV4eLjWrl3rvLH25MmT8vC41nHTrVs3LV26VNOmTdPUqVPVrFkzrV69Wm3atHG2efTRR7Vo0SLFxcVp3LhxatGihT799FN17969BHYRAABUdG6Pw2JGjMMCAEDFU2rjsAAAAJQHAgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADA9AgsAADC9YgWWhQsXKiwsTD4+Puratau2b99eaPsVK1aoZcuW8vHxUdu2bfXFF18U2HbkyJGyWCyKj48vTmkAAKAScjuwLF++XLGxsZo5c6Z27dql9u3bKzo6Wunp6fm237p1qwYMGKDhw4dr9+7diomJUUxMjPbv339D21WrVumbb75RcHCw+3sCAAAqLbcDyxtvvKERI0Zo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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N = 5\n",
    "R = pf.Parameter(\"R\")\n",
    "L_value = 1\n",
    "R_value = 1\n",
    "\n",
    "opt_values = []\n",
    "for k in range(1, N):\n",
    "    ctx_plt = make_ctx_ogm(ctx_name=f\"ctx_plt_{k}\", N=k, 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",
    "    x_k = ctx_plt[f\"x_{k}\"]\n",
    "    pb_plt.set_performance_metric(f(x_k) - f(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",
    "analytical_values = [L_value / (2 * theta(i, i) ** 2) for i in iters]\n",
    "plt.scatter(\n",
    "    iters,\n",
    "    analytical_values,\n",
    "    color=\"red\",\n",
    "    marker=\"x\",\n",
    "    s=100,\n",
    "    label=\"Analytical bound $\\\\frac{L}{2*\\\\theta_N^2}$\",\n",
    ")\n",
    "plt.scatter(iters, opt_values, color=\"blue\", marker=\"o\", label=\"Numerical values\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ed37cc6a-f1ef-448c-bb63-cc559584e5db",
   "metadata": {},
   "source": [
    "## Verification of convergence of OGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "44489287-120d-4550-8cb6-8b9b6e688579",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.06189185276165096\n"
     ]
    }
   ],
   "source": [
    "N = 2\n",
    "\n",
    "ctx_prf = make_ctx_ogm(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(f(ctx_prf[f\"x_{N}\"]) - f(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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "2a001a10-1e12-4c76-ab53-076988434790",
   "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": "f88f2dc5-4c06-4b50-8110-0dd9f3012b52",
   "metadata": {},
   "source": [
    "- It turns out for OGM no further relaxation is needed. Now we store the results.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "bb573689",
   "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 f\n",
    "lamb_sol = result.get_scalar_constraint_dual_value_in_numpy(f)\n",
    "# Dual variable associated with the Gram matrix G\n",
    "S_sol = result.get_gram_dual_matrix()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d8085350",
   "metadata": {},
   "source": [
    "### Verify closed form expression of $\\lambda$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "a4471bb0-7d43-44df-b748-c1fd4528906e",
   "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": 10,
   "id": "c275e7c3",
   "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.248 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.648 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.248 & 0.401 & 0.352 & 0.0 \\\\\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": 11,
   "id": "4b0f7947-4456-4d49-a6fc-2e6da800eea1",
   "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.248 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.648 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.248 & 0.401 & 0.352 & 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 2 * theta(i, N) ** 2 / theta(N, N) ** 2\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": 12,
   "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": "markdown",
   "id": "eaaf266e-6995-48ac-9148-56efc53999c3",
   "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": 13,
   "id": "bb947fd6-7b18-408b-ad59-a520e1794419",
   "metadata": {},
   "outputs": [],
   "source": [
    "pm = pf.ExpressionManager(ctx_prf, resolve_parameters={\"L\": L_value, \"R\": R_value})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "7a8d4dab",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.062 & -0.062 & -0.124 & -0.2 & -0.176 \\\\x_\\star & -0.062 & 0.062 & 0.124 & 0.2 & 0.176 \\\\\\nabla f(x_0) & -0.124 & 0.124 & 0.248 & 0.401 & 0.352 \\\\\\nabla f(x_1) & -0.2 & 0.2 & 0.401 & 0.648 & 0.569 \\\\\\nabla f(x_2) & -0.176 & 0.176 & 0.352 & 0.569 & 0.5 \\\\\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": 15,
   "id": "d87aeca4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.062 & -0.062 & -0.124 & -0.2 & -0.176 \\\\x_\\star & -0.062 & 0.062 & 0.124 & 0.2 & 0.176 \\\\\\nabla f(x_0) & -0.124 & 0.124 & 0.248 & 0.401 & 0.352 \\\\\\nabla f(x_1) & -0.2 & 0.2 & 0.401 & 0.648 & 0.569 \\\\\\nabla f(x_2) & -0.176 & 0.176 & 0.352 & 0.569 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x_N = ctx_prf[f\"x_{N}\"]\n",
    "x_star = ctx_prf[\"x_star\"]\n",
    "z_N = ctx_prf[f\"z_{N}\"]\n",
    "S_guess = (\n",
    "    L / theta(N, N) ** 2 * 1 / 2 * (z_N - theta(N, N) / L * f.grad(x_N) - x_star) ** 2\n",
    ")\n",
    "\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": 16,
   "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": "b17bba36",
   "metadata": {},
   "source": [
    "- Interestestingly, optimal method tends to have small rank of $S$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "eb3b1343",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1\n"
     ]
    }
   ],
   "source": [
    "rank = np.linalg.matrix_rank(S_sol.matrix)\n",
    "print(rank)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2bd9173b-7ea5-45d9-bf00-036c6b0d3374",
   "metadata": {},
   "source": [
    "### Symbolic calculation"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "19173938-8cef-4959-8d25-9cf84519870f",
   "metadata": {},
   "source": [
    "- Assemble the RHS of the proof."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "5ae99afb-aaec-4a1c-ab29-28981b4c43b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+-2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2 + 2*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+-2*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2 + 1*(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+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+-2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2 + 2*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+-2*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2 + 1*(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": 19,
   "id": "e4131eb7-7da2-4c5f-8e26-7fc4851d0437",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+-2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2 + 2*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+-2*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2 + 1*(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)-L/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*1/2*\\|z_2-(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)/L*\\nabla f(x_2)-x_\\star\\|^2$"
      ],
      "text/plain": [
       "0+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+-2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2 + 2*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+-2*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2 + 1*(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)-L/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*1/2*|z_2-(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)/L*grad_f(x_2)-x_star|^2"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "RHS = interp_scalar_sum - S_guess\n",
    "display(RHS)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b93e3a81-fdb7-4388-9c7a-9428aca627c6",
   "metadata": {},
   "source": [
    "- Assemble the LHS of the proof"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "688f2c26-6f2b-4e77-b7fb-21c3c8b51659",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_2)-f(x_\\star)-L/2*(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*\\|x_0-x_\\star\\|^2$"
      ],
      "text/plain": [
       "f(x_2)-f(x_star)-L/2*(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*|x_0-x_star|^2"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x_0 = ctx_prf[\"x_0\"]\n",
    "LHS = f(x_N) - f(x_star) - L / (2 * theta(N, N) ** 2) * (x_0 - x_star) ** 2\n",
    "display(LHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "d444695a-5771-4304-8ffe-7e4e4c67d08b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_2)-f(x_\\star)-L/2*(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*\\|x_0-x_\\star\\|^2-(0+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+-2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2 + 2*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(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)+-2*(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2 + 1*(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)-L/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*1/2*\\|z_2-(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)/L*\\nabla f(x_2)-x_\\star\\|^2)$"
      ],
      "text/plain": [
       "f(x_2)-f(x_star)-L/2*(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*|x_0-x_star|^2-(0+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+-2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2 + 2*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(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)+-2*(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2 + 1*(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)-L/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*1/2*|z_2-(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)/L*grad_f(x_2)-x_star|^2)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "diff = LHS - RHS\n",
    "display(diff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "fdeaded3-0e72-4a2a-a533-887346b55892",
   "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",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
