{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "33bbc288",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import math"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "642f5f31",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Data Generation\n",
    "\n",
    "n, d = 100, 30\n",
    "\n",
    "max_eq = 1\n",
    "\n",
    "L_max = 1\n",
    "L = np.ones(n)\n",
    "L[0:max_eq] = L_max\n",
    "\n",
    "mu = .6 * 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": "561d9ee5",
   "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",
    "    #k_0 = switch\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": "09094223",
   "metadata": {},
   "outputs": [],
   "source": [
    "# SPEG_constant_error = SPEG_constant(np.zeros(2*d), x_optimal, L_total,.1, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 50000, trials = 10)\n",
    "SPEG_switch_errror =  SPEG_switch(np.zeros(2*d), x_optimal,L_total, mu_total, delta_uniform, n, (M,z), operator_F, p_uniform, iterations = 50000)\n",
    "SPEG_Hsieh_error = SPEG_Hsieh(np.zeros(2*d), x_optimal, L_total, mu_total, 20, 100, n, (M,z), operator_F, p_uniform, iterations = 50000)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "e9c72a3b",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "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IEP9V65c4emnn2bz5s288cYbAEyaNIkWLVrw9ttvc/78edq1a0ezZs1ITEzkzz//5I8//iAoKIioqCiioqI4fPgwZcuWZf78+QDExcXZ7sWlS5do166d7YE+ZMgQWrZsSWxsLKNHj3Z6b+xZu3YtdevWdehHfHw8qampDmUpKSkkJCSwZs0aVq5cyXPPPceSJUuYM2eO09/vr7/+YtOmTfz666+cOnWK9u3b06hRIxISEtiwYQN//PEHYWFhzJw50+n1CQkJ/PXXX8TExFC6dGnq16/Pvffey5o1a3jvvfd44403eO211xz6cuzYMRYuXMj333/P+vXruXz5cobfB+C9996jU6dOtukr+3Pq1q3LmjVr6NKly7X/qLMgJSXFadvpSUxMzPb/vK8rEGePA621PguMutbFWusPgQ8BIiMjdY4Cih0/zrZ//uHT058CEN0tOvt15EM8EXRu165dNn90rV172ANcvSonfv55UfbtK8oPPzg/z1UFkt4nXmtNcHAw0dHR7Ny5k6VLlwLycDx+/Djr169nxIgRhIaGOlzv7+/P4MGDCQkJYfv27ezatYs777wTkH/aChUqEBISwubNmxk8eDDnz58nPj6e22+/nZCQEFq3bs3YsWPp168fvXv3JiQkhKCgIIoUKUJISAjFihWjZ8+elClThjJlyhAaGkpCQgJbtmzhzjvvtL3Z9urVi2LFimXol7P6AwMDKVq0qO3c6OhofvzxR2bOnGm511c5d+4cgYGBdOrUCWsE6759+7Jlyxa6du3KxIkTeemll+jevTutW7e23YsSJUrY6p06dSohISE89thjWd4be2JjY4mIiHAoDw4Oxs/Pz6HM39+fAQMG2O7hhAkTCAkJ4eeff+bvv//O8Ptt2rSJQYMGUapUKfz9/YmKimLXrl1cd911tlEEkOn1QUFBNGnShOrVqwNQrVo1evToQUhICE2aNOH333/P0Jdnn32WadOmUapUKQICAihevHiGc44dO8bSpUuJjo62jbrsz6lcuTI//PBDrtetuLr2JTAwkIYNG2arbl9XIEcA+/C3NwHHslOBUqoH0KNatWo5k0AplNa83vH1nF1vcIpS4Or/RdGi4O8P990HEye6fp0r7N+/H39/f8qVK4fWmnfffZfbb7/d4ZwVK1Zk6uZoDZOttaZu3br8/vvvGc4ZOnQo3333HQ0a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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig = plt.figure()\n",
    "\n",
    "markers_on = [10000,20000,30000,40000]\n",
    "plt.plot(np.arange(len(SPEG_switch_errror)), SPEG_switch_errror, '-b>', markevery = markers_on, label= 'Decreasing stepsize (Theorem 4.4)', linewidth = 0.1)\n",
    "plt.plot(np.arange(len(SPEG_Hsieh_error)), SPEG_Hsieh_error, '-rs', markevery = markers_on, label= 'Decreasing stepsize (Hsieh et al, 2019)', linewidth = 0.1)\n",
    "# plt.plot(np.arange(len(SPEG_decreasing_error)), SPEG_decreasing_error, label= 'SPEG decreasing')\n",
    "# plt.plot(np.arange(len(SPEG_switch_error)), SPEG_switch_error, label= 'SPEG switch')\n",
    "plt.axvline(x= 305, 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(\"Comparison on Non-Interpolated Data\")\n",
    "plt.savefig('Constant vs Hsieh Steps on Non-Interpolated Data.pdf', format = 'pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "ddb198e1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(1.0931944060644276, 0.25352326473621306)"
      ]
     },
     "execution_count": 46,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "1/mu_total, 1/(4 * L_total)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "c1bb58a4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "305"
      ]
     },
     "execution_count": 49,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mu = mu_total\n",
    "omega_hat = min(1/(4*L_total), mu/(18 * delta_uniform))\n",
    "math.ceil(4/(mu * omega_hat))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "b1b25620",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(50000,)"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "SPEG_switch_errror.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "4e7985fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "dictionary = {'A':A, 'B': B, 'C':C, 'a': a, 'c':c}\n",
    "\n",
    "import pickle\n",
    "with open('Non Interpolated Data.pickle', 'wb') as handle:\n",
    "    pickle.dump(dictionary, handle, protocol=pickle.HIGHEST_PROTOCOL)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "d6fc8b46",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('Non Interpolated Data.pickle', 'rb') as handle:\n",
    "    b = pickle.load(handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "922121f3",
   "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
}
