{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "85910674",
   "metadata": {},
   "source": [
    "<a id=\"sec-verif\"></a>\n",
    "# Verifying numerical rates of convergence <span class=\"toc-short\" data-short-title=\"Numerical verification\"></span>"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "15e7fbcb",
   "metadata": {},
   "source": [
    "Let’s explore the first application of PEP: numerically verifying the rate of convergence. As an illustrative example, we consider gradient descent (GD) on $L$-smooth convex functions. We look for the __worst__ function $f$ for which, after $N$ iterations of GD, the objective gap $f(x_N) - f(x_\\star)$ is as large as possible, where $x_\\star$ denotes a minimizer of $f$. In other words, we obtain the __worst-case__ convergence guarantee of GD by solving the following optimization problem \n",
    "\n",
    "```{math} \n",
    "\\begin{equation}\n",
    "\\begin{split}\n",
    "\\text{maximize} \\quad & \\text{some performance metric} && f(x_N) - f(x_\\star) \\\\\n",
    "\\text{subject to} \\quad & \\text{$f$ belongs to some function class} && \\text{$f$ is $L$-smooth convex} \\\\\n",
    "& \\text{$\\{x_k\\}_{k=1}^N$ is generated by an algorithm} \\qquad \\quad && x_{k+1} = x_k - \\alpha \\nabla f(x_k), \\;\\; k=0,\\ldots,N-1 \\\\\n",
    "& \\text{$x_0$ satisfies some initial condition} && \\|x_0 - x_\\star\\|^2 \\leq R^2 \\\\\n",
    "& \\text{$x_\\star$ is a minimizer of $f$} && \\nabla f(x_\\star) = 0.\n",
    "\\end{split} \\qquad \\;\\; \\tag{PEP}\n",
    "\\end{equation}\n",
    "```"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1b7264aa",
   "metadata": {},
   "source": [
    "We start by setting up a `PEPContext` object in <span class=\"brand-color\">PEPFlow</span>. This `PEPContext` object keeps track of all the mathematical objects such as points and scalars involved in the analysis."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fe842bd6",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pepflow as pf\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "ctx = pf.PEPContext(\"gd\").set_as_current()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0457644e-894b-409a-b404-8926799e645b",
   "metadata": {},
   "source": [
    "```{admonition} More on PEPContext\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "The above cell not only creates a new `PEPContext` object called `ctx`, but it also sets the `ctx` object as the current working context. This means that all newly generated `Scalar` and `Vector` objects representing mathematical objects such as scalars and vectors/points will be automatically managed by `ctx`.\n",
    "\n",
    "__NB__: In this tutorial, purple toggle lists provide extra details about the usage of <span class=\"brand-color\">PEPFlow</span>. They are useful for deeper exploration or research, but can be skipped on a first read."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "85f0470e-38c1-4d6d-aad8-c4f0c2a515ca",
   "metadata": {},
   "source": [
    "Then, we declare the four ingredients of algorithm analysis:\n",
    "- <span style=\"font-weight:bold\">function class</span>: $L$-smooth convex functions;\n",
    "- <span style=\"font-weight:bold\">algorithm of interest</span>: gradient descent method;\n",
    "- <span style=\"font-weight:bold\">initial condition</span>: initial point and optimum are not too far away $\\|x_0 - x_\\star\\|^2 \\leq R^2$;\n",
    "- <span style=\"font-weight:bold\">performance metric</span>: objective gap $f(x_k) - f(x_\\star)$.\n",
    "\n",
    "<span class=\"brand-color\">PEPFlow</span> allows us to describe these objects in a high-level way that mirrors the mathematical setup. We first create a `PEPBuilder` object which hold all the information needed to construct and solve PEPs."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "37a80ac1-9701-4ac7-985a-44e51518eb9c",
   "metadata": {},
   "outputs": [],
   "source": [
    "pep_builder = pf.PEPBuilder(ctx)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fc3ef767-1eb9-4540-8ebd-6984e135f1b6",
   "metadata": {},
   "source": [
    "```{admonition} More on PEPBuilder\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "A `PEPBuilder` object requires passing in a `PEPContext` object as an argument. Using the `Scalar` and `Vector` objects managed by the `PEPContext` object, we can define the performance metric and initial condition for the `PEPBuilder` object to store."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dc9e0853-6ac7-4a5f-8c61-c24442c47945",
   "metadata": {},
   "source": [
    "We now define an $L$-smooth convex function $f$ and set a stationary point called `x_star`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "f26f2bc5-f27d-49ad-9c8b-7419a4eb0bef",
   "metadata": {},
   "outputs": [],
   "source": [
    "L = pf.Parameter(\"L\")\n",
    "\n",
    "# Define function class\n",
    "f = pf.SmoothConvexFunction(is_basis=True, tags=[\"f\"], L=L)\n",
    "x_star = f.set_stationary_point(\"x_star\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "01961c0a-a12d-4878-a864-48610b476355",
   "metadata": {},
   "source": [
    "```{admonition} More on defining Function objects\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "<span class=\"brand-color\">PEPFlow</span> offers several types of functions classes such as `SmoothConvexFunction` and `ConvexFunction`. The `is_basis` argument takes a `boolean` that designates whether the created `Function` object is formed as a linear combination of other `Function` objects. The `tags` argument takes in a `list` of strings which can be used to access the `Function` object through the method `get_func_or_oper_by_tag`. The last element of the `tags` list will be used to display the `Function` object."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d78e6025-d122-4d3d-910e-596ae564a39f",
   "metadata": {},
   "source": [
    "```{admonition} More on Parameter objects\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "The `Parameter` objects can be thought of as mathematical variables. As we will see later on in the tutorial, when we perform specific operations such as solving the PEP problem, we will need to supply explicit values for the `Parameter` object."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1645b7b6-937d-484a-8e54-c0ce25da38ed",
   "metadata": {},
   "source": [
    "```{admonition} More on tags\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "Mathematical objects in <span class=\"brand-color\">PEPFlow</span> contain a list of user-provided strings called `tags`. These tags are used to access the mathematical objects easily and to display the mathematical object cleanly. "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "29d5e3bf-200f-4e4a-9e75-5a793d0fac65",
   "metadata": {},
   "source": [
    "Now, we declare the initial condition $\\lVert x_0 - x_\\star \\rVert^2 \\leq R^2$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "5f478d05-a291-4c16-b635-f062e5e0db17",
   "metadata": {},
   "outputs": [],
   "source": [
    "R = pf.Parameter(\"R\")\n",
    "\n",
    "# Set the initial condition\n",
    "x = pf.Vector(is_basis=True, tags=[\"x_0\"])  # The first iterate\n",
    "pep_builder.add_initial_constraint(\n",
    "    ((x - x_star) ** 2).le(R**2, name=\"initial_condition\")\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fe52f8f3-7602-4066-974b-3e99b4e14727",
   "metadata": {},
   "source": [
    "```{admonition} More on Vector objects\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "A `Vector` object represents an element of a pre-Hilbert space and is one of the key fundamental mathematical objects that <span class=\"brand-color\">PEPFlow</span> provides. The `is_basis` argument takes a `boolean` that designates whether the created `Vector` object is formed as a linear combination of other `Vector` objects. The `tags` argument takes in a `list` of strings which can be used to access the `Vector` object easily from a context that manages it. The last element of the `tags` list will be used to display the `Vector` object."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d31fa990-85e7-4757-82a2-2b79bcd996e3",
   "metadata": {},
   "source": [
    "```{admonition} More on ScalarConstraint objects\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "The code `((x - x_star) ** 2).le(R**2, name=\"initial_condition\")` creates a `ScalarConstraint` object which represent a scalar constraint used in the PEP. A `ScalarConstraint` object requires 4 components: \n",
    "- a `Scalar` object for the left hand side of the constraint;\n",
    "- a `Scalar` object for the right hand side of the constraint;\n",
    "- the type of constraint, _e.g._, (`le`, `geq`, `eq`);\n",
    "- a user-provided name for the constraint."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bf4b273d-4b99-413c-a68c-379af39e47a4",
   "metadata": {},
   "source": [
    "Next, we define generate the GD iterates $\\{x_i\\}_{i=1}^{N}$. Note that for each iterate $x_i$ we add a tag to display it nicely and for easy access from the `PEPContext` object which tracks it. Furthermore, the `Vector` object representing $\\nabla f(x)$ can be generated by calling `f.grad(x)`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d4b63383-2931-4aee-a076-c85bada9dc3a",
   "metadata": {},
   "outputs": [],
   "source": [
    "N = 8\n",
    "alpha = 1 / L\n",
    "\n",
    "# Define the gradient descent method\n",
    "for i in range(N):\n",
    "    x = x - alpha * f.grad(x)\n",
    "    x.add_tag(f\"x_{i + 1}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b7886abe-3373-4b33-bb7c-5bc719a462fa",
   "metadata": {},
   "source": [
    "```{admonition} Tags for composite Vectors and Scalars\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "`Vector` and `Scalar` objects formed as linear combinations of basis objects do not have any tags. The GD iterates generated by the line of code `x = x - alpha * f.grad(x)` are examples of composite `Vector` objects. It is important for the user to give these composite objects tags to access these objects easily from the `PEPContext` that manages them."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f0278a6b-2e2a-459b-8af3-bbd0bc30b739",
   "metadata": {},
   "source": [
    "Finally, we set the performance metric $f(x_k) - f(x_\\star)$. Observe that the `Scalar` object representing $f(x)$ can be generated by calling `f(x)`. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a9780ffa-0181-4c86-a9fb-9f1b1b1671c3",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set the performance metric\n",
    "x_N = ctx[f\"x_{N}\"]\n",
    "pep_builder.set_performance_metric(f(x_N) - f(x_star))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7cec3873-ee63-43f9-841b-730c7a45b3a2",
   "metadata": {},
   "source": [
    "```{admonition} Accessing Vector and Scalar objects from a PEPContext\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "The `Vector` and `Scalar` objects managed by a `PEPContext` object can be easily accessed through their tags. In the prior code block, we returned the iterate $x_N$ represented by a `Vector` object with the tag `x_N` managed by the `PEPContext` object `ctx` through the code: `ctx[f\"x_{N}\"]`. "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c2b7e7df-a953-4acf-88ba-2000ba43a61c",
   "metadata": {},
   "source": [
    "Solving the PEP numerically gives the worst-case value of the chosen performance metric. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "dc990b5b-688d-4627-972f-9e86e1fea9a1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "primal PEP optimal value = 0.0294\n"
     ]
    }
   ],
   "source": [
    "L_value = 1\n",
    "R_value = 1\n",
    "result = pep_builder.solve(resolve_parameters={\"L\": L_value, \"R\": R_value})\n",
    "print(f\"primal PEP optimal value = {result.opt_value:.4f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6573f2f6-97d8-4094-a5a9-5ea3acae4125",
   "metadata": {},
   "source": [
    "```{admonition} Resolving Parameter objects\n",
    ":class: dropdown pepflow-dropdown\n",
    "\n",
    "To solve the PEP problem represented by the `pep_builder` object, we need to resolve the `Parameter` objects `L` and `R` with concrete numerical values. To do this, we pass in a dictionary of the form: `{name_of_parameter: concrete_value}`."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1ea48571-aad0-4faa-a986-39a45879c0e7",
   "metadata": {},
   "source": [
    "In this example, the numerical result can be interpreted as: for all $1$-smooth convex functions $f$, $8$-step GD starting at $x_0$ with $\\|x_0 - x_\\star\\|^2 \\leq 1$ satisfies\n",
    "\n",
    "$$f(x_8) - f(x_\\star) \\leq 0.0294.$$\n",
    "\n",
    "To better understand how the error decreases with the number of iterations, we repeat the process for each $k = 1, 2, \\dots, 7$. This gives us a sequence of numerical values that can be compared to the known analytical rate."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "9740742e-b4b9-42db-9693-2604d5f3e309",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "opt_values = []\n",
    "for k in range(1, N):\n",
    "    x_k = ctx[f\"x_{k}\"]\n",
    "    pep_builder.set_performance_metric(f(x_k) - f(x_star))\n",
    "    result = pep_builder.solve(resolve_parameters={\"L\": L_value, \"R\": R_value})\n",
    "    opt_values.append(result.opt_value)\n",
    "\n",
    "plt.scatter(range(1, N), opt_values, color=\"blue\", marker=\"o\");"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4bde9496-7745-4b2a-8ee3-142c821d828e",
   "metadata": {},
   "source": [
    "The resulting values are visualized to show the trend of convergence. Each scatter point represents a guaranteed bound that holds for all $L$-smooth convex functions, while the continuous curve shows the known analytical rate for comparison (see, _e.g._, [^1]).\n",
    "\n",
    "After experimenting with different $L$, $R$, and $N$, we tend to draw the conclusion: for all $L$-smooth convex functions $f$, $N$-step GD with step size $\\alpha = 1/L$ satisfies\n",
    "\n",
    "```{math}\n",
    "f(x_N) - f(x_\\star) \\leq \\frac{L}{4N+2} \\|x_0 - x_\\star\\|_2^2.\n",
    "```"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "20115764-20fa-4f65-b59f-59acca09a066",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiwAAAGdCAYAAAAxCSikAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjMsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvZiW1igAAAAlwSFlzAAAPYQAAD2EBqD+naQAAUnNJREFUeJzt3XlcVPX+x/HXACIugDtuKJq5L7ggorklRV71apqZmXubWWqUpd2r1m1Ry0pLb6bXa2a5tKh5NbfcM80tzC23XAjFLQNFBWXO74/zY3AUlEHgDPB+Ph7nMYczZ858htJ5+z3fxWYYhoGIiIiIG/OwugARERGRO1FgEREREbenwCIiIiJuT4FFRERE3J4Ci4iIiLg9BRYRERFxewosIiIi4vYUWERERMTteVldQFax2+2cPHkSX19fbDab1eWIiIhIBhiGwcWLFylfvjweHum3o+SZwHLy5EkCAwOtLkNEREQyITo6mooVK6b7fJ4JLL6+voD5gf38/CyuRkRERDIiPj6ewMBAx/d4evJMYEm5DeTn56fAIiIiksvcqTuHOt2KiIiI21NgEREREbenwCIiIiJuL8/0YRERsYphGFy/fp3k5GSrSxFxO56ennh5ed31lCMKLCIidyEpKYlTp05x+fJlq0sRcVuFCxemXLlyeHt7Z/oaCiwiIplkt9s5evQonp6elC9fHm9vb01cKXIDwzBISkri7NmzHD16lHvvvfe2k8PdjgKLiEgmJSUlYbfbCQwMpHDhwlaXI+KWChUqRIECBTh+/DhJSUn4+Phk6jrqdCsicpcy+y9GkfwiK/6MqIXlNpKTYeNGOHUKypWDli3B09PqqkRERPIfBZZ0LFgAQ4fCH3+kHqtYESZNgq5dratLREQkP1I7ZhoWLIBHHnEOKwAxMebxBQusqUtERCS/UmC5SXKy2bJiGLc+l3Js2DDzPBERkbzi4Ycfpnjx4jzyyCNWl5ImBZabbNyY2rLiTSJDmMRcHsOLa4AZWqKjzfNERETyiqFDh/L5559bXUa6FFhucupU6v41CjCaf/EY82nMjnTPExERye3atGmDr6+v1WWkS4HlJuXKpe4beLCBVgC0Zn2654mISPratGnDsGHD3OY6Gb1eVr9fVnP3+rKaAstNWrY0RwOlTFa5ntYAtGIDYB4PDDTPExHJCzZv3oynpycdOnSwuhQg/S/iBQsW8Oabb+Z8QeIWFFhu4ulpDl0GM5yktLDcx494Yva0nThR87GISN4xY8YMXnjhBTZs2MDJkyetLiddJUqUcOtbFpK9FFjS0LUrfPMNVKgAu2jAX/jjTzwRAVF8843mYRGRvOPSpUvMnz+fQYMG0aFDBz777DOn59u0acOQIUN45ZVXKFGiBGXLluX11193Omf58uXcd999FCtWjJIlS9KxY0eOHDmS5vt9/vnnlCxZksTERKfjXbp0oXfv3vTr14/169czadIkbDYbNpuNY8eOOWq5seXFbrfz7rvvUq1aNQoWLEilSpV4++23Xa7pdq5fv87zzz+Pv78/pUqVYtSoURg3DCNNTExkyJAhlClTBh8fH+677z62bdvmeD4oKIiJEyc6XTM4ONjpd5iR33FCQgJ9+vShaNGilCtXjvfff9/lz5LbKbCko2tXOHYMVq/1JKHhfQD87+X1Cisikqd89dVX1KxZkxo1avDEE0/w3//+1+kLGWDWrFkUKVKEn3/+mXfffZd//etfrFq1yvF8QkICkZGRbN++ndWrV+Ph4cHDDz+M3W6/5f26d+9OcnIyixcvdhw7c+YMS5cuZcCAAUyaNImwsDCeeuopTp06xalTpwgMDEyz9pEjRzJu3DhGjRrFvn37mDNnDgEBAS7XdDuzZs3Cy8uLrVu3MmnSJD744AP+85//OJ5/5ZVX+Pbbb5k1axY7d+6kWrVqRERE8Oeff7r8Prf7HQ8fPpz169fz3XffsXLlStatW8fOnTtdeo87CQ8Pp3v37nz//fdUrFiRzZs3Z+n175qRR8TFxRmAERcXl/UXf/ddwwDD6Nw5668tIrnWlStXjH379hlXrlxJPWi3G8alSzm/2e2Z+gzNmzc3Jk6caBiGYVy7ds0oVaqUsXbtWsfzrVu3Nu677z6n14SEhBivvvpqutc8e/asARi7d+92XGPo0KGO5wcNGmS0b9/e8fP7779vVK1a1bD//2e4+fwba0k5Hh8fbxQsWNCYPn16hj7nzTXd7n1ufL5WrVqOugzDMF599VWjVq1ahmEYxqVLl4wCBQoYX375peP5pKQko3z58sa7775rGIZhVK5c2fjwww+drtugQQNjzJgxTu9zu9/xxYsXDW9vb+Orr75yPH/+/HmjUKFCt63fnaT5Z+X/ZfT7Wy0sGdHK7MfCxo3gYjoXkXzm8mUoWjTnt8uXXS71wIEDbN26lZ49ewLg5eVFjx49mDFjhtN59evXd/q5XLlynDlzxvHzoUOH6NmzJ1WrVsXPz4+goCAATpw4keb7PvXUU6xcuZKYmBgAPvvsM/r164ctZbRDBuzfv5/ExETatWuX5vOu1pSeZs2aOdUVFhbGoUOHSE5O5siRI1y7do0WLVo4ni9QoABNmzZl//79Lr3P7X7HR44cISkpidDQUMfzJUqUoEaNGi69R26ntYQyolEjKFIE/vwT9uyBm/7HEhHJjWbMmMH169cpX76845hhGBQsWJDJkyfj7+8PmF/CN7LZbE63Vjp16kTlypWZPn065cuXx263U7duXZKSktJ834YNG9KgQQM+//xzHnzwQfbu3cvSpUtdqr1QoUK3fd7VmrKLh4fHLbfYrl27dst5d/odu8qV8Jcdbv7MWUEtLBlRoACkJOj1629/rojkb4ULw6VLOb8VLuxSmdevX+fzzz/n/fffJyoqyrHt2rWL8uXLM3fu3Axd5/z58xw4cIB//vOftGvXjlq1anHhwoU7vu7JJ5/ks88+Y+bMmYSHhzv1U/H29ib5Duuf3HvvvRQqVIjVq1dnWU1p+fnnn51+3rJlC/feey+enp7cc889eHt7s2nTJsfz165dY9u2bdSuXRuA0qVLc+qGmUbj4+M5evSoSzXcc889FChQwKmWCxcucPDgwXRfYxiGpVt2UGDJqNbmfCxs2GBtHSLi3mw2s0U2pzcX/0W9ZMkSLly4wMCBA6lbt67T1q1bt1tuC6WnePHilCxZkmnTpnH48GHWrFlDZGTkHV/3+OOP88cffzB9+nQGDBjg9FxQUBA///wzx44d49y5c2m2NPj4+PDqq6/yyiuv8Pnnn3PkyBG2bNnCjBkzMl1TWk6cOEFkZCQHDhxg7ty5fPzxxwwdOhSAIkWKMGjQIIYPH87y5cvZt28fTz31FJcvX2bgwIEA3H///cyePZuNGzeye/du+vbti6eL82IULVqUgQMHMnz4cNasWcOePXvo168fHh53/gq/fPkylStX5uWXX3Yc++677+56wrno6GjatGlD7dq1qV+/Pl9//fVdXS8jdEsoo1L6sWzYYC4oZHFzm4jI3ZgxYwbh4eGO2z436tatG++++y6//vrrHa/j4eHBvHnzGDJkCHXr1qVGjRp89NFHtGnT5rav8/f3p1u3bixdupQuXbo4Pffyyy/Tt29fateuzZUrVzh69KijD8qNRo0ahZeXF6NHj+bkyZOUK1eOZ599NtM1paVPnz5cuXKFpk2b4unpydChQ3n66acdz48bNw673U7v3r25ePEiTZo0YcWKFRQvXhwwRzIdPXqUjh074u/vz5tvvulyCwvAe++9x6VLl+jUqRO+vr689NJLxMXF3fF1b7/9Ns2aNXM69uuvv9KgQYM7vjZliHu/fv1uec7Ly4uJEycSHBxMbGwsjRs35m9/+xtFihTJ0OfJDJuRXW03OSw+Ph5/f3/i4uLw8/PL+jdITIRixeDqVdi3D2rVyvr3EJFc5erVqxw9epQqVarg4+NjdTm5Trt27ahTpw4fffSR1aXkSYcOHWLEiBF06tSJPXv2MGHCBMAcWj5y5EiqVatG79696dChg1MIS3G7wHKzBg0asGTJknSHoN/uz0pGv78zdUtoypQpBAUF4ePjQ2hoKFu3bk333L1799KtWzeCgoKw2Wy3TKCTIiYmhieeeIKSJUtSqFAh6tWrx/bt2zNTXvYoWBDCwsz9dessLUVEJDe7cOECCxcuZN26dQwePNjqcvKsl19+mbFjx95yfP/+/RQqVIj27dvz/PPPpxlWXLFjxw6Sk5PTDStZxeVbQvPnzycyMpKpU6cSGhrKxIkTiYiI4MCBA5QpU+aW8y9fvkzVqlXp3r07L774YprXvHDhAi1atKBt27YsW7aM0qVLc+jQIUeTmtto2xbWroU1a2DQIKurERHJlRo2bMiFCxcYP358vhuam1O+++47qlevTvXq1fnpp58cx69cuUJMTAyPP/44X3zxBXXq1HF6XVJSEk2bNgVwTH6X0tCwdetWvL29nc7/888/6dOnD9OnT8/GT2NyObB88MEHPPXUU/Tv3x+AqVOnsnTpUv773/8yYsSIW84PCQkhJCQEIM3nAcaPH09gYCAzZ850HKtSpYqrpWW/du1g9GgztNjtkIEOTyIi4ixlqn3JPlu2bGHevHl8/fXXXLp0iWvXruHn50f79u0JCwsjJiYGL69bI4C3tzdRUVHAnW8JJSYm0qVLF0aMGEHz5s2z6ZOkcukbNykpiR07dhAeHp56AQ8PwsPD72oK38WLF9OkSRO6d+9OmTJlaNiw4R3TWmJiIvHx8U5btgsJMSdoOn8edu3K/vcTERHJhLFjxxIdHc2xY8eYMGECTz31FKNHj+bXX3+lZcuWzJw5k8cff5xLly5l6vqGYdCvXz/uv/9+evfuncXVp82lwHLu3DmSk5MdazWkCAgIIDY2NtNF/P7773zyySfce++9rFixgkGDBjFkyBBmzZqV7mvGjh2Lv7+/Y8vue2eAOR9LyvDmNWuy//1ERESy0K+//krdunVp1KgRzz333C1DyjNq06ZNzJ8/n0WLFhEcHExwcDC7d+/O4mqducWwZrvdTpMmTXjnnXcA8/7mnj17mDp1Kn379k3zNSNHjnQaVx8fH58zoeX++2HpUli9Gl56KfvfT0RE5C7ceEtn0qRJjv2BAwc65ou50+tudt99993VTLyZ4VILS6lSpfD09OT06dNOx0+fPk3ZsmUzXUS5cuUcswKmqFWr1m3XfChYsCB+fn5OW45IWbdiwwbI4SmeRURE8iuXAou3tzeNGzd2mgrZbrezevVqwlKG/GZCixYtOHDggNOxgwcPUrly5UxfM9vUqwelSkFCAtxmOLeIiIhkHZeHuURGRjJ9+nRmzZrF/v37GTRoEAkJCY5RQ3369GHkyJGO85OSkhxrVCQlJRETE0NUVBSHDx92nPPiiy+yZcsW3nnnHQ4fPsycOXOYNm2ae47P9/AwbwuBeVtIREREsp3LgaVHjx5MmDCB0aNHExwcTFRUFMuXL3d0xD1x4oTTQk8nT56kYcOGNGzYkFOnTjFhwgQaNmzIk08+6TgnJCSEhQsXMnfuXOrWrcubb77JxIkT6dWrVxZ8xGyQcltIgUVERCRHaGr+zDhyBKpVM0cNXbhgLjwmIvmOpuYXyRjLpubP96pWhUqV4No1+PFHq6sRERHJ8xRYMsNmS70t9MMP1tYiIiKSDyiwZFbKbL8rV1pbh4iISD6gwJJZDzxgtrT8+ivc0MlYREREsp4CS2aVLg2NG5v7amUREckxQUFBjhWEs0KbNm0YNmxYll0vLf369aNLly7Z+h55nQLL3YiIMB9XrLC2DhERF/Xr1w+bzca4ceOcji9atAibzWZRVRmzbds2nn76aavLkBymwHI3UgLLypWQnGxtLSIiLvLx8WH8+PFcuHDB6lIyJOn/l0MpXbo0hQsXtrgayWkKLHejWTPw9YXz52HnTqurEZFcLDkZ1q2DuXPNx5z4N1B4eDhly5Zl7Nix6Z7z+uuvExwc7HRs4sSJBAUFOX5Oud3xzjvvEBAQQLFixfjXv/7F9evXGT58OCVKlKBixYrMnDnT6TrR0dE8+uijFCtWjBIlStC5c2eOHTt2y3XffvttypcvT40aNYBbbwn99ddfPPPMMwQEBODj40PdunVZsmQJAOfPn6dnz55UqFCBwoULU69ePebOnZvh39HBgwex2Wz89ttvTsc//PBD7rnnHgCSk5MZOHAgVapUoVChQtSoUcNpkcG0pHVbKzg4mNdff93pcz355JOULl0aPz8/7r//fnbt2uV4fteuXbRt2xZfX1/8/Pxo3Lgx27dvz/Bny20UWO5GgQKpo4V0W0hEMmnBAggKgrZt4fHHzcegIPN4dvL09OSdd97h448/5o8//rira61Zs4aTJ0+yYcMGPvjgA8aMGUPHjh0pXrw4P//8M88++yzPPPOM432uXbtGREQEvr6+bNy4kU2bNlG0aFEeeughR0sKwOrVqzlw4ACrVq1yhJAb2e122rdvz6ZNm/jiiy/Yt28f48aNw9PTEzAnLGvcuDFLly5lz549PP300/Tu3ZutGVwLrnr16jRp0oQvv/zS6fiXX37J448/7qihYsWKfP311+zbt4/Ro0fz2muv8dVXX2Xqd5mie/funDlzhmXLlrFjxw4aNWpEu3bt+PPPPwHo1asXFStWZNu2bezYsYMRI0ZQoECBu3pPt2bkEXFxcQZgxMXF5ewbT51qGGAY992Xs+8rIpa7cuWKsW/fPuPKlSuZvsa33xqGzWb+NXLjZrOZ27ffZmHBN+jbt6/RuXNnwzAMo1mzZsaAAQMMwzCMhQsXGjd+NYwZM8Zo0KCB02s//PBDo3Llyk7Xqly5spGcnOw4VqNGDaNly5aOn69fv24UKVLEmDt3rmEYhjF79myjRo0aht1ud5yTmJhoFCpUyFixYoXjugEBAUZiYqLT+1euXNn48MMPDcMwjBUrVhgeHh7GgQMHMvzZO3ToYLz00kuOn1u3bm0MHTo03fM//PBD45577nH8fODAAQMw9u/fn+5rBg8ebHTr1s3x842/75s/Q4oGDRoYY8aMMQzDMDZu3Gj4+fkZV69edTrnnnvuMT799FPDMAzD19fX+Oyzz9KtwZ3c7s9KRr+/1cJyt1L6sWzeDHFx1tYiIrlKcjIMHWpGlJulHBs2LPtvD40fP96xoG1m1alTBw+P1K+UgIAA6tWr5/jZ09OTkiVLcubMGcC8nXH48GF8fX0pWrQoRYsWpUSJEly9epUjR444XlevXj28vb3Tfd+oqCgqVqxI9erV03w+OTmZN998k3r16lGiRAmKFi3KihUrOHHiRIY/22OPPcaxY8fYsmULYLauNGrUiJo1azrOmTJlCo0bN6Z06dIULVqUadOmufQeN9u1axeXLl2iZMmSjt9P0aJFOXr0qOP3ExkZyZNPPkl4eDjjxo1z+r3lRQosdysoCKpXN/9G0WKIIuKCjRvhdndiDAOio83zslOrVq2IiIhg5MiRtzzn4eGBcVOiunbt2i3n3XwrwmazpXnMbrcDcOnSJRo3bkxUVJTTdvDgQcetFoAid1irrVChQrd9/r333mPSpEm8+uqrrF27lqioKCIiIpxuO91J2bJluf/++5kzZw4Ac+bMcVqcd968ebz88ssMHDiQlStXEhUVRf/+/W/7Hnf6vV66dIly5crd8vs5cOAAw4cPB8z+RXv37qVDhw6sWbOG2rVrs3Dhwgx/rtzGy+oC8oSICDh40OzH0rWr1dWISC6R0Tknc2JuynHjxhEcHOzo2JqidOnSxMbGYhiGY7hzVFTUXb9fo0aNmD9/PmXKlLmrBWvr16/PH3/8wcGDB9NsZdm0aROdO3fmiSeeAMz+JgcPHqR27douvU+vXr145ZVX6NmzJ7///juPPfaY03s0b96c5557znHsTq0dpUuX5tQN/2Hj4+M5evSo4+dGjRoRGxuLl5eXUwfnm1WvXp3q1avz4osv0rNnT2bOnMnDDz/s0mfLLdTCkhUeesh8XLYs7bZdEZE0lCuXtefdjXr16tGrVy8++ugjp+Nt2rTh7NmzvPvuuxw5coQpU6awbNmyu36/Xr16UapUKTp37szGjRs5evQo69atY8iQIS51AG7dujWtWrWiW7durFq1iqNHj7Js2TKWL18OwL333suqVav46aef2L9/P8888wynT592ud6uXbty8eJFBg0aRNu2bSlfvrzjuXvvvZft27ezYsUKDh48yKhRo9i2bdttr3f//fcze/ZsNm7cyO7du+nbt6+jozCYI7jCwsLo0qULK1eu5NixY/z000/84x//YPv27Vy5coXnn3+edevWcfz4cTZt2sS2bduoVauWy58tt1BgyQpt2oCPj9l2u2eP1dWISC7RsiVUrGiu8pEWmw0C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eXPmq71JXFwcACVKlEj3nMTEROLj4522fMHHB4YONffffVcTyYmISL7gUmA5d+4cycnJBAQEOB0PCAggNjY2Swqy2+0MGzaMFi1aULdu3XTPGzt2LP7+/o4tMDAwS94/Vxg0CIoWNSeSW7bM6mpERESynduNEho8eDB79uxh3rx5tz1v5MiRxMXFObbo6OgcqtANFC8Ozzxj7o8fb20tIiIiOcClwFKqVCk8PT05ffq00/HTp0+n26HWFc8//zxLlixh7dq1VKxY8bbnFixYED8/P6ctX3nxRShQADZsgC1brK5GREQkW7kUWLy9vWncuDGrV692HLPb7axevZqwsLBMF2EYBs8//zwLFy5kzZo1VKlSJdPXyjcqVEjtcKtWFhERyeNcviUUGRnJ9OnTmTVrFvv372fQoEEkJCTQv39/APr06cPIkSMd5yclJREVFUVUVBRJSUnExMQQFRXF4cOHHecMHjyYL774gjlz5uDr60tsbCyxsbFcuXIlCz5iHjZ8uPn43Xfw22/W1iIiIpKNXB7WDDB58mTee+89YmNjCQ4O5qOPPiI0NBSANm3aEBQUxGeffQbAsWPH0mwxad26NevWrTOLsNnSfJ+ZM2fSL2XRvzvIN8Oab9alixlYBgzQZHIiIpLrZPT7O1OBxR3l28CyeTM0bw5eXnD4sDlHi4iISC6RLfOwiBsKC4N27eD6dRg3zupqREREsoUCS14wZoz5OGMG5Kfh3SIikm8osOQFLVtCmzbm2kIaMSQiInmQAktekdLKMn06xMRYW4uIiEgWU2DJK1q3NltakpLMNYZERETyEAWWvMJmS21lmTYNTp2yth4REZEspMCSl9x/vznE+epVeO89q6sRERHJMgoseYnNBqNHm/tTp8JNaz6JiIjkVgosec2DD0JoKFy5AmPHWl2NiIhIllBgyWtsNnjrLXP/k0/g+HFr6xEREckCCix5Ubt20LatOWLojTesrkZEROSuKbDkRTYbvPOOuT9rFuzfb209IiIid0mBJa9q1gw6dwa7PbUjroiISC6lwJKXvfWW2dryzTewY4fV1YiIiGSaAkteVrcu9Opl7r/2mrW1iIiI3AUFlrzujTfAywtWroR166yuRkREJFMUWPK6qlXh6afN/REjwDCsrUdERCQTFFjyg3/+EwoXhp9/hq+/troaERERlymw5AflysEr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ixYtaNu2LcuWLaN06dIcOnSI4sWLW11ajtm2bRvJycmOn/fs2cMDDzxA9+7dLawq54wfP55PPvmEWbNmUadOHbZv307//v3x9/dnyJAhOVpL/vmT56L27dvTvn17q8uwzPLly51+/uyzzyhTpgw7duygVatWFlWVszp16uT089tvv80nn3zCli1b8lVguXTpEr169WL69Om89dZbVpeT47y8vChbtqzVZVhi/PjxBAYGMnPmTMexKlWqWFhRzitdurTTz+PGjeOee+6hdevWFlWUs3766Sc6d+5Mhw4dALPFae7cuWzdujXHa9EtIcmQuLg4AEqUKGFxJdZITk5m3rx5JCQkEBYWZnU5OWrw4MF06NCB8PBwq0uxxKFDhyhfvjxVq1alV69enDhxwuqScszixYtp0qQJ3bt3p0yZMjRs2JDp06dbXZZlkpKS+OKLLxgwYECWLrLrzpo3b87q1as5ePAgALt27eLHH3+05B/0amGRO7Lb7QwbNowWLVpQt25dq8vJUbt37yYsLIyrV69StGhRFi5cSO3ata0uK8fMmzePnTt3sm3bNqtLsURoaCifffYZNWrU4NSpU7zxxhu0bNmSPXv24Ovra3V52e7333/nk08+ITIyktdee41t27YxZMgQvL296du3r9Xl5bhFixbx119/0a9fP6tLyTEjRowgPj6emjVr4unpSXJyMm+//Ta9evXK8VoUWOSOBg8ezJ49e/jxxx+tLiXH1ahRg6ioKOLi4vjmm2/o27cv69evzxehJTo6mqFDh7Jq1Sp8fHysLscSN/4rsn79+oSGhlK5cmW++uorBg4caGFlOcNut9OkSRPeeecdABo2bMiePXuYOnVqvgwsM2bMoH379pQvX97qUnLMV199xZdffsmcOXOoU6cOUVFRDBs2jPLly+f4/wMKLHJbzz//PEuWLGHDhg1UrFjR6nJynLe3N9WqVQOgcePGbNu2jUmTJvHpp59aXFn227FjB2fOnKFRo0aOY8nJyWzYsIHJkyeTmJiIp6enhRXmvGLFilG9enUOHz5sdSk5oly5creE81q1avHtt99aVJF1jh8/zg8//MCCBQusLiVHDR8+nBEjRvDYY48BUK9ePY4fP87YsWMVWMQ9GIbBCy+8wMKFC1m3bl2+62iXHrvdTmJiotVl5Ih27dqxe/dup2P9+/enZs2avPrqq/kurIDZAfnIkSP07t3b6lJyRIsWLW6ZzuDgwYNUrlzZooqsM3PmTMqUKePofJpfXL58GQ8P5+6unp6e2O32HK9FgSUdly5dcvpX1NGjR4mKiqJEiRJUqlTJwspyxuDBg5kzZw7fffcdvr6+xMbGAuDv70+hQoUsri5njBw5kvbt21OpUiUuXrzInDlzWLduHStWrLC6tBzh6+t7S5+lIkWKULJkyXzTl+nll1+mU6dOVK5cmZMnTzJmzBg8PT3p2bOn1aXliBdffJHmzZvzzjvv8Oijj7J161amTZvGtGnTrC4tR9ntdmbOnEnfvn3z1bB2MEdLvv3221SqVIk6derwyy+/8MEHHzBgwICcL8aQNK1du9YAbtn69u1rdWk5Iq3PDhgzZ860urQcM2DAAKNy5cqGt7e3Ubp0aaNdu3bGypUrrS7LUq1btzaGDh1qdRk5pkePHka5cuUMb29vo0KFCkaPHj2Mw4cPW11Wjvrf//5n1K1b1yhYsKBRs2ZNY9q0aVaXlONWrFhhAMaBAwesLiXHxcfHG0OHDjUqVapk+Pj4GFWrVjX+8Y9/GImJiTlei80wLJiuTkRERMQFmodFRERE3J4Ci4iIiLg9BRYRERFxewosIiIi4vYUWERERMTtKbCIiIiI21NgEREREbenwCIiIiJuT4FFRERE3J4Ci4iIiLg9BRYRERFxewosIiIi4vb+D/z3u0oDQnHYAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "iters = np.arange(1, N)\n",
    "cont_iters = np.arange(1, N, 0.01)\n",
    "plt.plot(\n",
    "    cont_iters,\n",
    "    1 / (4 * cont_iters + 2),\n",
    "    \"r-\",\n",
    "    label=\"Analytical bound $\\\\frac{1}{4k+2}$\",\n",
    ")\n",
    "plt.scatter(iters, opt_values, color=\"blue\", marker=\"o\", label=\"Numerical values\")\n",
    "plt.legend();"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2aca018d-b8b5-423a-ad0e-3c4338a81401",
   "metadata": {},
   "source": [
    "[^1]: Y. Drori and M. Teboulle. Performance of first-order methods for smooth convex minimization: a novel approach. _Mathematical Programming_, 145(1-2):451–482, 2014."
   ]
  }
 ],
 "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
}
