{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "33a942f6-6135-4250-8ae2-822ede292e27",
   "metadata": {},
   "source": [
    "# OGM-G Example\n",
    "\n",
    "This code tests OGM-G which is the state-of-the-art method\n",
    "that reduces gradient norm square respect to initial function value gap for L-smooth\n",
    "convex functions. Introduced in \"Optimizing the Efficiency of First-Order Methods for\n",
    "Decreasing the Gradient of Smooth Convex Functions\" by Donghwan Kim, Jeffrey A Fessler (2021).\n"
   ]
  },
  {
   "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": "5f9e7209-e49b-465b-9357-9d26b2135976",
   "metadata": {},
   "source": [
    "## Define the functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "548463f7-4fdd-49c9-93b4-628c5b2dafaa",
   "metadata": {},
   "outputs": [],
   "source": [
    "L = pf.Parameter(\"L\")\n",
    "f = pf.SmoothConvexFunction(is_basis=True, tags=[\"f\"], L=L)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6376d5cc-585a-4525-9f35-424b8986cdc7",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with OGM-G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3298db3d-2551-4be7-be8d-6f941e85236c",
   "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)))\n",
    "\n",
    "\n",
    "def reverse_theta(i, N):\n",
    "    return theta(N - i, N)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b2436040-63ee-469d-8a6f-38bc702630d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_ogm_g(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_ogm_g = pf.PEPContext(ctx_name).set_as_current()\n",
    "    x = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    z = x\n",
    "    z = z - stepsize * (reverse_theta(0, N) + 1) / 2 * f.grad(x)\n",
    "    z.add_tag(f\"z_{1}\")\n",
    "    f.set_stationary_point(\"x_star\")\n",
    "    for i in range(1, N + 1):\n",
    "        y = x - stepsize * f.grad(x)\n",
    "        x = (reverse_theta(i + 1, N) / reverse_theta(i, N)) ** 4 * y + (\n",
    "            1 - (reverse_theta(i + 1, N) / reverse_theta(i, N)) ** 4\n",
    "        ) * z\n",
    "        x.add_tag(f\"x_{i}\")\n",
    "\n",
    "        z = z - stepsize * reverse_theta(i, N) * f.grad(x)\n",
    "        z.add_tag(f\"z_{i + 1}\")\n",
    "    return ctx_ogm_g"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8e39e0db-bd3c-4ea5-910e-822b029cdb6d",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of OGM-G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "58864d65-a920-4141-9d99-9182c3c51ce3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7d71e297e890>"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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Dh06VO0+VWTSpEkqLS1V//799cADD2jGjBm65557XNvnz5+vMWPGaOLEierTp49yc3P10UcfqXXr1pKcTzINGjRIP//5zzVixAiNGjVKnTt39rgfzz//vAYOHKjbbrtNcXFxGjBggKKjoystn5+fr8GDB6tnz56KjIzUX//6V7ftZ0dFzj4+feEoSXl5uaZOnaqePXsqNjbW9dTTHXfcoT/96U967LHH9MEHH3i8H56yGLU5flNHbDabAgICVFxcXKOz3gKAJ06dOqW8vDx17NjR82kZKnt0uY4eaa5vw4YN03XXXaeXXnqpvrvS6BQUFKioqEhRUVEqLCxUdHS0vv32WzVv3lyS9Mc//lFNmjTR7t27ZbVadeutt2rQoEGu+o899pi6du2qyZMn6/HHH1doaKjuv/9+Sc75b+677z4lJyerQ4cOlfahsp8NTz6/GWEBgPp2qVBSlXlaGrCjR4/qvffeU1ZWlh544IH67k6jFBISoqioKEnOr8MJDAx0mxvm/vvvV3FxsV566SXddtttbmGluLhYn376qSZPnizJOTv93r17JTnDym9/+1tNmzbtkmGlpnAPCwDUp6qMoFz49NDgwY1mpOWGG27Q0aNH9dxzz10Vj+bWty1btshut7vd2Lx48WIFBATooYce0v/7f/9PDodDA3/8wrx///vfys3NdQWeoqIiTZ/u/PqahQsXavPmzSorK9PNN9+sMWPG1GrfuSQEADXE40tCpaXOLzLMza3a5Z7zw01EBFMrwCP/+9//NHDgQKWnp+vGG290rTcMQxaLRU8++aSefPJJ12tJSk5OVpcuXTTxxxvBb731Vj344IMefzkml4QAoCFj0krUkbKyMo0aNUqzZ892CyvSuXlazt50e/68LceOHZOPj48kZ7j48ssvNWTIEBmGocDAQOXk5EiSxo0bV+v7QGABgPo0dapzpKSql3fOTlo5dWrt9guNhmEYuuuuuzR06FDXSElVRUREaNOmTZKcX2b80EMPqVmzZsrNzdXtt9+uf/zjHzpy5IgCAwNro+tuCCwAUN+YtBK1aMOGDVq5cqVWr16tqKgoRUVFafv27VWqe+eddyorK0tdunRRWVmZHnnkEUnSl19+qZ/97Gc6ePCgvvzyS91www21uQuSuOkWAIBGbcCAAdWe2j8wMFBffPHFReu//PJL3X///dq1a5fefffdKk0seKUYYQEAAB7Jz89XWFiYfvnLX+rtt9+u9VluJUZYAACAh9566y1JzntcTpw4USfvyQgLAAAwPQILAAAwPQILAAAwPQILAAAwPQILAAAwPQILAAAwPQILAAAwPQILAAAwPQILAAAwPQILAAAwPQILAKDBCg8PV2pqao21N3jwYM2cObPG2qvIXXfdpVGjRtXqezRGBBYAuMrdddddslgsmj9/vtv61atXy2Kx1FOvqmbz5s2655576rsbqAMEFgAwGbtdysqS3nnH+a/dXvvv6evrq+eee05Hjx6t/TerAeXl5ZKktm3bqlmzZvXcG9QFAgsAmMiqVVJ4uDRkiHTnnc5/w8Od62tTXFycgoODlZKSUmmZJ598UlFRUW7rUlNTFR4e7np99nLHs88+q6CgILVq1UpPP/20zpw5o4cfflht2rRRhw4d9MYbb7i1k5+fr1/96ldq1aqV2rRpo5EjR2rfvn0Xtfv73/9eoaGh6tatm6SLLwkdO3ZM9957r4KCguTr66vrr79e//rXvyRJR44c0fjx49W+fXs1a9ZMvXr10jvvvFPlY/Ttt9/KYrHov//9r9v6F154QZ07d5Yk2e12TZkyRR07dpSfn5+6deumF1988ZLtVnRZKyoqSk8++aTbft19991q27at/P39NXToUH311Veu7V999ZWGDBmili1byt/fX9HR0friiy+qvG8NAYEFAExi1Srp9tulAwfc13//vXN9bYYWq9WqZ599Vi+//LIOXNgBD61du1YHDx7Up59+qkWLFmnu3Ln6+c9/rtatW2vjxo267777dO+997re5/Tp04qPj1fLli21fv16bdiwQS1atNDw4cNdIymSlJmZqV27dmnNmjWuEHI+h8OhW2+9VRs2bNBbb72lb775RvPnz5fVapUknTp1StHR0Xr//ff19ddf65577tHEiRO1adOmKu1X165d1bdvX/3lL39xW/+Xv/xFd955p6sPHTp00F//+ld98803Sk5O1qOPPqp33323WsfyrDvuuEOHDh3Shx9+qC1btqhPnz4aNmyY/ve//0mSJkyYoA4dOmjz5s3asmWLZs+eraZNm17Re5qO0QgUFxcbkozi4uL67gqAq1hpaanxzTffGKWlpR7XPXPGMDp0MAyp4sViMYywMGe5mjZ58mRj5MiRhmEYxv/93/8Zv/nNbwzDMIz33nvPOP9jYu7cuUbv3r3d6r7wwgvGtdde69bWtddea9jtdte6bt26GQMHDjxvX88YzZs3N9555x3DMAxj+fLlRrdu3QyHw+EqU1ZWZvj5+RkfffSRq92goCCjrKzM7f2vvfZa44UXXjAMwzA++ugjw8vLy9i1a1eV933EiBHG7373O9frQYMGGTNmzKi0/AsvvGB07tzZ9XrXrl2GJGPnzp2V1nnggQeMMWPGuF6ff7wv3IezevfubcydO9cwDMNYv3694e/vb5w6dcqtTOfOnY3XXnvNMAzDaNmypbF06dJK+1DfKvvZ8OTzmxEWADCB9esvHlk5n2FI+fnOcrXpueee07Jly7Rz585qt3HdddfJy+vcx0tQUJB69erlem21WnXNNdfo0KFDkpyXM3Jzc9WyZUu1aNFCLVq0UJs2bXTq1Cnt2bPHVa9Xr17y9vau9H1zcnLUoUMHde3atcLtdrtd8+bNU69evdSmTRu1aNFCH330kfbv31/lfRs3bpz27dunzz//XJJzdKVPnz7q3r27q0xaWpqio6PVtm1btWjRQn/60588eo8LffXVVzpx4oSuueYa1/Fp0aKF8vLyXMcnMTFRd999t+Li4jR//ny349ZYVCuwpKWlKTw8XL6+voqJianycNqKFStksVguepzr7B3q5y/Dhw+vTtcAoEEqKKjZctV10003KT4+XnPmzLlom5eXlwzDcFt3+vTpi8pdeCnCYrFUuM7hcEiSTpw4oejoaOXk5Lgt3377retSiyQ1b978kn338/O75Pbnn39eL774oh555BF98sknysnJUXx8vNtlp8sJDg7W0KFD9fbbb0uS3n77bU2YMMG1fcWKFZo1a5amTJmijz/+WDk5OUpISLjke1zuuJ44cUIhISEXHZ9du3bp4YcfluS8v2jHjh0aMWKE1q5dq549e+q9996r8n41BE08rbBy5UolJiZq8eLFiomJUWpqquLj47Vr1y61a9eu0nr79u3TrFmzNHDgwAq3Dx8+3O0mLB8fH0+7BgANVkhIzZa7EvPnz1dUVJTrxtaz2rZtq8LCQhmG4XrcOScn54rfr0+fPlq5cqXatWsnf3//arcTGRmpAwcO6Ntvv61wlGXDhg0aOXKkfv3rX0ty3m/y7bffqmfPnh69z4QJE5SUlKTx48dr7969GjdunNt73Hjjjbr//vtd6y432tG2bVsVnJdEbTab8vLyXK/79OmjwsJCNWnSxO0G5wt17dpVXbt21W9/+1uNHz9eb7zxhn75y196tG9m5vEIy6JFizR16lQlJCSoZ8+eWrx4sZo1a6YlS5ZUWsdut2vChAl66qmn1KlTpwrL+Pj4KDg42LW0bt3a064BQIM1cKDUoYNU2bQnFosUFuYsV9t69eqlCRMm6KWXXnJbP3jwYB0+fFgLFizQnj17lJaWpg8//PCK32/ChAkKDAzUyJEjtX79euXl5SkrK0sPPfSQRzcADxo0SDfddJPGjBmjNWvWKC8vTx9++KEyMjIkSV26dNGaNWv02WefaefOnbr33ntVVFTkcX9Hjx6t48ePa9q0aRoyZIhCQ0Nd27p06aIvvvhCH330kb799ls98cQT2rx58yXbGzp0qJYvX67169dr+/btmjx5sutGYcn5BFdsbKxGjRqljz/+WPv27dNnn32mxx57TF988YVKS0v14IMPKisrS9999502bNigzZs3q0ePHh7vm5l5FFjKy8u1ZcsWxcXFnWvAy0txcXHKzs6utN7TTz+tdu3aacqUKZWWycrKUrt27dStWzdNmzZNR44cqbRsWVmZbDab2wIADZnVKp19+vXC0HL2dWqqs1xdePrpp12XbM7q0aOH/vjHPyotLU29e/fWpk2bNGvWrCt+r2bNmunTTz/VT37yE40ePVo9evTQlClTdOrUKY9HXP7+97+rX79+Gj9+vHr27KmkpCTZf5zI5vHHH1efPn0UHx+vwYMHKzg4uFozzrZs2VK33XabvvrqK7fLQZJ07733avTo0Ro7dqxiYmJ05MgRt9GWisyZM0eDBg3Sz3/+c40YMUKjRo1yPSYtOS+fffDBB7rpppuUkJCgrl27aty4cfruu+8UFBQkq9WqI0eOaNKkSeratat+9atf6dZbb9VTTz3l8b6ZmcW48MLZJRw8eFDt27fXZ599ptjYWNf6pKQkrVu3Ths3bryozn/+8x+NGzdOOTk5CgwM1F133aVjx45p9erVrjIrVqxQs2bN1LFjR+3Zs0ePPvqoWrRooezsbLeUedaTTz5Z4f+I4uLiKxpOBIArcerUKeXl5aljx47y9fWtVhurVkkzZrjfgBsW5gwro0fXTD+BulbZz4bNZlNAQECVPr89vofFE8ePH9fEiROVnp6uwMDASsudf/2v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Mv35z3/W4MGDtXHjRvXp06e+u1cnHA6HZs6cqZ/+9Ke6/vrrKy1X2XnV2O/3uVBVj1e3bt20ZMkSRUZGqri4WAsXLtSNN96oHTt21PoXoda37du3KzY2VqdOnVKLFi303nvvqWfPnhWWvdrPK0+O1dV8TknSihUrtHXrVm3evLlK5evj3CKwoNZ069ZN3bp1c72+8cYbtWfPHr3wwgtavnx5Pfas7jzwwAP6+uuv9Z///Ke+u9IgVPV4xcbGuv2lfOONN6pHjx567bXXNG/evNruZr3q1q2bcnJyVFxcrL/97W+aPHmy1q1bV+kH8dXMk2N1NZ9T+fn5mjFjhtasWWPqG40JLB4KDg5WUVGR27qioiL5+/szulIF/fv3v2o+vB988EH961//0qeffnrZv9AqO6+Cg4Nrs4um4snxulDTpk11ww03KDc3t5Z6Zx7e3t6KiIiQJEVHR2vz5s168cUX9dprr11U9mo/rzw5Vhe6ms6pLVu26NChQ24j33a7XZ9++qleeeUVlZWVyWq1utWpj3OLe1g8FBsbq8zMTLd1a9asueR1UZyTk5OjkJCQ+u5GrTIMQw8++KDee+89rV27Vh07drxsnav5vKrO8bqQ3W7X9u3bG/25VRGHw6GysrIKt13N51VFLnWsLnQ1nVPDhg3T9u3blZOT41r69u2rCRMmKCcn56KwItXTuVVrt/M2EMePHze+/PJL48svvzQkGYsWLTK+/PJL47vvvjMMwzBmz55tTJw40VV+7969RrNmzYyHH37Y2Llzp5GWlmZYrVYjIyOjvnahznh6rF544QVj9erVxu7du43t27cbM2bMMLy8vIx///vf9bULdWLatGlGQECAkZWVZRQUFLiWkydPuspMnDjRmD17tuv1hg0bjCZNmhgLFy40du7cacydO9do2rSpsX379vrYhTpVneP11FNPGR999JGxZ88eY8uWLca4ceMMX19fY8eOHfWxC3Vm9uzZxrp164y8vDxj27ZtxuzZsw2LxWJ8/PHHhmFwXp3P02N1tZ5TlbnwKSEznFtXfWA5++jthcvkyZMNwzCMyZMnG4MGDbqoTlRUlOHt7W106tTJeOONN+q83/XB02P13HPPGZ07dzZ8fX2NNm3aGIMHDzbWrl1bP52vQxUdI0lu58mgQYNcx+2sd9991+jatavh7e1tXHfddcb7779ftx2vJ9U5XjNnzjR+8pOfGN7e3kZQUJDxs5/9zNi6dWvdd76O/eY3vzGuvfZaw9vb22jbtq0xbNgw1wewYXBenc/TY3W1nlOVuTCwmOHcshiGYdTe+A0AAMCV4x4WAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgegQWAABgev8fOMJGJNKRa2YAAAAASUVORK5CYII=",
      "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_g(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",
    "        (f(ctx_plt[\"x_0\"]) - f(ctx_plt[\"x_star\"])).le(R, name=\"initial_condition\")\n",
    "    )\n",
    "    x_k = ctx_plt[f\"x_{k}\"]\n",
    "    pb_plt.set_performance_metric((f.grad(x_k)) ** 2)\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 = [2 * L_value / reverse_theta(0, 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": "0c364c9c",
   "metadata": {},
   "source": [
    "## Verification of convergence of OGM-G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "d3bb4440",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.2475766627276036\n"
     ]
    }
   ],
   "source": [
    "N = 2\n",
    "\n",
    "ctx_prf = make_ctx_ogm_g(ctx_name=\"ctx_prf\", N=N, stepsize=1 / L)\n",
    "pb_prf = pf.PEPBuilder(ctx_prf)\n",
    "pb_prf.add_initial_constraint(\n",
    "    (f(ctx_prf[\"x_0\"]) - f(ctx_prf[\"x_star\"])).le(R, name=\"initial_condition\")\n",
    ")\n",
    "pb_prf.set_performance_metric((f.grad(ctx_prf[f\"x_{N}\"])) ** 2)\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": "88088d2d-29f1-4eb2-a2b0-74190cf9ad96",
   "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": "d437bcc3-a0fb-46d7-bb1d-0c862b2d3d90",
   "metadata": {},
   "source": [
    "- It turns out for OGM no further relaxation is needed. Now we store the results."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a955d39f-8464-43d0-9d76-eec29a34c817",
   "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": "2206e8be-1a87-4ee6-84ab-1a3e34dc710d",
   "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.382 & -0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 1.0 & 0.0 \\\\x_2 & 0.134 & 0.618 & 0.0 & 0.248 \\\\x_\\star & 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": [
    "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": "939fc693",
   "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.382 & 0.0 & 0.0 \\\\x_1 & 0.0 & 0.0 & 1.0 & 0.0 \\\\x_2 & 0.134 & 0.618 & 0.0 & 0.248 \\\\x_\\star & 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": [
    "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:  # Additional constraint 1 (between x_★)\n",
    "        if j == 0:\n",
    "            return 1 / reverse_theta(1, N) ** 2 - 2 / reverse_theta(0, N) ** 2\n",
    "        elif j < N:\n",
    "            return 1 / reverse_theta(j + 1, N) ** 2 - 1 / reverse_theta(j, N) ** 2\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 1 / reverse_theta(i + 1, N) ** 2\n",
    "    if i == N and j == N + 1:\n",
    "        return 2 / reverse_theta(0, 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": "7dad45a3",
   "metadata": {},
   "source": [
    "- Check whether our candidate of $\\lambda$ matches with solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "a5dbc3ab",
   "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": "eb1f998f",
   "metadata": {},
   "source": [
    "### Closed form expression of $S$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "7a8d4dab",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|ccccc}\n",
       "         & x_0 & \\nabla f(x_0) & x_\\star & \\nabla f(x_1) & \\nabla f(x_2) \\\\\n",
       "        \\hline\n",
       "        x_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 \\\\x_\\star & -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": [
    "S_sol.pprint()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "84ef3c4f-cdcb-4757-89a1-d5dc65645c0b",
   "metadata": {},
   "source": [
    "- Interestestingly, $S$ has a rank of 0."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "4badd2c0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n"
     ]
    }
   ],
   "source": [
    "rank = np.linalg.matrix_rank(np.round(S_sol.matrix, 3))\n",
    "print(rank)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "789476fb-f26d-4b97-b2eb-9a4828cec781",
   "metadata": {},
   "source": [
    "### Symbolic calculation"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a5778692-3466-4bd3-967d-d7d35962a4f3",
   "metadata": {},
   "source": [
    "- Assemble the RHS of the proof."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "444cfdce-fd24-4988-8b4f-84b7a3e290ed",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+(1/2 + sqrt(5)/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*(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 + (1/2 + sqrt(5)/2)^(-2)*(f(x_0)-f(x_2)+\\nabla f(x_0)*(x_2-x_0)+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_0)\\|^2)+1 - 1/(1/2 + sqrt(5)/2)^2*(f(x_1)-f(x_2)+\\nabla f(x_1)*(x_2-(x_1))+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_1)\\|^2)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(f(x_\\star)-f(x_2)+\\nabla f(x_\\star)*(x_2-x_\\star)+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_\\star)\\|^2)$"
      ],
      "text/plain": [
       "0+(1/2 + sqrt(5)/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*(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 + (1/2 + sqrt(5)/2)**(-2)*(f(x_0)-f(x_2)+grad_f(x_0)*(x_2-x_0)+1/2*L*|grad_f(x_2)-grad_f(x_0)|^2)+1 - 1/(1/2 + sqrt(5)/2)**2*(f(x_1)-f(x_2)+grad_f(x_1)*(x_2-(x_1))+1/2*L*|grad_f(x_2)-grad_f(x_1)|^2)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(f(x_star)-f(x_2)+grad_f(x_star)*(x_2-x_star)+1/2*L*|grad_f(x_2)-grad_f(x_star)|^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": 30,
   "id": "b345c7fa-d127-402e-82f9-ec38f969e4ae",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+(1/2 + sqrt(5)/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*(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 + (1/2 + sqrt(5)/2)^(-2)*(f(x_0)-f(x_2)+\\nabla f(x_0)*(x_2-x_0)+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_0)\\|^2)+1 - 1/(1/2 + sqrt(5)/2)^2*(f(x_1)-f(x_2)+\\nabla f(x_1)*(x_2-(x_1))+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_1)\\|^2)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)^2)/2)^2*(f(x_\\star)-f(x_2)+\\nabla f(x_\\star)*(x_2-x_\\star)+1/2*L*\\|\\nabla f(x_2)-\\nabla f(x_\\star)\\|^2)$"
      ],
      "text/plain": [
       "0+(1/2 + sqrt(5)/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*(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 + (1/2 + sqrt(5)/2)**(-2)*(f(x_0)-f(x_2)+grad_f(x_0)*(x_2-x_0)+1/2*L*|grad_f(x_2)-grad_f(x_0)|^2)+1 - 1/(1/2 + sqrt(5)/2)**2*(f(x_1)-f(x_2)+grad_f(x_1)*(x_2-(x_1))+1/2*L*|grad_f(x_2)-grad_f(x_1)|^2)+2/(1/2 + sqrt(1 + 8*(1/2 + sqrt(5)/2)**2)/2)**2*(f(x_star)-f(x_2)+grad_f(x_star)*(x_2-x_star)+1/2*L*|grad_f(x_2)-grad_f(x_star)|^2)"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "RHS = interp_scalar_sum\n",
    "display(RHS)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8e65ef9f-c9d0-4ab0-8ceb-48ada7858f57",
   "metadata": {},
   "source": [
    "- Assemble the LHS of the proof"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "a02b5e69-1500-427e-944b-fb5e7e0f2fe4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 1/L*\\|\\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))$"
      ],
      "text/plain": [
       "1/L*|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))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x_0 = ctx_prf[\"x_0\"]\n",
    "x_N = ctx_prf[f\"x_{N}\"]\n",
    "x_star = ctx_prf[\"x_star\"]\n",
    "LHS = 1 / L * (f.grad(x_N)) ** 2 - (2 / reverse_theta(0, N) ** 2) * (f(x_0) - f(x_star))\n",
    "display(LHS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "31cca40f",
   "metadata": {},
   "outputs": [],
   "source": [
    "diff = LHS - RHS\n",
    "display(diff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "8a9df5dc-f30c-4289-b1c9-8cd0d464bd70",
   "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
}
