{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "8baf79c6",
   "metadata": {},
   "source": [
    "# DRS Example\n",
    "\n",
    "This code tests the proof of Douglas--Rachford splitting methods, introduced in Proposition 3.1 of\n",
    "\"Exact worst-case convergence rates for Douglas--Rachford and Davis--Yin splitting methods\"\n",
    "by Edward Duc Hien Nguyen, Jaewook J. Suh, Xin Jiang, Shiqian Ma (2025).\n",
    "\n",
    "The implementation in terms of the PEP is based on the PDHG implementation done in the paper \n",
    "\"Interpolation Conditions for Linear Operators and Applications to Performance Estimation\n",
    "Problems\" by Nizar Bousselmi, Julien M. Hendrickx, François Glineur (2023)."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e16d420d-b93c-4972-a50f-022ad0683d12",
   "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": "78c56a55",
   "metadata": {},
   "source": [
    "## Define the functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c58e9735-cda7-49ff-a6c2-3bdaa335f4ae",
   "metadata": {},
   "outputs": [],
   "source": [
    "f = pf.ConvexFunction(is_basis=True, tags=[\"f\"])\n",
    "g = pf.ConvexFunction(is_basis=True, tags=[\"g\"])\n",
    "h = (f + g).add_tag(\"h\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7a618c96-9efa-46b7-a452-0522e08a2875",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPContext associated with DRS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "673641da",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_ctx_drs(\n",
    "    ctx_name: str, N: int | sp.Integer, stepsize: pf.Parameter\n",
    ") -> pf.PEPContext:\n",
    "    ctx_drs = pf.PEPContext(ctx_name).set_as_current()\n",
    "\n",
    "    # Declare the initial points.\n",
    "    x_0 = pf.Vector(is_basis=True, tags=[\"x_0\"])\n",
    "    u_0 = pf.Vector(is_basis=True, tags=[\"u_0\"])\n",
    "\n",
    "    # Declare the points used in the primal-dual gap function.\n",
    "    _ = pf.Vector(is_basis=True, tags=[\"x_{{tilde}}\"])\n",
    "    u_tilde = pf.Vector(is_basis=True, tags=[\"u_{{tilde}}\"])\n",
    "\n",
    "    # Moving Average\n",
    "    x_sum = pf.Vector.zero()\n",
    "    u_sum = pf.Vector.zero()\n",
    "\n",
    "    x = x_0\n",
    "    u = u_0\n",
    "\n",
    "    for i in range(N):\n",
    "        x_old = x\n",
    "\n",
    "        x = f.prox(x - stepsize * u, stepsize, tag=f\"x_{i + 1}\")\n",
    "\n",
    "        t = u + 1 / stepsize * (2 * x - x_old)\n",
    "        p = g.prox(stepsize * t, stepsize, tag=f\"p_{i + 1}\")\n",
    "\n",
    "        u = t - 1 / stepsize * p\n",
    "        u.add_tag(f\"u_{i + 1}\")\n",
    "\n",
    "        x_sum = x_sum + x\n",
    "        u_sum = u_sum + u\n",
    "\n",
    "    _ = (x_sum / float(N)).add_tag(\"x_{{avg}}\")\n",
    "    u_avg = (u_sum / float(N)).add_tag(\"u_{{avg}}\")\n",
    "\n",
    "    # Define p_tilde and p_avg such that u_tilde \\in \\partial g(p_tilde) and u_avg \\in \\partial g(u_avg)\n",
    "    p_tilde = pf.Vector(is_basis=True, tags=[\"p_{{tilde}}\"])\n",
    "    p_avg = pf.Vector(is_basis=True, tags=[\"p_{{avg}}\"])\n",
    "    g.add_point_with_grad_restriction(p_tilde, u_tilde)\n",
    "    g.add_point_with_grad_restriction(p_avg, u_avg)\n",
    "    return ctx_drs"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c11b9ddd-96f6-4687-bfb8-832dd6197291",
   "metadata": {},
   "source": [
    "## Write a function to return the PEPBuilder for DRS "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e42aad29-4ce9-4b00-8e30-93555919fbc9",
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_pb_drs(\n",
    "    ctx: pf.PEPContext, stepsize: pf.Parameter, init_distance: pf.Parameter\n",
    ") -> pf.PEPBuilder:\n",
    "    x_0 = ctx[\"x_0\"]\n",
    "    x_tilde = ctx[\"x_{{tilde}}\"]\n",
    "    x_avg = ctx[\"x_{{avg}}\"]\n",
    "\n",
    "    u_0 = ctx[\"u_0\"]\n",
    "    u_tilde = ctx[\"u_{{tilde}}\"]\n",
    "    u_avg = ctx[\"u_{{avg}}\"]\n",
    "\n",
    "    p_tilde = ctx[\"p_{{tilde}}\"]\n",
    "    p_avg = ctx[\"p_{{avg}}\"]\n",
    "\n",
    "    pb = pf.PEPBuilder(ctx_plt)\n",
    "\n",
    "    # Initial Condition\n",
    "    pb.add_initial_constraint(\n",
    "        ((1.0 / stepsize) * (x_0 - x_tilde) ** 2 + stepsize * (u_0 - u_tilde) ** 2).le(\n",
    "            init_distance, name=\"initial_condition\"\n",
    "        )\n",
    "    )\n",
    "    # Set performance metric of primal-dual gap function\n",
    "    pb.set_performance_metric(\n",
    "        f(x_avg)\n",
    "        - f(x_tilde)\n",
    "        + g(p_tilde)\n",
    "        - g(p_avg)\n",
    "        + u_tilde * x_avg\n",
    "        - u_tilde * p_tilde\n",
    "        - u_avg * x_tilde\n",
    "        + u_avg * p_avg\n",
    "    )\n",
    "    return pb"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0fa3265b-6480-405f-9a61-1cb8b44f2f86",
   "metadata": {},
   "source": [
    "## Numerical evidence of convergence of DRS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "44b1e2a1-7c6a-4b5e-b834-9fbebc901b07",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f005b4474d0>"
      ]
     },
     "execution_count": 5,
     "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",
    "alpha = pf.Parameter(\"alpha\")\n",
    "R = pf.Parameter(\"R\")\n",
    "alpha_value = 1\n",
    "R_value = 1\n",
    "\n",
    "opt_values = []\n",
    "\n",
    "for k in range(1, N):\n",
    "    ctx_plt = make_ctx_drs(ctx_name=f\"ctx_plt_{k}\", N=k, stepsize=alpha)\n",
    "    pb_plt = make_pb_drs(ctx_plt, alpha, R)\n",
    "    result = pb_plt.solve(resolve_parameters={\"alpha\": alpha_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",
    "    1 / (cont_iters + 1),\n",
    "    \"r-\",\n",
    "    label=\"Analytical bound $\\\\frac{1}{k+1}$\",\n",
    ")\n",
    "plt.scatter(iters, opt_values, color=\"blue\", marker=\"o\", label=\"Numerical values\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "86845791-541d-4e44-990b-a2be76048df3",
   "metadata": {},
   "source": [
    "## Verification of convergence of DRS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "d7c03085-0c8d-4d48-a7bf-b885d9022808",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.3333323794791133\n"
     ]
    }
   ],
   "source": [
    "N = sp.S(2)\n",
    "alpha_value = sp.S(1)\n",
    "R_value = sp.S(1)\n",
    "\n",
    "ctx_prf = make_ctx_drs(ctx_name=\"ctx_prf\", N=N, stepsize=alpha)\n",
    "pb_prf = make_pb_drs(ctx_prf, alpha, R)\n",
    "\n",
    "result = pb_prf.solve(resolve_parameters={\"alpha\": alpha_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)\n",
    "# Dual variables associated with the interpolations conditions of g with no relaxation\n",
    "gamm_dense = result.get_scalar_constraint_dual_value_in_numpy(g)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "c0baa2b5-d119-4793-801c-6880e50a2717",
   "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={\"alpha\": alpha_value, \"R\": R_value}\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "07019e94-0e9f-4741-bee5-e9c9a8abf6cb",
   "metadata": {},
   "source": [
    "### Solve the problem again with the found relaxation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "992bd5ef-3129-4a60-9aff-7daeb9624e8d",
   "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_{n} as \"n-1\".\n",
    "    We index \"x_avg\" as \"N\" where N is the last iterate.\n",
    "    We index \"x_tilde\" 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) - 1\n",
    "    elif idx == \"{{avg}}\":\n",
    "        return N\n",
    "    elif idx == \"{{tilde}}\":\n",
    "        return N + 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "d420464b",
   "metadata": {},
   "outputs": [],
   "source": [
    "relaxed_constraints = []\n",
    "\n",
    "for tag_i in lamb_dense.row_names:\n",
    "    i = tag_to_index(tag_i)\n",
    "    for tag_j in lamb_dense.col_names:\n",
    "        j = tag_to_index(tag_j)\n",
    "        if i == N + 1 and j in range(N):\n",
    "            continue\n",
    "        if i in range(N) and j == N:\n",
    "            continue\n",
    "        relaxed_constraints.append(f\"f:{tag_i},{tag_j}\")\n",
    "\n",
    "for tag_i in gamm_dense.row_names:\n",
    "    i = tag_to_index(tag_i)\n",
    "    for tag_j in gamm_dense.col_names:\n",
    "        j = tag_to_index(tag_j)\n",
    "        if i == N and j in range(N):\n",
    "            continue\n",
    "        if i in range(N) and j == N + 1:\n",
    "            continue\n",
    "        relaxed_constraints.append(f\"g:{tag_i},{tag_j}\")\n",
    "\n",
    "pb_prf.set_relaxed_constraints(relaxed_constraints)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "c8aaa2dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "result = pb_prf.solve(resolve_parameters={\"alpha\": alpha_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 interpolations conditions of g\n",
    "gamm_sol = result.get_scalar_constraint_dual_value_in_numpy(g)\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": 11,
   "id": "7326d189",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_1 & x_2 & x_{{avg}} & x_{{tilde}} \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.0 & 0.0 & 0.5 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.5 & 0.0 \\\\x_{{avg}} & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{{tilde}} & 0.5 & 0.5 & 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": "code",
   "execution_count": 12,
   "id": "3075a90c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & x_1 & x_2 & x_{{avg}} & x_{{tilde}} \\\\\n",
       "        \\hline\n",
       "        x_1 & 0.0 & 0.0 & 0.5 & 0.0 \\\\x_2 & 0.0 & 0.0 & 0.5 & 0.0 \\\\x_{{avg}} & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{{tilde}} & 0.5 & 0.5 & 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 + 1 and j in range(N):\n",
    "        return 1 / N\n",
    "    if i in range(N) and j == N:\n",
    "        return 1 / N\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": 13,
   "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": "55f644fd",
   "metadata": {},
   "source": [
    "## Verify closed form expression of $\\gamma$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ff225273",
   "metadata": {},
   "source": [
    "- Print the values of $\\gamma$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "436dc521",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & p_1 & p_2 & p_{{avg}} & p_{{tilde}} \\\\\n",
       "        \\hline\n",
       "        p_1 & 0.0 & 0.0 & 0.0 & 0.5 \\\\p_2 & 0.0 & 0.0 & 0.0 & 0.5 \\\\p_{{avg}} & 0.5 & 0.5 & 0.0 & 0.0 \\\\p_{{tilde}} & 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": [
    "gamm_sol.pprint()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "ef2ec4ae",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccc}\n",
       "         & p_1 & p_2 & p_{{avg}} & p_{{tilde}} \\\\\n",
       "        \\hline\n",
       "        p_1 & 0.0 & 0.0 & 0.0 & 0.5 \\\\p_2 & 0.0 & 0.0 & 0.0 & 0.5 \\\\p_{{avg}} & 0.5 & 0.5 & 0.0 & 0.0 \\\\p_{{tilde}} & 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": [
    "# We consider proper shifting since x_0 is not in the gamma matrix\n",
    "def gamm(tag_i, tag_j, N=N):\n",
    "    i = tag_to_index(tag_i)\n",
    "    j = tag_to_index(tag_j)\n",
    "    if i == N and j in range(N):\n",
    "        return 1 / N\n",
    "    if i in range(N) and j == N + 1:\n",
    "        return 1 / N\n",
    "    return 0\n",
    "\n",
    "\n",
    "gamm_cand = pf.pprint_labeled_matrix(\n",
    "    gamm, gamm_sol.row_names, gamm_sol.col_names, return_matrix=True\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "86b343c9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Did we guess the right closed form of gamma? True\n"
     ]
    }
   ],
   "source": [
    "print(\n",
    "    \"Did we guess the right closed form of gamma?\",\n",
    "    np.allclose(gamm_sol.matrix, gamm_cand, 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": 17,
   "id": "05ddf829",
   "metadata": {},
   "outputs": [],
   "source": [
    "pm = pf.ExpressionManager(\n",
    "    ctx_prf, resolve_parameters={\"alpha\": alpha_value, \"R\": R_value}\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "98ba117b",
   "metadata": {},
   "source": [
    "- Print the values of $S$ obtained from the solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "d17dee1d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccccc}\n",
       "         & x_0 & u_0 & x_{{tilde}} & u_{{tilde}} & \\nabla f(x_1) & \\nabla g(p_1) & \\nabla f(x_2) & \\nabla g(p_2) & p_{{tilde}} & p_{{avg}} & \\nabla f(x_{{avg}}) & \\nabla f(x_{{tilde}}) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.333 & 0.0 & -0.333 & -0.0 & -0.25 & -0.25 & -0.25 & -0.25 & 0.0 & 0.0 & -0.0 & 0.0 \\\\u_0 & 0.0 & 0.333 & -0.0 & -0.333 & 0.25 & 0.25 & 0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{{tilde}} & -0.333 & -0.0 & 0.333 & 0.0 & 0.25 & 0.25 & 0.25 & 0.25 & -0.0 & -0.0 & 0.0 & 0.0 \\\\u_{{tilde}} & -0.0 & -0.333 & 0.0 & 0.333 & -0.25 & -0.25 & -0.25 & -0.25 & -0.0 & -0.0 & -0.0 & 0.0 \\\\\\nabla f(x_1) & -0.25 & 0.25 & 0.25 & -0.25 & 0.5 & 0.5 & 0.25 & 0.25 & -0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(p_1) & -0.25 & 0.25 & 0.25 & -0.25 & 0.5 & 0.5 & 0.25 & 0.25 & -0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_2) & -0.25 & 0.25 & 0.25 & -0.25 & 0.25 & 0.25 & 0.5 & 0.5 & 0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla g(p_2) & -0.25 & 0.25 & 0.25 & -0.25 & 0.25 & 0.25 & 0.5 & 0.5 & 0.0 & -0.0 & 0.0 & 0.0 \\\\p_{{tilde}} & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\p_{{avg}} & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{avg}}) & -0.0 & 0.0 & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{tilde}}) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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": [
    "S_sol.pprint()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f43a1265",
   "metadata": {},
   "source": [
    "- Subtract the decomposed closed-form expressions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "c6fad11c",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = ctx_prf.tracked_point(f)\n",
    "p = ctx_prf.tracked_point(g)\n",
    "u = [ctx_prf[f\"u_{i}\"] for i in range(1, N + 1)]\n",
    "v = [f.grad(x[i]) + u[i] for i in range(N)]\n",
    "\n",
    "x_0 = ctx_prf[\"x_0\"]\n",
    "x_tilde = ctx_prf[\"x_{{tilde}}\"]\n",
    "u_0 = ctx_prf[\"u_0\"]\n",
    "u_tilde = ctx_prf[\"u_{{tilde}}\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "15c3618e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[x_1, x_2, x_{{avg}}, x_{{tilde}}]\n",
      "[p_1, p_2, p_{{avg}}, p_{{tilde}}]\n",
      "[u_1, u_2]\n"
     ]
    }
   ],
   "source": [
    "print(x)\n",
    "print(p)\n",
    "print(u)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "7c4c0e27",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccccc}\n",
       "         & x_0 & u_0 & x_{{tilde}} & u_{{tilde}} & \\nabla f(x_1) & \\nabla g(p_1) & \\nabla f(x_2) & \\nabla g(p_2) & p_{{tilde}} & p_{{avg}} & \\nabla f(x_{{avg}}) & \\nabla f(x_{{tilde}}) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & 0.0 \\\\u_0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{{tilde}} & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 \\\\u_{{tilde}} & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.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.125 & 0.125 & -0.125 & -0.125 & -0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(p_1) & -0.0 & 0.0 & 0.0 & -0.0 & 0.125 & 0.125 & -0.125 & -0.125 & -0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_2) & 0.0 & -0.0 & -0.0 & 0.0 & -0.125 & -0.125 & 0.125 & 0.125 & 0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla g(p_2) & 0.0 & -0.0 & 0.0 & 0.0 & -0.125 & -0.125 & 0.125 & 0.125 & 0.0 & -0.0 & 0.0 & 0.0 \\\\p_{{tilde}} & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\p_{{avg}} & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{avg}}) & -0.0 & 0.0 & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{tilde}}) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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": [
    "S_guess1_x = (\n",
    "    1\n",
    "    / (alpha * (N + 1))\n",
    "    * (x_0 - x_tilde - alpha * (N + 1) / (2 * N) * sum(v[i] for i in range(N))) ** 2\n",
    ")\n",
    "S_guess1_u = (\n",
    "    alpha\n",
    "    / (N + 1)\n",
    "    * (u_0 - u_tilde + (N + 1) / (2 * N) * sum(v[i] for i in range(N))) ** 2\n",
    ")\n",
    "\n",
    "remainder1 = S_sol.matrix - pm.eval_scalar(S_guess1_x + S_guess1_u).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder1, S_sol.row_names, S_sol.col_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "3915864b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccccc}\n",
       "         & x_0 & u_0 & x_{{tilde}} & u_{{tilde}} & \\nabla f(x_1) & \\nabla g(p_1) & \\nabla f(x_2) & \\nabla g(p_2) & p_{{tilde}} & p_{{avg}} & \\nabla f(x_{{avg}}) & \\nabla f(x_{{tilde}}) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & 0.0 \\\\u_0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{{tilde}} & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & -0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 \\\\u_{{tilde}} & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.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 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(p_1) & -0.0 & 0.0 & 0.0 & -0.0 & 0.0 & 0.0 & -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 & -0.0 & -0.0 & 0.0 & -0.0 & 0.0 & 0.0 \\\\\\nabla g(p_2) & 0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 & -0.0 & 0.0 & 0.0 \\\\p_{{tilde}} & 0.0 & 0.0 & -0.0 & -0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\p_{{avg}} & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & -0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{avg}}) & -0.0 & 0.0 & 0.0 & -0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{tilde}}) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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": [
    "S_guess2 = (\n",
    "    alpha\n",
    "    / (4 * N**2)\n",
    "    * sum((v[i] - v[j]) ** 2 for i, j in itertools.product(range(N), range(N)))\n",
    ")\n",
    "\n",
    "remainder2 = remainder1 - pm.eval_scalar(S_guess2).inner_prod_coords\n",
    "pf.pprint_labeled_matrix(remainder2, 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": 23,
   "id": "b444e21f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\displaystyle \n",
       "        \\begin{array}{c|cccccccccccc}\n",
       "         & x_0 & u_0 & x_{{tilde}} & u_{{tilde}} & \\nabla f(x_1) & \\nabla g(p_1) & \\nabla f(x_2) & \\nabla g(p_2) & p_{{tilde}} & p_{{avg}} & \\nabla f(x_{{avg}}) & \\nabla f(x_{{tilde}}) \\\\\n",
       "        \\hline\n",
       "        x_0 & 0.333 & 0.0 & -0.333 & 0.0 & -0.25 & -0.25 & -0.25 & -0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\u_0 & 0.0 & 0.333 & 0.0 & -0.333 & 0.25 & 0.25 & 0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\x_{{tilde}} & -0.333 & 0.0 & 0.333 & 0.0 & 0.25 & 0.25 & 0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\u_{{tilde}} & 0.0 & -0.333 & 0.0 & 0.333 & -0.25 & -0.25 & -0.25 & -0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_1) & -0.25 & 0.25 & 0.25 & -0.25 & 0.5 & 0.5 & 0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(p_1) & -0.25 & 0.25 & 0.25 & -0.25 & 0.5 & 0.5 & 0.25 & 0.25 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_2) & -0.25 & 0.25 & 0.25 & -0.25 & 0.25 & 0.25 & 0.5 & 0.5 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla g(p_2) & -0.25 & 0.25 & 0.25 & -0.25 & 0.25 & 0.25 & 0.5 & 0.5 & 0.0 & 0.0 & 0.0 & 0.0 \\\\p_{{tilde}} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\p_{{avg}} & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{avg}}) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 \\\\\\nabla f(x_{{tilde}}) & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 0.0 & 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": [
    "S_guess = S_guess1_x + S_guess1_u + S_guess2\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": "code",
   "execution_count": 24,
   "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(S_guess_eval, S_sol.matrix, atol=1e-4),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "467d206a-1453-4d46-b2a6-059fca04c472",
   "metadata": {},
   "source": [
    "### Symbolic calculation"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "834db0c4-f57e-46b6-9ccd-fd6c92ecd939",
   "metadata": {},
   "source": [
    "- Assemble the RHS of the proof."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "475dc370-3a8b-40a7-98f3-67fb400ee153",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/2*(f(x_{{avg}})-f(x_1)+\\nabla f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+\\nabla f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+\\nabla f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+\\nabla f(x_2)*(x_{{tilde}}-(x_2)))$"
      ],
      "text/plain": [
       "0+1/2*(f(x_{{avg}})-f(x_1)+grad_f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+grad_f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+grad_f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+grad_f(x_2)*(x_{{tilde}}-(x_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": 26,
   "id": "e3acc948-f17f-4ae1-9a00-b48b7e38c1ed",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/2*(f(x_{{avg}})-f(x_1)+\\nabla f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+\\nabla f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+\\nabla f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+\\nabla f(x_2)*(x_{{tilde}}-(x_2)))+1/2*(g(p_{{tilde}})-g(p_1)+u_{{tilde}}*(p_1-p_{{tilde}}))+1/2*(g(p_{{tilde}})-g(p_2)+u_{{tilde}}*(p_2-p_{{tilde}}))+1/2*(g(p_1)-g(p_{{avg}})+\\nabla g(p_1)*(p_{{avg}}-(p_1)))+1/2*(g(p_2)-g(p_{{avg}})+\\nabla g(p_2)*(p_{{avg}}-(p_2)))$"
      ],
      "text/plain": [
       "0+1/2*(f(x_{{avg}})-f(x_1)+grad_f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+grad_f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+grad_f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+grad_f(x_2)*(x_{{tilde}}-(x_2)))+1/2*(g(p_{{tilde}})-g(p_1)+u_{{tilde}}*(p_1-p_{{tilde}}))+1/2*(g(p_{{tilde}})-g(p_2)+u_{{tilde}}*(p_2-p_{{tilde}}))+1/2*(g(p_1)-g(p_{{avg}})+grad_g(p_1)*(p_{{avg}}-(p_1)))+1/2*(g(p_2)-g(p_{{avg}})+grad_g(p_2)*(p_{{avg}}-(p_2)))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for x_i, x_j in itertools.product(ctx_prf.tracked_point(g), ctx_prf.tracked_point(g)):\n",
    "    if gamm(x_i.tag, x_j.tag) != 0:\n",
    "        interp_scalar_sum += gamm(x_i.tag, x_j.tag) * g.interp_ineq(x_i.tag, x_j.tag)\n",
    "\n",
    "display(interp_scalar_sum)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "2338d0b1-e8f2-4293-ac3e-30547c3620c8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle 0+1/2*(f(x_{{avg}})-f(x_1)+\\nabla f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+\\nabla f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+\\nabla f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+\\nabla f(x_2)*(x_{{tilde}}-(x_2)))+1/2*(g(p_{{tilde}})-g(p_1)+u_{{tilde}}*(p_1-p_{{tilde}}))+1/2*(g(p_{{tilde}})-g(p_2)+u_{{tilde}}*(p_2-p_{{tilde}}))+1/2*(g(p_1)-g(p_{{avg}})+\\nabla g(p_1)*(p_{{avg}}-(p_1)))+1/2*(g(p_2)-g(p_{{avg}})+\\nabla g(p_2)*(p_{{avg}}-(p_2)))-(1/alpha*3*\\|x_0-x_{{tilde}}-alpha*3/4*(\\nabla f(x_1)+u_1+\\nabla f(x_2)+u_2)\\|^2+alpha/3*\\|u_0-u_{{tilde}}+3/4*(\\nabla f(x_1)+u_1+\\nabla f(x_2)+u_2)\\|^2+alpha/16*(0+\\|\\nabla f(x_1)+u_1-(\\nabla f(x_1)+u_1)\\|^2+\\|\\nabla f(x_1)+u_1-(\\nabla f(x_2)+u_2)\\|^2+\\|\\nabla f(x_2)+u_2-(\\nabla f(x_1)+u_1)\\|^2+\\|\\nabla f(x_2)+u_2-(\\nabla f(x_2)+u_2)\\|^2))$"
      ],
      "text/plain": [
       "0+1/2*(f(x_{{avg}})-f(x_1)+grad_f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+grad_f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+grad_f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+grad_f(x_2)*(x_{{tilde}}-(x_2)))+1/2*(g(p_{{tilde}})-g(p_1)+u_{{tilde}}*(p_1-p_{{tilde}}))+1/2*(g(p_{{tilde}})-g(p_2)+u_{{tilde}}*(p_2-p_{{tilde}}))+1/2*(g(p_1)-g(p_{{avg}})+grad_g(p_1)*(p_{{avg}}-(p_1)))+1/2*(g(p_2)-g(p_{{avg}})+grad_g(p_2)*(p_{{avg}}-(p_2)))-(1/alpha*3*|x_0-x_{{tilde}}-alpha*3/4*(grad_f(x_1)+u_1+grad_f(x_2)+u_2)|^2+alpha/3*|u_0-u_{{tilde}}+3/4*(grad_f(x_1)+u_1+grad_f(x_2)+u_2)|^2+alpha/16*(0+|grad_f(x_1)+u_1-(grad_f(x_1)+u_1)|^2+|grad_f(x_1)+u_1-(grad_f(x_2)+u_2)|^2+|grad_f(x_2)+u_2-(grad_f(x_1)+u_1)|^2+|grad_f(x_2)+u_2-(grad_f(x_2)+u_2)|^2))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "RHS = interp_scalar_sum - S_guess\n",
    "display(RHS)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dd47682c-fb1f-43a9-a2c9-834afa2fc29c",
   "metadata": {},
   "source": [
    "- Assemble the LHS of the proof"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "338a482e-ba07-49c3-b591-c50cab48815f",
   "metadata": {},
   "outputs": [],
   "source": [
    "x_avg = ctx_prf[\"x_{{avg}}\"]\n",
    "u_avg = ctx_prf[\"u_{{avg}}\"]\n",
    "p_avg = ctx_prf[\"p_{{avg}}\"]\n",
    "p_tilde = ctx_prf[\"p_{{tilde}}\"]\n",
    "\n",
    "perf_metric = (\n",
    "    f(x_avg)\n",
    "    - f(x_tilde)\n",
    "    + g(p_tilde)\n",
    "    - g(p_avg)\n",
    "    + u_tilde * x_avg\n",
    "    - u_tilde * p_tilde\n",
    "    - u_avg * x_tilde\n",
    "    + u_avg * p_avg\n",
    ")\n",
    "LHS = perf_metric - 1 / (alpha * (N + 1)) * (\n",
    "    (x_0 - x_tilde) ** 2 + alpha**2 * (u_0 - u_tilde) ** 2\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "9923eac6-9bff-43eb-9050-ed1ede7327bf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/latex": [
       "$\\displaystyle f(x_{{avg}})-f(x_{{tilde}})+g(p_{{tilde}})-g(p_{{avg}})+u_{{tilde}}*x_{{avg}}-u_{{tilde}}*p_{{tilde}}-u_{{avg}}*x_{{tilde}}+u_{{avg}}*p_{{avg}}-1/alpha*3*(\\|x_0-x_{{tilde}}\\|^2+alpha^2*\\|u_0-u_{{tilde}}\\|^2)-(0+1/2*(f(x_{{avg}})-f(x_1)+\\nabla f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+\\nabla f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+\\nabla f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+\\nabla f(x_2)*(x_{{tilde}}-(x_2)))+1/2*(g(p_{{tilde}})-g(p_1)+u_{{tilde}}*(p_1-p_{{tilde}}))+1/2*(g(p_{{tilde}})-g(p_2)+u_{{tilde}}*(p_2-p_{{tilde}}))+1/2*(g(p_1)-g(p_{{avg}})+\\nabla g(p_1)*(p_{{avg}}-(p_1)))+1/2*(g(p_2)-g(p_{{avg}})+\\nabla g(p_2)*(p_{{avg}}-(p_2)))-(1/alpha*3*\\|x_0-x_{{tilde}}-alpha*3/4*(\\nabla f(x_1)+u_1+\\nabla f(x_2)+u_2)\\|^2+alpha/3*\\|u_0-u_{{tilde}}+3/4*(\\nabla f(x_1)+u_1+\\nabla f(x_2)+u_2)\\|^2+alpha/16*(0+\\|\\nabla f(x_1)+u_1-(\\nabla f(x_1)+u_1)\\|^2+\\|\\nabla f(x_1)+u_1-(\\nabla f(x_2)+u_2)\\|^2+\\|\\nabla f(x_2)+u_2-(\\nabla f(x_1)+u_1)\\|^2+\\|\\nabla f(x_2)+u_2-(\\nabla f(x_2)+u_2)\\|^2)))$"
      ],
      "text/plain": [
       "f(x_{{avg}})-f(x_{{tilde}})+g(p_{{tilde}})-g(p_{{avg}})+u_{{tilde}}*x_{{avg}}-u_{{tilde}}*p_{{tilde}}-u_{{avg}}*x_{{tilde}}+u_{{avg}}*p_{{avg}}-1/alpha*3*(|x_0-x_{{tilde}}|^2+alpha**2*|u_0-u_{{tilde}}|^2)-(0+1/2*(f(x_{{avg}})-f(x_1)+grad_f(x_{{avg}})*(x_1-x_{{avg}}))+1/2*(f(x_{{avg}})-f(x_2)+grad_f(x_{{avg}})*(x_2-x_{{avg}}))+1/2*(f(x_1)-f(x_{{tilde}})+grad_f(x_1)*(x_{{tilde}}-(x_1)))+1/2*(f(x_2)-f(x_{{tilde}})+grad_f(x_2)*(x_{{tilde}}-(x_2)))+1/2*(g(p_{{tilde}})-g(p_1)+u_{{tilde}}*(p_1-p_{{tilde}}))+1/2*(g(p_{{tilde}})-g(p_2)+u_{{tilde}}*(p_2-p_{{tilde}}))+1/2*(g(p_1)-g(p_{{avg}})+grad_g(p_1)*(p_{{avg}}-(p_1)))+1/2*(g(p_2)-g(p_{{avg}})+grad_g(p_2)*(p_{{avg}}-(p_2)))-(1/alpha*3*|x_0-x_{{tilde}}-alpha*3/4*(grad_f(x_1)+u_1+grad_f(x_2)+u_2)|^2+alpha/3*|u_0-u_{{tilde}}+3/4*(grad_f(x_1)+u_1+grad_f(x_2)+u_2)|^2+alpha/16*(0+|grad_f(x_1)+u_1-(grad_f(x_1)+u_1)|^2+|grad_f(x_1)+u_1-(grad_f(x_2)+u_2)|^2+|grad_f(x_2)+u_2-(grad_f(x_1)+u_1)|^2+|grad_f(x_2)+u_2-(grad_f(x_2)+u_2)|^2)))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "diff = LHS - RHS\n",
    "display(diff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "95b2a1ca-ede2-4432-8ce1-197a92f235fb",
   "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(\n",
    "        ctx_prf,\n",
    "        sympy_mode=True,\n",
    "        resolve_parameters={\"alpha\": alpha_value, \"R\": R_value},\n",
    "    )\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "32920422",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "594c0bc6",
   "metadata": {},
   "source": [
    "#### With the verified closed form expression, we have verified the below for fixed $N$"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "04bafdc8",
   "metadata": {},
   "source": [
    "\\begin{align*}\n",
    "&\\mathcal{L}(\\bar{x}^N, \\tilde{u}) - \\mathcal{L}(\\tilde{x}, \\bar{u}^N) \n",
    "- \\frac{1}{\\alpha (N+1)} \\left( \\|x^0 - \\tilde{x}\\|^2 + \\alpha^2 \\|u^0 - \\tilde{u}\\|^2 \\right) \\\\ \n",
    "&= \\frac{1}{N} \\sum_{k=1}^N \n",
    "\\Big( f(\\bar{x}^N) - f(x^k) + \n",
    "\\big\\langle \\tilde{\\nabla} f(\\bar{x}^N), x^k - \\bar{x}^N \\big\\rangle \\Big)\n",
    "+ \\frac{1}{N} \\sum_{k=1}^N \n",
    "\\Big( f(x^k) - f(\\tilde{x}) + \n",
    "\\big\\langle \\tilde{\\nabla} f(x^k), \\tilde{x} - x^k \\big\\rangle \\Big)\\\\\n",
    "& + \\frac{1}{N} \\sum_{k=1}^N \n",
    "\\Big( g(p^k) - g(\\bar{p}^N) + \n",
    "\\big\\langle u^k, \\bar{p}^N - p^k \\big\\rangle \\Big)\n",
    "+ \\frac{1}{N} \\sum_{k=1}^N \n",
    "\\Big( g(\\tilde{p}) - g(p^k) + \n",
    "\\big\\langle \\tilde{u}, p^k - \\tilde{p} \\big\\rangle \\Big) \\\\\n",
    "& + \\frac{1}{\\alpha (N+1)} \n",
    "\\left( \n",
    "\\left\\| x^0 - \\tilde{x} - \\frac{\\alpha (N+1)}{2N} \\sum_{k=1}^N v^k \\right\\|^2\n",
    "+ \\alpha^2 \n",
    "\\left\\| u^0 - \\tilde{u} - \\frac{N+1}{2N} \\sum_{k=1}^N v^k \\right\\|^2\n",
    "\\right) \\\\\n",
    "& + \\frac{\\alpha}{2N^2} \n",
    "\\sum_{k=1}^N \\sum_{l=1}^{k-1} \n",
    "\\|v^k - v^l\\|^2\n",
    "\\end{align*}\n",
    "holds, where we denote $v^k = \\tilde{\\nabla} f(x^k) + u^k$."
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "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.13.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
