{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1f22cb4a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import math\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "196f623e",
   "metadata": {},
   "outputs": [],
   "source": [
    "n, d = 3, 4\n",
    "\n",
    "L = 3\n",
    "mu = 1\n",
    "\n",
    "M = []\n",
    "z = []\n",
    "\n",
    "for idx in range(n):\n",
    "    temp1 = np.ones(d)\n",
    "    temp1[idx] = L\n",
    "    temp2 = np.zeros(d)\n",
    "    temp2[idx] = L\n",
    "    \n",
    "    if idx == 0:\n",
    "        temp2[d - 1] = L\n",
    "        \n",
    "    M.append(np.diag(temp1))\n",
    "    z.append(- np.diag(temp1) @ temp2)\n",
    "\n",
    "# Store Data to use for F\n",
    "dat = (M, z)\n",
    "\n",
    "M_mean = np.mean(M, axis = 0)\n",
    "eigen, _ = np.linalg.eig(M_mean.T @ M_mean)\n",
    "L_total = np.sqrt(np.max(eigen))\n",
    "\n",
    "mu_total = 1\n",
    "\n",
    "# Commpute the Optimal Point\n",
    "x_optimal = - np.linalg.inv(np.mean(M, axis = 0)) @ np.mean(z, axis = 0)\n",
    "\n",
    "# Uniform Sampling parameters\n",
    "p_uniform = np.ones(n)/n\n",
    "delta_uniform = 2 * (L**2)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "d2633e1a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(18, 1.6666666666666667)"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "delta_uniform, L_total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "1723bd3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def operator_F(x, data, i):\n",
    "    return data[0][i] @ x + data[1][i]\n",
    "\n",
    "def SPEG_constant(x_0, x_optimal, L, mu, delta, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    omega_hat = min(.25/L, mu/(18 * delta))\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "        op = operator(x_0, data, i)\n",
    "        x = x_0 - omega_hat * op\n",
    "        \n",
    "        for k in range(iterations):\n",
    "            x_mid = x - omega_hat * op\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_mid, data, i)\n",
    "            x = x - omega_hat * op\n",
    "            \n",
    "            relative_error[trial, k] = np.sum((x - x_optimal)**2)/ initial_error\n",
    "\n",
    "    return np.mean(relative_error, axis = 0)\n",
    "\n",
    "def SPEG_Hsieh(x_0, x_optimal, L, mu, gamma, b, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "        op = operator(x_0, data, i)\n",
    "        \n",
    "        x = x_0 - (gamma * op)/b\n",
    "        \n",
    "        for k in range(iterations):\n",
    "            step = gamma/(k + 1 + b)\n",
    "            x_mid = x - step * op\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_mid, data, i)\n",
    "            x = x - step * op\n",
    "            \n",
    "            relative_error[trial, k] = np.sum((x - x_optimal)**2)/ initial_error\n",
    "    \n",
    "\n",
    "    return np.mean(relative_error, axis = 0)\n",
    "        \n",
    "\n",
    "def SPEG_decreasing(x_0, x_optimal, L, mu, delta, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    omega_hat = min(1/(4 * L), mu / (18 * delta))\n",
    "    k_0 = math.ceil(iterations / 2)\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        if iterations <= 2/(mu * omega_hat):\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_0, data, i)\n",
    "            x = x_0 - omega_hat * op\n",
    "            for k in range(iterations):\n",
    "                x_mid = x - omega_hat * op\n",
    "                i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "                op = operator(x_mid, data, i)\n",
    "                x = x - omega_hat * op\n",
    "                \n",
    "                relative_error[trial, k] = np.sum((x - x_optimal)**2)/initial_error\n",
    "        else:\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_0, data, i)\n",
    "            x = x_0 - omega_hat * op\n",
    "            for k in range(iterations):\n",
    "                if k <= k_0:\n",
    "                    step = omega_hat\n",
    "                else:\n",
    "                    step = (4 * omega_hat)/(4 + mu * omega_hat * (k - k_0))\n",
    "                \n",
    "                x_mid = x - step * op\n",
    "                i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "                op = operator(x_mid, data, i)\n",
    "                x = x - step * op\n",
    "                \n",
    "                relative_error[trial, k] = np.sum((x - x_optimal)**2)/initial_error\n",
    "\n",
    "    \n",
    "    return np.mean(relative_error, axis = 0)\n",
    "\n",
    "\n",
    "def SPEG_switch(x_0, x_optimal, L, mu, delta, n, data, operator, prob, iterations = 1000, trials = 5):\n",
    "    relative_error = np.zeros((trials, iterations))\n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    \n",
    "    omega_hat = min(1/(4*L), mu/(18 * delta))\n",
    "    k_0 = math.ceil(4/(mu * omega_hat))\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "        op = operator(x_0, data, i)\n",
    "        x = x_0 - omega_hat * op\n",
    "        \n",
    "        for k in range(iterations):\n",
    "            if k <= k_0:\n",
    "                step = omega_hat\n",
    "            else:\n",
    "                step = (4*k + 2)/(mu * (k + 1)**2)\n",
    "                \n",
    "            x_mid = x - step * op\n",
    "            i = np.random.choice(range(n), 1, p = prob)[0]\n",
    "            op = operator(x_mid, data, i)\n",
    "            x = x - step * op\n",
    "            \n",
    "            relative_error[trial, k] = np.sum((x - x_optimal)**2)/ initial_error\n",
    "            \n",
    "    return np.mean(relative_error, axis = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "bc59ead0",
   "metadata": {},
   "outputs": [],
   "source": [
    "no_iterations = int(1e5)\n",
    "\n",
    "SPEG_constant_error = SPEG_constant(np.zeros(d), x_optimal, L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = no_iterations, trials = 10)\n",
    "SPEG_switch_errror =  SPEG_switch(np.zeros(d), x_optimal, L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = no_iterations, trials = 10)\n",
    "#SPEG_error = SPEG_decreasing(np.zeros(2*d), x_optimal, L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = no_iterations, trials = 10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "416cfd28",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZMAAAELCAYAAAAcKWtPAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAABvFUlEQVR4nO2dd3gU1drAfychNKlKVVSagNRQBUUCIkUEEQSEC0i5XsTO9cInForlXhFR0YuCgoAdFBsgV0UgIqLSREEkCAjSq5QAgZC83x9nZ3e2ZnczySbh/J5nnpk5c8p7ZmfnndPeV4kIBoPBYDBkh7hYC2AwGAyG/I9RJgaDwWDINkaZGAwGgyHbGGViMBgMhmxjlInBYDAYsk2hWAuQW5QrV06qVq0aVdpTp05x0UUXOStQHsfU+cLA1Lngk936rl279rCIlM8q3gWjTKpWrcqaNWuiSpucnEzbtm0B2HV8FwCXl77cKdHyJPY6XyiYOl8YXGh1zm59lVI7w4l3wSgTpxj4yUAAkgcnx1YQg8FgyEMYZRIhj7d5PNYiGAwGQ57DKJMIubH6jbEWwWAwGPIcRplEyPa/tgNQvWz1GEtiAEhPT2f37t2kpaVlO6/SpUvz22+/OSBV/sHUueATbn2LFi1KlSpVSEhIiKoco0wiZOhnQwEzZpJX2L17NyVLlqRq1aoopbKV18mTJylZsqRDkuUPTJ0LPuHUV0Q4cuQIu3fvplq1alGVky+ViVLqIuBV4ByQLCLvOl3G4YRKlDt/AIC2tvAnEsvCJ584XZwhStLS0hxRJAbDhYxSiksuuYRDhw5FnUeeWbSolJqplDqolNroE95ZKZWilNqqlBrtCu4JzBORfwC35IQ8liLxJWn9XyRVTcqJIg1RYhSJwZB9svs/yjPKBJgNdLYHKKXigVeAm4C6QD+lVF2gCrDLFS0jF2Uk5RJIOZySm0UaDAZDnifPdHOJyHKlVFWf4BbAVhHZDqCUmgN0B3ajFcp6QihEpdQwYBhAxYoVSU5ODluetkHC7+oGx965ncmJk8POKz+Smpoa0f2KFaVLl+bkyZOO5JWRkRFVXgcOHODhhx9m3bp1FClShCuuuIIJEyZw1VVXOSLXwoULqVmzJnXq1Ikq/c6dO/nxxx/p06eP37X09HSGDx/O8uXLUUpRpEgR3nzzTapWrcqkSZMYOXJkdsX3YtGiRWzevJmHHnooW/ns37+f+++/n7vuuotx48YBsH37dipXrkyxYsWoV68eSUlJrFu3jueff94rbbS/c06SkZFBUlISlStX5sMPP/S7vmXLFu6++25+/vlnxo4dywMPPADAuXPnuOWWW1i4cCGFCgV+nUdS37S0tOj/9yKSZzagKrDRdt4LmGE7HwhMAS4CZgFTgf7h5N20aVOJCAi4fXc58t2f30WWVz5k2bJlsRYhLDZt2hR23MREkbvvFtm7N/D1EydORFx+ZmamtGzZUqZOneoO++mnn2T58uUR5xWMQYMGyYcffhh1+mXLlsnNN98c8Nobb7wht912m2RkZIiIyK5du+To0aMiInLRRRdFXWZOM3LkSPn000+9wpKSkmT16tXu81mzZsm9997rlzaa39nO+fPns5U+EM8//7z069cv6O904MABWbVqlTz66KPy3HPPeV0bP368vPPOO0HzjqS+gf5PwBoJ4x2bl7q5AhGoE09E5JSIDBGRuyWLwXelVDel1OvHjx93RKBrd8G1l1/rSF6G3GX9enjjDaheHe65B/bty36ey5YtIyEhgeHDh7vDEhMTuf766xERRo0aRf369WnQoAFz584FPOYtevXqRZ06dejfv7/1scTo0aOpW7cuDRs2ZOTIkaxcuZL58+czatQoEhMT2bZtG9OnT6d58+Y0atSI2267jdOnTwMwePBgHnjgAa699lqqV6/OvHnz3Hl+++23JCYm8uKLL3rJf+DAASpXrkxcnH4VVKlShbJlyzJ69GjOnDlDYmIi/fv3B+Cdd96hRYsWJCYmctddd5GRoXuYS5Qowb/+9S+aNGlC+/bt3YO4L7/8srsuffv2BWD27Nncd9997vtkbcWKFeObb77h1KlTDB06lObNm9O4cWM+++yzgPf9o48+onPnzgGv2dm7dy+dO3fmqquu4v/+7//c4V999RWtWrWiSZMm9O7dm9TUVACWLFlC48aNadCgAUOHDuXs2bOANsf05JNP0rp1az788MOg6atWrcqjjz5Kq1ataNasGevWraNTp07UqFGDadOmBZRx9+7dfP7559x5551B61GhQgWaN28ecNrurbfeyrvvOj4HKXLC0Ti5teHfMmkFfGk7fwR4JJq8I22ZHCpUMWDLZMPVl8iGAxsiyis/UhBbJvafsnBhkaJFvVsq0XyxvvTSSzJi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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "omega_hat = min(1/(4*L_total), mu_total/(18 * delta_uniform))\n",
    "switch_point = math.ceil(4/(mu_total * omega_hat))\n",
    "\n",
    "\n",
    "markers_on = np.arange(0, no_iterations, no_iterations/10).astype(int)\n",
    "plt.plot(np.arange(len(SPEG_constant_error)), SPEG_constant_error, '-b>', markevery = markers_on ,label = 'Constant Stepsize (Theorem 4.1)', linewidth = 0.5)\n",
    "plt.plot(np.arange(len(SPEG_switch_errror)), SPEG_switch_errror, '-rs', markevery = markers_on, label = 'Decreasing Stepsize (Theorem 4.4)', linewidth = 0.5)\n",
    "#plt.plot(np.arange(len(SPEG_decreasing_error)), SPEG_decreasing_error, label = 'SPEG Decreasing')\n",
    "plt.axvline(x= switch_point, color = 'green', linestyle = 'dotted', label = 'Change of step-size')\n",
    "\n",
    "plt.yscale('log')\n",
    "plt.grid(True)\n",
    "plt.ylabel(\"Relative Error\", fontsize = 15)\n",
    "plt.xlabel(\"Number of Iterations\", fontsize = 15)\n",
    "plt.legend(fontsize = 10)\n",
    "\n",
    "plt.savefig(f'constant_switch_comparison_condition = {L_total/mu_total}.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cc09831c",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
