{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "3e23641c",
   "metadata": {},
   "source": [
    "# Gradient Descent Method Example"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8baf79c6",
   "metadata": {},
   "source": [
    "## Import the required libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3d7b57bb",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with GD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ad6701aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_gd(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_gd = 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",
    "    for i in range(N):\n",
    "        x = x - 1 / L * f.grad(x)\n",
    "        x.add_tag(f\"x_{i + 1}\")\n",
    "    return ctx_gd"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9f1f0159",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of GD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "425348ad",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x71a351e930d0>"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "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_gd(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(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",
    "cont_iters = np.arange(1, N, 0.01)\n",
    "plt.plot(\n",
    "    cont_iters,\n",
    "    L_value / (4 * cont_iters + 2),\n",
    "    \"r-\",\n",
    "    label=\"Analytical bound $\\\\frac{L}{4k + 2}$\",\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 GM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d0e252cb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.09999778825848689\n"
     ]
    }
   ],
   "source": [
    "N = sp.S(2)\n",
    "L_value = sp.S(1)\n",
    "R_value = sp.S(1)\n",
    "\n",
    "ctx_prf = make_ctx_gd(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": 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",
    "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 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": "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": 13,
   "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": 14,
   "id": "d17dee1d",
   "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.1 & -0.1 & -0.125 & -0.208 & -0.167 \\\\x_\\star & -0.1 & 0.1 & 0.125 & 0.208 & 0.167 \\\\\\nabla f(x_0) & -0.125 & 0.125 & 0.25 & 0.208 & 0.167 \\\\\\nabla f(x_1) & -0.208 & 0.208 & 0.208 & 0.667 & 0.167 \\\\\\nabla f(x_2) & -0.167 & 0.167 & 0.167 & 0.167 & 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": 15,
   "id": "c6fad11c",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = ctx_prf.tracked_point(f)\n",
    "x_0 = ctx_prf[\"x_0\"]\n",
    "x_star = ctx_prf[\"x_star\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "7c4c0e27",
   "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.0 & -0.0 & 0.0 & 0.0 & -0.0 \\\\x_\\star & -0.0 & 0.0 & -0.0 & -0.0 & 0.0 \\\\\\nabla f(x_0) & 0.0 & -0.0 & 0.094 & -0.052 & -0.042 \\\\\\nabla f(x_1) & 0.0 & -0.0 & -0.052 & 0.233 & -0.181 \\\\\\nabla f(x_2) & -0.0 & 0.0 & -0.042 & -0.181 & 0.222 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "tau_cand = L / sp.S(4 * N + 2)\n",
    "\n",
    "grad_terms = sum(lamb(x_star.tag, x[i].tag) * f.grad(x[i]) for i in range(N + 1))\n",
    "z_N = x_star - x_0 + 1 / (2 * tau_cand) * grad_terms\n",
    "iter_diff_square = tau_cand * z_N**2\n",
    "\n",
    "remainder_1 = S_sol.matrix - pm.eval_scalar(iter_diff_square).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder_1, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "3915864b",
   "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.0 & -0.0 & 0.0 & 0.0 & -0.0 \\\\x_\\star & -0.0 & 0.0 & -0.0 & -0.0 & 0.0 \\\\\\nabla f(x_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 \\\\\\nabla f(x_2) & -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": [
    "grad_diff_square = pf.Scalar.zero()\n",
    "for i in range(N + 1):\n",
    "    for j in range(i + 1, N + 1):\n",
    "        const_1 = (2 * N + 1) * lamb(x_star.tag, x[i].tag) - 1\n",
    "        const_2 = lamb(x_star.tag, x[j].tag)\n",
    "        grad_diff_square += (\n",
    "            1 / (2 * L) * (const_1 * const_2 * (f.grad(x[i]) - f.grad(x[j])) ** 2)\n",
    "        )\n",
    "\n",
    "remainder_2 = remainder_1 - pm.eval_scalar(grad_diff_square).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder_2, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "7c78a359",
   "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.1 & -0.1 & -0.125 & -0.208 & -0.167 \\\\x_\\star & -0.1 & 0.1 & 0.125 & 0.208 & 0.167 \\\\\\nabla f(x_0) & -0.125 & 0.125 & 0.25 & 0.208 & 0.167 \\\\\\nabla f(x_1) & -0.208 & 0.208 & 0.208 & 0.667 & 0.167 \\\\\\nabla f(x_2) & -0.167 & 0.167 & 0.167 & 0.167 & 0.5 \\\\\n",
       "        \\end{array}\n",
       "        $"
      ],
      "text/plain": [
       "<IPython.core.display.Math object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "S_guess = iter_diff_square + grad_diff_square\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": "markdown",
   "id": "e35888e9",
   "metadata": {},
   "source": [
    "- Check whether our candidate of $S$ matches with solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "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(pm.eval_scalar(S_guess).inner_prod_coords, 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": 20,
   "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(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": 21,
   "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)-(L/10*\\|x_\\star-x_0+1/2*L/10*(1/4*\\nabla f(x_0)+5/12*\\nabla f(x_1)+1/3*\\nabla f(x_2))\\|^2+0+1/2*L*5/48*\\|\\nabla f(x_0)-\\nabla f(x_1)\\|^2+1/2*L*1/12*\\|\\nabla f(x_0)-\\nabla f(x_2)\\|^2+1/2*L*13/36*\\|\\nabla f(x_1)-\\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)-(L/10*|x_star-x_0+1/2*L/10*(1/4*grad_f(x_0)+5/12*grad_f(x_1)+1/3*grad_f(x_2))|^2+0+1/2*L*5/48*|grad_f(x_0)-grad_f(x_1)|^2+1/2*L*1/12*|grad_f(x_0)-grad_f(x_2)|^2+1/2*L*13/36*|grad_f(x_1)-grad_f(x_2)|^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": 22,
   "id": "a95f8b25-8ef3-4ed1-b907-3fbe8d5375fc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_2)-f(x_\\star)-L/10*\\|x_0-x_\\star\\|^2$"
      ],
      "text/plain": [
       "f(x_2)-f(x_star)-L/10*|x_0-x_star|^2"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "LHS = f(x[N]) - f(x_star) - L / (4 * N + 2) * (x[0] - x_star) ** 2\n",
    "display(LHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "f501f6a9-77b5-4711-8a10-4a168d0dd82d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_2)-f(x_\\star)-L/10*\\|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)-(L/10*\\|x_\\star-x_0+1/2*L/10*(1/4*\\nabla f(x_0)+5/12*\\nabla f(x_1)+1/3*\\nabla f(x_2))\\|^2+0+1/2*L*5/48*\\|\\nabla f(x_0)-\\nabla f(x_1)\\|^2+1/2*L*1/12*\\|\\nabla f(x_0)-\\nabla f(x_2)\\|^2+1/2*L*13/36*\\|\\nabla f(x_1)-\\nabla f(x_2)\\|^2))$"
      ],
      "text/plain": [
       "f(x_2)-f(x_star)-L/10*|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)-(L/10*|x_star-x_0+1/2*L/10*(1/4*grad_f(x_0)+5/12*grad_f(x_1)+1/3*grad_f(x_2))|^2+0+1/2*L*5/48*|grad_f(x_0)-grad_f(x_1)|^2+1/2*L*1/12*|grad_f(x_0)-grad_f(x_2)|^2+1/2*L*13/36*|grad_f(x_1)-grad_f(x_2)|^2))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "diff = LHS - RHS\n",
    "display(diff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "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) - f(x_\\star) - \\frac{L}{4N+2} \\| x_0 - x_\\star \\|^2  \\\\\n",
    "    &= \\sum _{i=1}^N \\lambda_{i,i-1} \\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 +\\sum _{i=0}^N \\lambda_{\\star,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{1}{2L} \\sum_{i=0}^N \\sum_{j=i+1}^N \\lambda_{\\star,j} \\left( (2N+1) \\lambda_{\\star,i} - 1 \\right) \\| \\nabla f(x_j) -\\nabla f(x_i) \\|^2 \\\\\n",
    "     &\\quad-  \\frac{L}{4N+2} \\left\\| x_0 - \\frac{2N+1}{L} \\sum_{i=0}^{N} \\lambda_{\\star,i} \\nabla f(x_i) - x_\\star \\right\\|^2 \n",
    "\\end{align*}\n",
    "where \n",
    "\\begin{align*}\n",
    "\\lambda_{i-1,i} = \\frac{i}{2N + 1 - i}, \\quad i = 1, \\dots, N,\n",
    "\\qquad\n",
    "\\lambda_{\\star,i} =\n",
    "\\begin{cases}\n",
    "\\lambda_{0,1} & i = 0 \\\\\n",
    "\\lambda_{i,i+1} - \\lambda_{i-1,i} & i = 1, \\dots, N-1 \\\\\n",
    "1 - \\lambda_{i-1,i} & i = N\n",
    "\\end{cases}\n",
    "\\end{align*}"
   ]
  }
 ],
 "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
}
