{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "17239386",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import math"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "9323660c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Data Generation\n",
    "\n",
    "n, d = 100, 30\n",
    "\n",
    "max_eq = 1\n",
    "\n",
    "L_max = 20\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 and computation of Lipchitz Constants\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",
    "\n",
    "# Computation of \\mu\n",
    "muA_list = list(map(lambda X: np.min(np.linalg.eig(X)[0]), A))\n",
    "muB_list = list(map(lambda X: np.min(np.linalg.eig(X)[0]), B))\n",
    "mu_list = list(map(lambda x,y: min(x,y), muA_list, muB_list))\n",
    "\n",
    "A_total = np.mean(A, axis = 0)\n",
    "mu_A = np.min(np.linalg.eig(A_total)[0])\n",
    "B_total = np.mean(B, axis = 0)\n",
    "mu_B = np.min(np.linalg.eig(B_total)[0])\n",
    "mu_total = min(mu_A, mu_B)\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)\n",
    "\n",
    "# Compute Variance\n",
    "var = []\n",
    "for k in range(n):\n",
    "    var.append(np.linalg.norm(M[k] @ x_optimal + z[k]))\n",
    "\n",
    "\n",
    "# Important Sampling\n",
    "epsilon = 1e-8\n",
    "sigma_opt = np.mean(var)**2\n",
    "#alpha = np.mean(L_M)/(np.mean(L_M) + ((2 * sigma_opt)/ (mu_total * epsilon)))\n",
    "alpha = 1\n",
    "p_important = alpha * (L_M/ np.sum(L_M))+ (1 - alpha) * (var/np.sum(var))\n",
    "delta_important = (2/(n**2))*np.sum(np.array(L_M)**2/p_important)\n",
    "#p_important = alpha * (L/ np.sum(L))+ (1 - alpha)/n\n",
    "#delta_important = (2/(n**2))*np.sum(np.array(L)**2/p_important)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "29cd6cf7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def operator_F(x, data, sample):\n",
    "    result = 0\n",
    "    for i in sample:\n",
    "        result += data[0][i] @ x + data[1][i]\n",
    "    return result/len(sample)\n",
    "\n",
    "def SPEG_constant(x_0, x_optimal, L, mu, delta, batchsize, n, data, operator, prob, iterations = 1000, trials = 10):\n",
    "    relative_error = []\n",
    "    total_trajectory = []\n",
    "    \n",
    "    initial_error = np.sum((x_0 - x_optimal)**2)\n",
    "    omega_hat = min(1/(4 * L), mu/(18 * delta))\n",
    "    \n",
    "    for trial in range(trials):\n",
    "        error = []\n",
    "        trajectory = []\n",
    "        \n",
    "        sample = np.random.choice(n, batchsize, p = prob)\n",
    "        op = operator(x_0, data, sample)\n",
    "        x = x_0 - omega_hat * op\n",
    "        \n",
    "        for iteration in range(iterations):\n",
    "            x_mid = x - omega_hat * op\n",
    "            sample = np.random.choice(n, batchsize, p = prob)\n",
    "            op = operator(x_mid, data, sample)\n",
    "            x = x - omega_hat * op\n",
    "            \n",
    "            trajectory.append(x)\n",
    "            error.append(np.sum((x - x_optimal)**2)/initial_error)\n",
    "            \n",
    "        relative_error.append(error)\n",
    "        total_trajectory.append(trajectory)\n",
    "        \n",
    "        return np.mean(relative_error, axis = 0), np.mean(total_trajectory, axis = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "id": "32939618",
   "metadata": {},
   "outputs": [],
   "source": [
    "no_iterations = 1000\n",
    "\n",
    "IS_err, IS_trajectory = SPEG_constant(np.ones(2*d), x_optimal, L_total, mu_total, delta_important, 1, n, (M,z), operator_F, p_important, iterations= no_iterations)\n",
    "US_err, US_trajectory = SPEG_constant(np.ones(2*d), x_optimal, L_total, mu_total, delta_uniform, 1, n, (M,z), operator_F, p_uniform, iterations= no_iterations)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "be42a617",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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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 = np.arange(0, no_iterations, no_iterations/10).astype(int)\n",
    "plt.plot(np.arange(len(IS_err)), IS_err, '-bD', markevery = markers_on, label = 'Important')\n",
    "plt.plot(np.arange(len(US_err)), US_err, '-r>', markevery = markers_on, label = 'Uniform')\n",
    "plt.yscale('log')\n",
    "plt.grid(True)\n",
    "plt.ylabel(\"Relative Error\", fontsize = 20)\n",
    "plt.xlabel(\"Iterations\", fontsize = 20)\n",
    "plt.legend()\n",
    "#plt.title(\"Sampling\")\n",
    "plt.savefig(f\"Uniform vs Important Sampling with Lmax = {L_max} .pdf\", format=\"pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1da1d769",
   "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
}
