{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "7d41a417",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import math"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "a1e09491",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Data Generation\n",
    "\n",
    "n, d = 100, 30\n",
    "\n",
    "max_eq = 1\n",
    "\n",
    "L_max = 10\n",
    "L = np.ones(n)\n",
    "L[0:max_eq] = L_max\n",
    "\n",
    "mu = .1 * np.ones(n)\n",
    "\n",
    "# Generate A\n",
    "\n",
    "A = []\n",
    "for k in range(n):\n",
    "    evalues = np.random.uniform(mu[k], L[k], d)\n",
    "    if k < max_eq:\n",
    "        evalues[0] = L_max\n",
    "    elif k == (n - 1):\n",
    "        evalues[0] = mu[k]\n",
    "    rndm_mx = np.random.normal(0, 1, (d, d))\n",
    "    _, Q = np.linalg.eig(rndm_mx.T @ rndm_mx)\n",
    "    \n",
    "    A.append(Q @ np.diag(evalues) @ Q.T)\n",
    "\n",
    "# Generate C\n",
    "\n",
    "C = []\n",
    "for k in range(n):\n",
    "    evalues = np.random.uniform(mu[k], L[k], d)\n",
    "    if k < max_eq:\n",
    "        evalues[0] = L_max\n",
    "    elif k == (n - 1):\n",
    "        evalues[0] = mu[k]\n",
    "    rndm_mx = np.random.normal(0, 1, (d, d))\n",
    "    _, Q = np.linalg.eig(rndm_mx.T @ rndm_mx)\n",
    "    \n",
    "    C.append(Q @ np.diag(evalues) @ Q.T)\n",
    "\n",
    "# Generate B\n",
    "\n",
    "mu = np.zeros(n)\n",
    "B = []\n",
    "for k in range(n):\n",
    "    evalues = np.random.uniform(mu[k], L[k], d)\n",
    "    if k < max_eq:\n",
    "        evalues[0] = L_max\n",
    "    elif k == (n - 1):\n",
    "        evalues[0] = mu[k]\n",
    "    rndm_mx = np.random.normal(0, 1, (d, d))\n",
    "    _, Q = np.linalg.eig(rndm_mx.T @ rndm_mx)\n",
    "    \n",
    "    B.append(Q @ np.diag(evalues) @ Q.T)\n",
    "\n",
    "# Generate a and c from Normal(0,1)\n",
    "a = []\n",
    "c = []\n",
    "for k in range(n):\n",
    "    a.append(np.random.normal(0,1,d))\n",
    "    c.append(np.random.normal(0,1,d))\n",
    "    \n",
    "data = (A, B, C, a, c)\n",
    "\n",
    "# Store A,B,C in a Block Matrix \n",
    "M = []\n",
    "z = []\n",
    "L_M = []\n",
    "for k in range(n):\n",
    "    M.append(np.block([[data[0][k], data[1][k]], [-data[1][k].T, data[2][k]]]))\n",
    "    z.append(np.concatenate((data[3][k], data[4][k])))\n",
    "    eigenvalues, _ = np.linalg.eig(M[k].T @ M[k])\n",
    "    L_M.append(np.sqrt(max(eigenvalues)))\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",
    "mu_total = np.sqrt(np.min(eigen))\n",
    "\n",
    "# Store the Data to use for Operator F\n",
    "dat = (M,z)\n",
    "\n",
    "# Compute the Optimal point\n",
    "x_optimal = - np.linalg.inv(M_mean) @ np.mean(z, axis = 0)\n",
    "\n",
    "# Uniform Sampling parameters\n",
    "p_uniform = np.ones(n)/n\n",
    "delta_uniform = 2 * np.mean(np.array(L_M)**2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "780c5338",
   "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",
    "        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": 5,
   "id": "8e83bd1b",
   "metadata": {},
   "outputs": [],
   "source": [
    "SPEG_constant_error = SPEG_constant(np.zeros(2*d), x_optimal, L_total,.3, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 100000, trials = 10)\n",
    "SPEG_switch_errror =  SPEG_switch(np.zeros(2*d), x_optimal, L_total, .3, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 100000, trials = 10)\n",
    "# SPEG_decreasing_error = SPEG_decreasing(np.zeros(2*d), x_optimal, L_total, .08, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 1000000, trials = 10)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e363625c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "5739"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "L = L_total\n",
    "mu = .3\n",
    "delta = delta_uniform\n",
    "omega_hat = min(1/(4*L), mu/(18 * delta))\n",
    "math.ceil(4/(mu * omega_hat))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "8d73bf7f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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FWLr1q2sXbuWRYsWMWPGDNatW5cu7ODBg/n2229p2LAh8+bNS9Vpah9vRumnleWee+7hnnvuISQkhG+//ZZ27dqly8ugQYN4/fXXM43POsx05cqVbNiwgWXLljFp0qRUSg/ENXqfPn34+OOPueWWW1LS+eqrr6hVq5bT+AsXLuyyxwNH91lrTYcOHfjiiy9SXbtz584M47K6cXcWPm2afn5+qdL38/MjMTF1xXTz5s0sW7aMVatWERcXx+XLlxkwYAALFixwKX9Wbty4QXBwcJbCeBqfsUy01v201hW01gFa64pa608sx1dprWtqratprf/nJeHcjqLZrc1odmszDwhjcIfAQChcGIYNgyNH4P33oXz57MfXtm1bbty4wccff5xybNu2baxfv57WrVuzePFikpKSOHv2LBs2bKB58+ZO44qNjeXSpUt06dKFadOmpRRuN910U6pa55UrV6hQoQIJCQkutZWnDW/Pzp07OXnyJADJycns3r2b2267DYCAgAASEhIAqeEvXbqUM2fOAHDhwoWU2nhycnKKQv38889p1aoVycnJREdH06ZNG6ZMmcLFixdT+hWsDBkyhCFDhnD33XenHOvUqRPvvfdeiiL8/fff08lcs2ZNjh49mmm+ndGiRQs2b97M4cOHAbh27RoHDx6kdu3aHD16NOX4/PnzCQ9PPzfMWfjs8Prrr3P8+HGOHj3KokWLaNu2bZYVyfnz5ylbtqxbrlA8gc9YJvmdnTE7AQgrH+ZVOQoyYWFw551iibijQOxRSvHNN98wZswYJk+eTHBwMKGhoUybNo3WrVvzyy+/0LBhQ5RSTJkyhfLly7N//36HcV25coVu3boRFxeH1jqls7xv3748/PDDTJ8+naVLlzJp0iTuuOMObrvtNurXr59p80aDBg0oVKgQDRs2ZPDgwTz55JMp586ePcuYMWNSOpqbN2+e0kE+fPhwGjRoQOPGjVm4cCGvvvoqHTt2JDk5mYCAAN5//31uu+02ihYtyt69e2nSpAklSpRIUaADBgzg0qVLaK158sknKVmyZEq6x44dY+nSpRw8eJA5c+YAMHv2bF566SXGjBlDgwYN0FoTGhrKihUrUuWnaNGiVKtWjcOHD1O9evWsPTCgbNmyzJs3j379+qXk+9VXX6VmzZrMnTuX3r17k5iYSLNmzVINrHAlvCexdtiPGDGC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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure()\n",
    "markers_on = [10000, 20000, 30000, 40000, 50000, 60000, 70000, 80000, 90000]\n",
    "#plt.plot(np.arange(len(SPEG_decreasing_error)), SPEG_decreasing_error, label = 'Decreasing Step')\n",
    "plt.plot(np.arange(len(SPEG_constant_error)), SPEG_constant_error, '-b>', markevery = markers_on ,label = 'Constant Stepsize (Theorem 4.1)', linewidth = 0.1)\n",
    "plt.plot(np.arange(len(SPEG_switch_errror)), SPEG_switch_errror, '-rs', markevery = markers_on, label = 'Decreasing Stepsize (Theorem 4.4)', linewidth = 0.1)\n",
    "plt.axvline(x= 5874, color = 'green', linestyle = 'dotted', label = 'Change of step-size')\n",
    "\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",
    "#plt.title(\"Constant vs Switch\")\n",
    "plt.savefig('Constant vs Switch Experiment 2.pdf', format = 'pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "8ba1057b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.2275685273660395"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "L_total/mu_total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d6b49dbc",
   "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
}
