{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "$Assumptions = {Element[_, Reals]};\n",
    "$Assumptions = \n",
    " Join[$Assumptions, {0 < \\[Mu] < L, \\[Mu] > 0, L > 0, L > \\[Mu], 0 < \\[Epsilon], \\[Epsilon] <= 1, 0 < \\[Eta], Subscript[\\[Lambda], 1] >= 0, \\[Nu] > 0}]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Analyzing EF21 using our Lyapunov function\n",
    "We have the following Lyapunov function:\n",
    "$$\n",
    "L(\\xi_k) \\coloneqq (1-c) ||\\nabla f(x_k)||^2 - 2c \\nabla f(x_k)^\\top d_k + c ||d_k||^2,\n",
    "$$\n",
    "where $c \\coloneqq \\frac{1}{2 + \\sqrt{1 - \\delta}}$. We now perform the same exercise of trying to retreive a closed form using Mathematica."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "const = 1 / (2 + Sqrt[\\[Epsilon]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "Lyapunov[gk_, dk_] := ((1-const) * gk^2 - 2 * const * gk * dk + const * dk^2) / const\n",
    "objective := FullSimplify[Lyapunov[Subscript[g, k+1], Subscript[d, k+1]]]\n",
    "objective"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "\n",
    "initialization := FullSimplify[Lyapunov[Subscript[g, k], Subscript[d, k]] - 1]\n",
    "initialization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "CvxConstraint[xi_, xj_, fi_, fj_, gi_, gj_] := fj - fi + gj*(xi - xj) + 1 / (2*L)*(gi - gj)^2 + \\[Mu] / (2*(1 - \\[Mu] / L)) * (xi - xj - 1/L * (gi - gj))^2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "constr1 := CvxConstraint[SubStar[x], Subscript[x, k], SubStar[f], Subscript[f, k], 0, Subscript[g, k]]\n",
    "constr1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "constr2 := CvxConstraint[Subscript[x, k], SubStar[x], Subscript[f, k], SubStar[f], Subscript[g, k], 0]\n",
    "constr2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "\n",
    "constr3 := CvxConstraint[SubStar[x], Subscript[x, k+1], SubStar[f], Subscript[f, k+1], 0, Subscript[g, k+1]]\n",
    "constr3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "constr4 := CvxConstraint[Subscript[x, k+1], SubStar[x], Subscript[f, k+1], SubStar[f], Subscript[g, k+1], 0]\n",
    "constr4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "constr5 := CvxConstraint[Subscript[x, k+1], Subscript[x, k], Subscript[f, k+1], Subscript[f, k], Subscript[g, k+1], Subscript[g, k]]\n",
    "constr5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "constr6 := CvxConstraint[Subscript[x, k], Subscript[x, k+1], Subscript[f, k], Subscript[f, k+1], Subscript[g, k], Subscript[g, k+1]]\n",
    "constr6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "CompressConstraint[o_, c_] := (o - c)^2 - \\[Epsilon]*o^2;"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "compress := CompressConstraint[Subscript[g, k+1] - Subscript[d, k], Subscript[c, k]];\n",
    "compress"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "methodx = Subscript[x, k+1] -> Subscript[x, k] - \\[Eta] * Subscript[d, k];\n",
    "methodx"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "methodd = Subscript[d, k+1] -> Subscript[d, k] + Subscript[c, k];\n",
    "methodd"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "Lagrangian[objective_, constraints_, variables_] := objective - Sum[variables[[i]] * constraints[[i]], {i, 1, Length[constraints]}]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "Lagrange := Lagrangian[objective, \n",
    "                       {initialization, constr1, constr2, constr3, constr4, constr5, constr6, compress},\n",
    "                       {\\[Rho], Subscript[\\[Lambda], 1], Subscript[\\[Lambda], 2], Subscript[\\[Lambda], 3], Subscript[\\[Lambda], 4], Subscript[\\[Lambda], 5], Subscript[\\[Lambda], 6], \\[Nu]}]\n",
    "\\[ScriptCapitalL] := FullSimplify[Lagrange /. {methodx, methodd}]\n",
    "\\[ScriptCapitalL]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "LMIfull := FullSimplify[D[-\\[ScriptCapitalL], {{Subscript[x, k], SubStar[x], Subscript[g, k], Subscript[g, k+1], Subscript[c, k], Subscript[d, k]}, 2}] / 2];\n",
    "LMIfull // MatrixForm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "linconstr = FullSimplify[D[-\\[ScriptCapitalL], {{Subscript[f, k], Subscript[f, k+1], SubStar[f]}, 1}]]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Looking at the LMI\n",
    "Let us now solve the dual and verify our implementation."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "SolveDualProblem[LL_, mu_, eta_, delta_] := Module[\n",
    "  {d = delta, S},\n",
    "  \n",
    "  (* Define the dual variables and constraints *)\n",
    "  dualVars = {S, s, r, l1, l2, l3, l4, l5, l6, nu};\n",
    "  subrules = {\n",
    "      Subscript[\\[Lambda], 1] -> l1,\n",
    "      Subscript[\\[Lambda], 2] -> l2,\n",
    "      Subscript[\\[Lambda], 3] -> l3,\n",
    "      Subscript[\\[Lambda], 4] -> l4,\n",
    "      Subscript[\\[Lambda], 5] -> l5,\n",
    "      Subscript[\\[Lambda], 6] -> l6,\n",
    "      \\[Rho] -> r, \n",
    "      \\[Epsilon] -> 1 - d,\n",
    "      L -> LL, \n",
    "      \\[Mu] -> mu, \n",
    "      \\[Eta] -> eta,\n",
    "      \\[Nu] -> nu\n",
    "  };\n",
    "  LMIsubbed = LMIfull /. subrules;\n",
    "  linconstrsubbed = linconstr /. subrules;\n",
    "\n",
    "  constraints = {\n",
    "    S == LMIsubbed,\n",
    "    s == linconstrsubbed,\n",
    "    VectorGreaterEqual[{S, 0}, {\"SemidefiniteCone\", 6}],\n",
    "    l1 >= 0,\n",
    "    l2 >= 0,\n",
    "    l3 >= 0,\n",
    "    l4 >= 0,\n",
    "    l5 >= 0,\n",
    "    l6 >= 0,\n",
    "    r >= 0,\n",
    "    nu >= 0\n",
    "  };\n",
    "  \n",
    "  (* Solve the dual problem *)\n",
    "  SemidefiniteOptimization[r, constraints, dualVars, Method -> \"MOSEK\", Tolerance -> 10^(-10)]\n",
    "]\n",
    "SolveDualProblem[1.0, 0.1, 0.1, 0.9]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "localDelta = 0.5;\n",
    "localL = 1.0;\n",
    "localMu = 0.1;\n",
    "etas = Range[0.2, 0.3, 0.005];\n",
    "dualValues = Table[\n",
    "  SolveDualProblem[localL, localMu, eta, localDelta], \n",
    "  {eta, etas}\n",
    "];\n",
    "\n",
    "SValues = Table[S /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "rValues = Table[r /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "l1Values = Table[l1 /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "l2Values = Table[l2 /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "l3Values = Table[l3 /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "l4Values = Table[l4 /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "l5Values = Table[l5 /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "l6Values = Table[l6 /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "nuValues = Table[nu /. dualValues[[i]], {i, Length[dualValues]}];\n",
    "\n",
    "\n",
    "(* Create the line plot *)\n",
    "ListLinePlot[\n",
    "  {\n",
    "    Transpose[{etas, rValues}],\n",
    "    Transpose[{etas, l1Values}],\n",
    "    Transpose[{etas, l2Values}],\n",
    "    Transpose[{etas, l3Values}],\n",
    "    Transpose[{etas, l4Values}],\n",
    "    Transpose[{etas, l5Values}],\n",
    "    Transpose[{etas, l6Values}],\n",
    "    Transpose[{etas, nuValues}]\n",
    "  },\n",
    "  PlotLegends -> {\"ρ (contraction)\",\"λ1\", \"λ2\", \"λ3\", \"λ4\", \"λ5\", \"λ6\", \"ν\"},\n",
    "  PlotStyle -> {Red, Green, Blue, Orange, Purple, Pink, Yellow},\n",
    "  AxesLabel -> {\"η\", \"Value\"},\n",
    "  PlotLabel -> \"Multipliers as Functions of η (δ = \" <> ToString[localDelta] <> \")\",\n",
    "  GridLines -> Automatic,\n",
    "  PlotMarkers -> {Automatic, 6},\n",
    "  ImageSize -> Large,\n",
    "  PlotRange -> All\n",
    "]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Two observations:\n",
    "1. Lambda 5 and 6 are literally the same.\n",
    "2. the other interpolation constraint multipliers are 0 until we reach that \"switch regime\". We won't analyze that regime so we can set them to 0."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "LMISimplified = FullSimplify[LMIfull /. {Subscript[\\[Lambda], 5] -> \\[Lambda], \n",
    "                                     Subscript[\\[Lambda], 6] -> \\[Lambda],\n",
    "                                     Subscript[\\[Lambda], 1] -> 0,\n",
    "                                     Subscript[\\[Lambda], 2] -> 0,\n",
    "                                     Subscript[\\[Lambda], 3] -> 0,\n",
    "                                     Subscript[\\[Lambda], 4] -> 0}];\n",
    "LMISimplified // MatrixForm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "(* Extract non-zero rows and columns from LMISimplified *)\n",
    "nonZeroRows = DeleteCases[Range[Length[LMISimplified]], \n",
    "  i_ /; VectorQ[LMISimplified[[i]], # == 0 &]];\n",
    "\n",
    "nonZeroCols = DeleteCases[Range[Length[Transpose[LMISimplified]]], \n",
    "  j_ /; VectorQ[Transpose[LMISimplified][[j]], # == 0 &]];\n",
    "\n",
    "(* Create a new matrix with only non-zero rows and columns *)\n",
    "LMIReduced = LMISimplified[[nonZeroRows, nonZeroCols]];\n",
    "LMIReduced // MatrixForm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "nuValues"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Clearly we can set\n",
    "$$\n",
    "\\nu = 1\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "LMI2 = FullSimplify[LMIReduced /. {\\[Nu] -> 1}];\n",
    "LMI2 // MatrixForm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "LMI = {{LMI2[[1, 1]], LMI2[[1, 2]], LMI2[[1, 4]]}, \n",
    "       {LMI2[[2, 1]], LMI2[[2, 2]], LMI2[[2, 4]]}, \n",
    "       {LMI2[[4, 1]], LMI2[[4, 2]], LMI2[[4, 4]]}};\n",
    "LMI // MatrixForm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "FullSimplify[Solve[Det[LMI] == 0, \\[Rho]]]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Cutting straight to the chase: do we magically get closed forms when using optimal step size?"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "etaOpt = \\[Eta] -> 2 / (L + \\[Mu]) * (1 - \\[Epsilon]) / (1 + Sqrt[\\[Epsilon]])^2;\n",
    "etaOpt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "LMIOpt = FullSimplify[LMI /. etaOpt];\n",
    "LMIOpt // MatrixForm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "rhosOptimal = FullSimplify[Solve[Det[LMIOpt] == 0, \\[Rho]]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "rhos1Optimal = rhosOptimal[[1, 1]][[2]] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "rhos2Optimal = rhosOptimal[[2, 1]][[2]] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "diff = rhos1Optimal - rhos2Optimal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "lambdasOptimal = Reduce[diff == 0, \\[Lambda]] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "lambdasOptimal[[2]][[2]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "lOptimal = Reduce[lambdasOptimal[[2]][[2]], \\[Lambda]][[2, 3, 2]] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "lOptimal /. {\\[Epsilon] -> 1 - localDelta, L -> localL, \\[Mu] -> localMu}"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Let us verify that it is correct quickly:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "etaOptNumeric = (etaOpt /. {\\[Epsilon] -> 1 - localDelta, L -> localL, \\[Mu] -> localMu})[[2]];\n",
    "lNumeric = SolveDualProblem[localL, localMu, etaOptNumeric, localDelta]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Finally, we get the same rate:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "rate = rhosOptimal[[1, 1, 2]] /. {\\[Lambda] -> lOptimal} // FullSimplify"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Natural proof"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "multipliers = {\\[Nu] -> 1, etaOpt}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "SOS = \\[Lambda] * (constr5 + constr6) + \\[Nu] * compress - (Lyapunov[Subscript[g, k+1], Subscript[d, k+1]] - \\[Rho] * Lyapunov[Subscript[g, k], Subscript[d, k]]);\n",
    "SOS = SOS /. methodx /. methodd /. multipliers // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "SOS = Collect[SOS, Subscript[c, k]] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "Variables[SOS]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We begin by Cholesky:ing out $d_k$:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "a = Coefficient[SOS, Subscript[d, k]^2] // FullSimplify\n",
    "b = Coefficient[SOS, Subscript[d, k] * Subscript[g, k+1]] / a / 2 // FullSimplify\n",
    "c = Coefficient[SOS, Subscript[d, k] * Subscript[g, k]] / a / 2 // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "a + \\[Epsilon] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "b * a // FullSimplify\n",
    "c * a // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "aSubbed = a /. {\\[Rho] -> rate, \\[Lambda] -> lOptimal} // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "SOS1 = a * (Subscript[d, k] + b * Subscript[g, k+1] + c * Subscript[g, k])^2;\n",
    "rest = Collect[SOS - SOS1, Subscript[d, k]] // FullSimplify"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "wolfram"
    }
   },
   "outputs": [],
   "source": [
    "Coefficient[rest, Subscript[g, k+1]^2] /. {\\[Rho] -> rate, \\[Lambda] -> lOptimal} // FullSimplify\n",
    "Coefficient[rest, Subscript[g, k]^2] /. {\\[Rho] -> rate, \\[Lambda] -> lOptimal} // FullSimplify\n",
    "Coefficient[rest, Subscript[g, k] * Subscript[g, k+1]] /. {\\[Rho] -> rate, \\[Lambda] -> lOptimal} // FullSimplify"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Wolfram Language 14.2",
   "language": "Wolfram Language",
   "name": "wolframlanguage14.2"
  },
  "language_info": {
   "codemirror_mode": "mathematica",
   "file_extension": ".m",
   "mimetype": "application/vnd.wolfram.m",
   "name": "Wolfram Language",
   "pygments_lexer": "mathematica",
   "version": "12.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
