{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "9cf4569d-bd1b-4e71-a47e-ac15eb611c41",
   "metadata": {},
   "source": [
    "# Accelerated Gradient Method Example\n",
    "\n",
    "This code tests Nesterov's accelerated gradient method (AGM), which is arguably the first accelerated method,\n",
    "introduced in \"A Method of Solving a Convex Programming Problem with Convergence Rate O(1/k^2)\" by Yurii Nesterov (1983).\n",
    "\n",
    "AGM reduces the function value with respect to the initial distance to the solution for L-smooth convex functions and achieves an O(1/k²) rate.\n",
    "\n",
    "This code recovers the rate for the secondary sequence, based on the values introduced in\n",
    "\"Optimized First-Order Methods for Smooth Convex Minimization\" by Donghwan Kim and Jeffrey A. Fessler (2016)."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1f9760ff",
   "metadata": {},
   "source": [
    "## Import the required libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b1f76844",
   "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": "7345d782",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with AGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "25fc3ad3-b58e-46db-87d1-c9c8213775e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "@functools.cache\n",
    "def theta(i):\n",
    "    if i == -1:\n",
    "        return 0\n",
    "    return 1 / sp.S(2) * (sp.S(1) + sp.sqrt(4 * theta(i - 1) ** 2 + sp.S(1)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "d780dde1-d017-4266-bd17-29eb6792bc5e",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_agm(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_agm = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    f.set_stationary_point(\"x_star\")\n",
    "    z = x\n",
    "    for i in range(N):\n",
    "        y = x - stepsize * f.grad(x)\n",
    "        z = z - stepsize * theta(i) * f.grad(x)\n",
    "        x = (1 - 1 / theta(i + 1)) * y + 1 / theta(i + 1) * z\n",
    "        z.add_tag(f\"z_{i + 1}\")\n",
    "        x.add_tag(f\"x_{i + 1}\")\n",
    "    return ctx_agm"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8e435ee9-41f9-4a1b-aa22-90a71291062e",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of AGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "aa4ca8de-9083-4c52-a1df-319c775ae2dc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7229349641d0>"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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fjJ8fHzKrN/dyvq74ez3NyMskm1YdepUKTTUuX+5FP+LKDTUth7PEgJwrdw8aeXBgwe1fft2DR8+XJJUp04dDR06VG+99ZZDv65duzq8btasmbKysuyvv/vuOw0fPlytW7eWn5+fQkJCJEnHjx8v8XPHjRunL774QidOnJAkLV26VA8++KAsFovTtR84cED5+fm67bbbSnzf1ZpKc9NNNznUFRkZqe+++06FhYU6cuSILl68qL59+9rfr1u3rnr37q0DBw649DllHeMjR46ooKBAERER9vcbNmyo9u3bu/QZ1VEddxdgGuPGSX/4g/PhIzhY+vZbwgqAilMcWopDSvGXXxVMWvnWW2/p119/VfPmze1thmHIy8tLCxYskL+/vyTbl/ClLBaLw08rd999t1q2bKklS5aoefPmKioqUufOnVVQUFDi53bv3l3dunXTe++9p9tvv1379u3Tp59+6lLtPlf5/7CrNVUWDw+PK35iu3jx4hX9rnaMXeVK+KsIl+9jRWGE5VKuhg/CCoCK5oZJK3/99Ve99957mjt3rtLS0uzLN998o+bNm+uDDz5wajs//fSTDh48qOeee0633XabOnbsqLNnz151vYcfflhLly7VO++8o+joaIfrVDw9PVVYWFjm+m3btpWPj4+Sk5MrrKaSfP311w6vv/rqK7Vt21ZWq1Vt2rSRp6entmzZYn//4sWL2rFjhzp16iRJatKkiU6dOmV/PycnR8eOHXOphjZt2qhu3boOtZw9e1aHDh0qdR3DMKp0qSyMsACAmZQ2aWUljrB88sknOnv2rMaOHWsfSSk2ZMgQvfXWW3rssceuup3rr79ejRo10uLFi9WsWTMdP35c06ZNu+p6DzzwgJ566iktWbJE7733nsN7ISEh+vrrr/X999/L19dXDRs2lIeH47+1vb299fTTT2vq1Kny9PRU3759dfr0ae3bt09jxowpV00lOX78uOLj4/Xoo49q165deuONN+x36NSvX1/jx4/XlClT1LBhQ91www2aNWuWLly4oLFjx0qSBgwYoKVLl+ruu+/WddddpxkzZshqtbpUg6+vr8aOHaspU6aoUaNGatq0qZ599tkrjklV+Oijj7R+/XqdPXtWCQkJlX6HESMsAGAWbpq08q233lJ0dPQVYUWyBZadO3fq22+/vep2PDw8tGLFCqWmpqpz58568sknNXv27Kuu5+/vryFDhsjX11exsbEO7z311FOyWq3q1KmTmjRpUup1J88//7z++Mc/asaMGerYsaOGDh2qrKysctdUklGjRikvL0+9e/fWhAkTNHnyZD3yyCP292fOnKkhQ4Zo5MiR6tGjhw4fPqx169bp+uuvl2S7kykqKkp33XWXBg4cqNjYWLVp08blOmbPnq1+/frp7rvvVnR0tG6++WaFh4eX2j89PV39+/dXp06d1LVrV3300UcO7xePihTfPn35KElBQYHGjRunTp06KTIy0n7X03333afFixfr2Wef1WeffebyfrjKYlTm+E0VysnJkb+/v7Kzs+Xn5+fucgDUQr/88ouOHTumVq1aydvb27WVS7t1uYpuaXa32267TTfeeKNef/11d5dS45w6dUqZmZkKCwtTRkaGwsPDdejQIdWvX1+S9Oc//1l16tTRd999J6vVqjvvvFNRUVH29Z999lm1a9dOo0eP1nPPPafmzZvr8ccfl2Sb/+axxx7TjBkzFBQUVGoNZf3dcPb7mxEWAHC3Wjxp5dmzZ7V69WqlpKRowoQJ7i6nRmrWrJnCwsIkSYGBgWrcuLHD3DCPP/64srOz9frrr+vuu+92CCvZ2dnatGmTRo8eLUlq1aqVjh49KskWVp588kmNHz++zLBSUbiGBQDcyZVJK4v79e9fY0ZaunfvrrNnz+rVV1+tFbfmultqaqoKCwsdLmxetGiR/P39NWnSJP3rX/9SUVGR+vXrJ0n697//rcOHD9sDT2Zmpp544glJ0pw5c7Rjxw7l5+frd7/7nYYMGVKptfOTEABUEJd/EsrLsz3I8PBh537uuTTchIYytQJc8vPPP6tfv35asmSJ+vTpY283DEMWi0UvvPCCXnjhBftrSZoxY4batm2rkf93Ifidd96piRMnuvxwTH4SAoDqjEkrUUXy8/MVGxuradOmOYQV6bd5Woovur103pb//ve/8vLykmQLFrt379att94qwzDUuHFjpaWlSZKGDRtW6ftAYAEAdxo3zjZS4uzPO8WTVo4bV7l1ocYwDEMPPvigBgwYYB8pcVZoaKi2b98uSXrxxRc1adIk1atXT4cPH9a9996rf/7zn/rpp5/UuHHjyijdAYEFANyNSStRibZs2aKVK1dqzZo1CgsLU1hYmPbs2ePUug888IBSUlLUtm1b5efn6+mn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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_agm(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) ** 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": "31702013-dab6-4ff8-acfa-85a6f651c135",
   "metadata": {},
   "source": [
    "## Verification of convergence of AGM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "b61342a4-a0ad-40b4-aed3-634cfc903b79",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.05762868749542287\n"
     ]
    }
   ],
   "source": [
    "N = sp.S(3)\n",
    "L_value = sp.S(1)\n",
    "R_value = sp.S(1)\n",
    "\n",
    "ctx_prf = make_ctx_agm(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": "markdown",
   "id": "5653252a",
   "metadata": {},
   "source": [
    "- For simplicity of calculation, this example considers a looser upper bound compared to the exact tight rate"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d3bb4440",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.05762868749542287\n"
     ]
    }
   ],
   "source": [
    "print(result.opt_value)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "52f042f9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.0661257368537568\n"
     ]
    }
   ],
   "source": [
    "desired_upper_bound = sp.N(L_value / (2 * theta(N) ** 2))\n",
    "print(desired_upper_bound)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "c046ba8d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Is the optimal value 0.05762868749542287 less than or equal to our desired upper bound 0.0661257368537568? True\n"
     ]
    }
   ],
   "source": [
    "print(\n",
    "    f\"Is the optimal value {result.opt_value} less than or equal to our desired upper bound {desired_upper_bound}?\",\n",
    "    result.opt_value <= desired_upper_bound,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "26845181",
   "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": "fc263456-0fbd-470b-a898-44b149490930",
   "metadata": {},
   "source": [
    "### Solve the problem again with the found relaxation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "eaff29cb-63be-471f-91a0-bffc01ab759e",
   "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": 12,
   "id": "54b52e21",
   "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",
    "pb_prf.set_relaxed_constraints(relaxed_constraints)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "99c0eb43-ae60-4e75-924d-0353491bd907",
   "metadata": {},
   "source": [
    "- Solve the PEP problem again with the relaxed constraints and store the results.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "e8f1d3a5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.05762927790762932\n"
     ]
    }
   ],
   "source": [
    "result = pb_prf.solve(resolve_parameters={\"L\": L_value, \"R\": R_value})\n",
    "print(result.opt_value)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5b0dc129",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e13709ac",
   "metadata": {},
   "source": [
    "### Recover and verify the calculation in the paper \"Optimized first-order methods for smooth convex minimization\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b90f8085",
   "metadata": {},
   "source": [
    "- To recover the same value (which may help simplify the calculation), we set an additional constraint that relaxes the optimal value to $\\frac{L}{2\\theta_N^2}$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "674f812e",
   "metadata": {},
   "outputs": [],
   "source": [
    "pb_prf.add_dual_val_constraint(\"initial_condition\", \"==\", desired_upper_bound)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "529b9db7",
   "metadata": {},
   "source": [
    "- Under similar reason, we set $\\lambda_{i-1,i} = \\theta_{i-1}^2 / \\theta_N^2$ "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "7b5447ba",
   "metadata": {},
   "outputs": [],
   "source": [
    "for i in range(N + 1):\n",
    "    pb_prf.add_dual_val_constraint(\n",
    "        f\"f:x_{i},x_{i + 1}\", \"==\", theta(i) ** 2 / theta(N) ** 2\n",
    "    )"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dfac1956",
   "metadata": {},
   "source": [
    "- Now we solve the dual problem with the additional constraints and proceed with the remaining steps as usual"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "e658867c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.06612574139322015\n"
     ]
    }
   ],
   "source": [
    "result_dual = pb_prf.solve_dual(resolve_parameters={\"L\": L_value, \"R\": R_value})\n",
    "print(result_dual.opt_value)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1cb4c4f2",
   "metadata": {},
   "source": [
    "- Store the dual variables."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "bb573689",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Dual variable associated with the initial condition\n",
    "tau_sol = result_dual.dual_var_manager.dual_value(\"initial_condition\")\n",
    "# Dual variable associated with the interpolations conditions of f\n",
    "lamb_sol = result_dual.get_scalar_constraint_dual_value_in_numpy(f)\n",
    "# Dual variable associated with the Gram matrix G\n",
    "S_sol = result_dual.get_gram_dual_matrix()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d8085350",
   "metadata": {},
   "source": [
    "### Verify closed form expression of $\\lambda$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c413d8be",
   "metadata": {},
   "source": [
    "- Print the values of $\\lambda$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "c275e7c3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_0 & x_1 & x_2 & x_3 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.132 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.346 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.636 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.132 & 0.214 & 0.29 & 0.364 & 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": [
    "- The closed form expression of $\\lambda$ suggested in the paper"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "939fc693",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_0 & x_1 & x_2 & x_3 & x_\\star \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.132 & 0.0 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 0.346 & 0.0 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.0 & 0.636 & 0.0 \\\\x_3 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_\\star & 0.132 & 0.214 & 0.29 & 0.364 & 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 < N + 1:\n",
    "            return theta(j) / theta(N) ** 2\n",
    "    if i < N and i + 1 == j:  # Additional constraint 2 (consecutive)\n",
    "        return theta(i) ** 2 / theta(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 the values of $\\lambda$ we obtained from the solver recover the values in the paper"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "9c45d58e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Is our closed-form expression for lambda correct for N=3? True\n"
     ]
    }
   ],
   "source": [
    "print(\n",
    "    f\"Is our closed-form expression for lambda correct for N={N}?\",\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": "59bcf310-ed42-4c15-80c1-de3f555c3437",
   "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": 21,
   "id": "17461868",
   "metadata": {},
   "outputs": [],
   "source": [
    "pm = pf.ExpressionManager(ctx_prf, resolve_parameters={\"L\": L_value, \"R\": R_value})"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "12b7503a-c150-47b0-bf46-b0047700b1b7",
   "metadata": {},
   "source": [
    "- Print the values of $S$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "7a8d4dab",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) & \\nabla f(x_3) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.066 & -0.066 & -0.066 & -0.107 & -0.145 & -0.182 \\\\x_\\star & -0.066 & 0.066 & 0.066 & 0.107 & 0.145 & 0.182 \\\\\\nabla f(x_0) & -0.066 & 0.066 & 0.132 & 0.107 & 0.145 & 0.182 \\\\\\nabla f(x_1) & -0.107 & 0.107 & 0.107 & 0.346 & 0.235 & 0.294 \\\\\\nabla f(x_2) & -0.145 & 0.145 & 0.145 & 0.235 & 0.636 & 0.399 \\\\\\nabla f(x_3) & -0.182 & 0.182 & 0.182 & 0.294 & 0.399 & 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": 23,
   "id": "d87aeca4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) & \\nabla f(x_3) \\\\\n",
       "        \\hline\n",
       "        x_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 \\\\\\nabla f(x_0) & -0.0 & 0.0 & 0.066 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_1) & -0.0 & 0.0 & 0.0 & 0.173 & 0.0 & 0.0 \\\\\\nabla f(x_2) & -0.0 & 0.0 & 0.0 & 0.0 & 0.318 & 0.0 \\\\\\nabla f(x_3) & 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": [
    "z_N = ctx_prf[f\"z_{N}\"]\n",
    "x_N = ctx_prf[f\"x_{N}\"]\n",
    "x_star = ctx_prf[\"x_star\"]\n",
    "S_guess1 = L / theta(N) ** 2 * 1 / 2 * (z_N - theta(N) / L * f.grad(x_N) - x_star) ** 2\n",
    "\n",
    "S_guess1_eval = pm.eval_scalar(S_guess1).inner_prod_coords\n",
    "\n",
    "remainder1 = S_sol.matrix - S_guess1_eval\n",
    "pf.pprint_labeled_matrix(remainder1, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2caf1acc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccc}\n",
       "         & x_0 & x_\\star & \\nabla f(x_0) & \\nabla f(x_1) & \\nabla f(x_2) & \\nabla f(x_3) \\\\\n",
       "        \\hline\n",
       "        x_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 \\\\\\nabla f(x_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 \\\\\\nabla f(x_2) & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_3) & 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": [
    "coef = 1 / 2 * 1 / theta(N) ** 2 / L\n",
    "S_guess2 = coef * sum(theta(i) ** 2 * f.grad(ctx_prf[f\"x_{i}\"]) ** 2 for i in range(N))\n",
    "S_guess2_eval = pm.eval_scalar(S_guess2).inner_prod_coords\n",
    "\n",
    "remainder2 = remainder1 - S_guess2_eval\n",
    "pf.pprint_labeled_matrix(remainder2, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "1a17abbe",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Is our closed-form expression for S correct for N=3? True\n"
     ]
    }
   ],
   "source": [
    "S_guess_eval = S_guess1_eval + S_guess2_eval\n",
    "\n",
    "print(\n",
    "    f\"Is our closed-form expression for S correct for N={N}?\",\n",
    "    np.allclose(S_guess1_eval + S_guess2_eval, S_sol.matrix, atol=1e-3),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b897eef4",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "912d49be",
   "metadata": {},
   "source": [
    "### Above calculation corresponds to the equality below:"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e326710b",
   "metadata": {},
   "source": [
    "\\begin{align*}\n",
    "    f(x_N) - f(x_\\star) - \\frac{L}{2\\theta_N^2} \\| x_0 - x_\\star \\|^2  \n",
    "    &= \\frac{1}{\\theta_N^2} \\sum _{i=1}^N \\theta_{i-1}^2 \\left( f(x_{i})-f(x_{i-1}) + \\langle \\nabla f(x_{i}) , x_{i-1} - x_{i} \\rangle + \\frac{1}{2 L} \\| \\nabla f(x_{i-1}) - \\nabla f(x_i) \\|^2 \\right)  \\\\ \n",
    "    &\\quad + \\frac{1}{\\theta_N^2} \\sum _{i=0}^N \\theta_{i} \\left( f(x_i)-f(x_{\\star}) + \\langle \\nabla f(x_{i}) , x_{\\star} - x_{i} \\rangle + \\frac{1}{2 L} \\| \\nabla f(x_i) \\|^2 \\right) \\\\\n",
    "    &\\quad-  \\frac{L}{2\\theta_N^2} \\left\\| z_{N+1} - x_\\star \\right\\|^2  \\\\\n",
    "    &\\quad - \\frac{1}{2\\theta_N^2L} \\sum_{i=0}^{N-1} \\theta_i^2 \\|\\nabla f(x_i) \\|^2      \n",
    "\\end{align*}"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3cfce76b",
   "metadata": {},
   "source": [
    "### Symbolic calculation to check"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3c6574f5-2ba6-49ac-a7d7-c2346e2f3f1f",
   "metadata": {},
   "source": [
    "- Assemble the RHS of the proof."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "992f6f82",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*(f(x_3)-f(x_2)+\\nabla f(x_3)*(x_2-(x_3))+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_3)\\|^2)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(5)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*(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/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)*(f(x_3)-f(x_\\star)+\\nabla f(x_3)*(x_\\star-(x_3))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_3)\\|^2)$"
      ],
      "text/plain": [
       "0+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*(f(x_3)-f(x_2)+grad_f(x_3)*(x_2-(x_3))+1/2*L*|grad_f(x_2)-grad_f(x_3)|^2)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(5)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*(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/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)*(f(x_3)-f(x_star)+grad_f(x_3)*(x_star-(x_3))+1/2*L*|grad_f(x_star)-grad_f(x_3)|^2)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x = ctx_prf.tracked_point(f)\n",
    "interp_scalar_sum = pf.Scalar.zero()\n",
    "for x_i, x_j in itertools.product(x, x):\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": 34,
   "id": "b8618e2c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*(f(x_3)-f(x_2)+\\nabla f(x_3)*(x_2-(x_3))+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_3)\\|^2)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(5)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*(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/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)*(f(x_3)-f(x_\\star)+\\nabla f(x_3)*(x_\\star-(x_3))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_3)\\|^2)-(L/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*1/2*\\|z_3-(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)/L*\\nabla f(x_3)-x_\\star\\|^2+0.5/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2/L*(0+1*\\|\\nabla f(x_0)\\|^2+(1/2 + sqrt(5)/2)^2*\\|\\nabla f(x_1)\\|^2+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2*\\|\\nabla f(x_2)\\|^2))$"
      ],
      "text/plain": [
       "0+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*(f(x_3)-f(x_2)+grad_f(x_3)*(x_2-(x_3))+1/2*L*|grad_f(x_2)-grad_f(x_3)|^2)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(5)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*(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/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)*(f(x_3)-f(x_star)+grad_f(x_3)*(x_star-(x_3))+1/2*L*|grad_f(x_star)-grad_f(x_3)|^2)-(L/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*1/2*|z_3-(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)/L*grad_f(x_3)-x_star|^2+0.5/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2/L*(0+1*|grad_f(x_0)|^2+(1/2 + sqrt(5)/2)**2*|grad_f(x_1)|^2+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2*|grad_f(x_2)|^2))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "RHS = interp_scalar_sum - (S_guess1 + S_guess2)\n",
    "display(RHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "80b4a936-eeac-4ce7-9f67-be776fed9379",
   "metadata": {},
   "outputs": [],
   "source": [
    "LHS = f(x_N) - f(x_star) - L / (2 * theta(i) ** 2) * (x[0] - x_star) ** 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "9c185dd7-856a-44ae-8727-e48ce23d82ce",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_3)-f(x_\\star)-L/2*(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*\\|x_0-x_\\star\\|^2-(0+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(5)/2)^2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*(f(x_3)-f(x_2)+\\nabla f(x_3)*(x_2-(x_3))+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_3)\\|^2)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(5)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*(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/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)*(f(x_3)-f(x_\\star)+\\nabla f(x_3)*(x_\\star-(x_3))+1/2*L*\\|\\nabla f(x_\\star)-\\nabla f(x_3)\\|^2)-(L/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2*1/2*\\|z_3-(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)/L*\\nabla f(x_3)-x_\\star\\|^2+0.5/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2)/2)^2/L*(0+1*\\|\\nabla f(x_0)\\|^2+(1/2 + sqrt(5)/2)^2*\\|\\nabla f(x_1)\\|^2+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)^2)/2)^2*\\|\\nabla f(x_2)\\|^2)))$"
      ],
      "text/plain": [
       "f(x_3)-f(x_star)-L/2*(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*|x_0-x_star|^2-(0+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(5)/2)**2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*(f(x_3)-f(x_2)+grad_f(x_3)*(x_2-(x_3))+1/2*L*|grad_f(x_2)-grad_f(x_3)|^2)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(5)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/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)+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*(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/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)*(f(x_3)-f(x_star)+grad_f(x_3)*(x_star-(x_3))+1/2*L*|grad_f(x_star)-grad_f(x_3)|^2)-(L/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2*1/2*|z_3-(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)/L*grad_f(x_3)-x_star|^2+0.5/(1/2 + sqrt(1 + 4*(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2)/2)**2/L*(0+1*|grad_f(x_0)|^2+(1/2 + sqrt(5)/2)**2*|grad_f(x_1)|^2+(1/2 + sqrt(1 + 4*(1/2 + sqrt(5)/2)**2)/2)**2*|grad_f(x_2)|^2)))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "diff = LHS - RHS\n",
    "display(diff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "c2b4c1f7",
   "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
}
