{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.23770981  0.19158669  0.33925719  0.47224395  0.75410794]\n"
     ]
    }
   ],
   "source": [
    "import itertools\n",
    "from tqdm import tqdm\n",
    "import autograd.numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from pathos.multiprocessing import ProcessingPool\n",
    "\n",
    "import sys\n",
    "sys.path.append('../')\n",
    "import expUtils\n",
    "\n",
    "def oneTrial(noise, U, S, theta, lams):\n",
    "    X = U @ np.diag(S)\n",
    "    Y = X @ theta + noise\n",
    "    Y -= Y.mean()\n",
    "    return expUtils.runTrialUOnlyNoisy(U, S, Y, lams)\n",
    "D = 5\n",
    "thetaStar = np.random.normal(size=D); thetaStar /= np.linalg.norm(thetaStar)\n",
    "print(thetaStar)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Making data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  0%|          | 0/4000 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Done making data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 4000/4000 [01:05<00:00, 61.32it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Making data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 6/4000 [00:00<01:13, 54.23it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Done making data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 4000/4000 [01:07<00:00, 59.70it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Making data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 6/4000 [00:00<01:10, 56.34it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Done making data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 4000/4000 [01:05<00:00, 60.71it/s]\n"
     ]
    }
   ],
   "source": [
    "Ns = [20,35,50,]\n",
    "\n",
    "nUs = 4000\n",
    "nSs = 25\n",
    "nCores = 10\n",
    "\n",
    "\n",
    "nonQVX = np.empty((len(Ns), nUs,nSs), dtype=np.bool)\n",
    "\n",
    "#losses = np.empty((nSs,nUs,nYs,lams.shape[0]))\n",
    "alphas = np.logspace(np.log10(1e-2),np.log10(10),nSs)\n",
    "Ss = np.empty((nSs,D))\n",
    "for ss in range(nSs):\n",
    "    Ss[ss] = np.exp(np.arange(D)*alphas[ss])\n",
    "    Ss[ss] /= Ss[ss].max()\n",
    "\n",
    "for nn, N in enumerate(Ns):\n",
    "    lams = np.logspace(-8, 3.0, 200) * N\n",
    "    lams = np.append(0.0, lams)\n",
    "\n",
    "    print('Making data')\n",
    "    Us = np.empty((nUs,N,D))\n",
    "    for uu in range(nUs):\n",
    "        Us[uu] = expUtils.getUFast(N, D)\n",
    "\n",
    "\n",
    "    noises = np.empty((nUs,N))\n",
    "    for uu in range(nUs):\n",
    "        E = np.random.normal(scale=5.0, size=N)\n",
    "        #E -= Us[uu] @ Us[uu].T @ E\n",
    "        #E /= np.linalg.norm(E)\n",
    "        #noises[uu] = E*noiseNorm\n",
    "        noises[uu] = np.random.normal(scale=0.5, size=N)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    print('Done making data')\n",
    "\n",
    "    for uu in tqdm(range(nUs)):\n",
    "        res = ProcessingPool(nCores).map(oneTrial,\n",
    "                                         itertools.cycle([noises[uu]]),\n",
    "                                         itertools.cycle([Us[uu]]),\n",
    "                                         Ss,\n",
    "                                         itertools.cycle([thetaStar]),\n",
    "                                         itertools.cycle([lams]))\n",
    "        nonQVX[nn,uu] = np.array([r[1] for r in res])\n",
    "        #losses[ss] = np.array([r[0] for r in res])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "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"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[[0.         0.         0.         ... 0.33333333 0.33333333 0.33333333]\n",
      " [0.         0.         0.         ... 0.66666667 0.66666667 0.66666667]\n",
      " [0.         0.         0.         ... 0.66666667 0.66666667 0.66666667]\n",
      " ...\n",
      " [0.         0.         0.         ... 0.         0.         0.        ]\n",
      " [0.         0.         0.         ... 0.66666667 0.66666667 0.66666667]\n",
      " [0.         0.         0.         ... 0.33333333 0.33333333 0.33333333]]\n"
     ]
    }
   ],
   "source": [
    "colors = ['b', 'r', 'y']\n",
    "for nn in range(len(Ns)):\n",
    "    \n",
    "    plt.scatter(np.log(np.abs((Ss - np.ones(D))).sum(1)), nonQVX[nn].mean(0), label=f'N={Ns[nn]}', c=colors[nn])\n",
    "    #try:\n",
    "    #    biggestInd = np.where(nonQVX[nn].mean(0) == 0)[0].max()\n",
    "    #    plt.gca().axvline(np.log(np.abs((Ss - np.ones(D))).sum(1))[biggestInd])\n",
    "    #except:\n",
    "    #    print(nn)\n",
    "#plt.xscale('log')\n",
    "plt.xlabel(r'$log || S - \\mathbf{1}||_1$', fontsize=18)\n",
    "plt.ylabel(r'Fraction non-QVX', fontsize=18)\n",
    "plt.legend(fontsize=16)\n",
    "plt.xticks(fontsize=14)\n",
    "plt.yticks(fontsize=14)\n",
    "plt.tight_layout()\n",
    "plt.savefig('/home/will/projects/model_selection/neurips2021/figs/howCloseToUniform.png')\n",
    "plt.show()\n",
    "print(nonQVX.mean(0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.0000e+00 0.0000e+00 0.0000e+00 0.0000e+00 0.0000e+00 0.0000e+00\n",
      " 0.0000e+00 0.0000e+00 0.0000e+00 2.5000e-02 5.0000e-02 4.2500e-01\n",
      " 1.0750e+00 3.1750e+00 8.3500e+00 1.5125e+01 2.5750e+01 3.8075e+01\n",
      " 5.0375e+01 5.4875e+01 5.1200e+01 5.3750e+01 4.0375e+01 3.8125e+01\n",
      " 3.7350e+01]\n",
      "[0.09851652 0.13072379 0.17317679 0.22891992 0.3017404  0.3962269\n",
      " 0.51772934 0.67212938 0.86529356 1.10205931 1.38464221 1.710528\n",
      " 2.070306   2.44649115 2.81483563 3.14916528 3.42868257 3.6438655\n",
      " 3.79689955 3.89704345 3.95579996 3.98503606 3.99637461 3.99944602\n",
      " 3.9999546 ]\n"
     ]
    }
   ],
   "source": [
    "print(nonQVX[2].mean(0) * 100)\n",
    "print(np.abs(Ss - np.ones(D)).sum(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "N = 150\n",
    "D = 5\n",
    "nUs = 4000\n",
    "nSs = 25\n",
    "nCores = 10\n",
    "\n",
    "noiseNorm = 0.1 * N\n",
    "\n",
    "\n",
    "    \n",
    "thetaStar = np.random.normal(size=D); thetaStar /= np.linalg.norm(thetaStar)\n",
    "lams = np.logspace(-8, 3.0, 200) * N\n",
    "lams = np.append(0.0, lams)\n",
    "\n",
    "nonQVX = np.empty((nUs,nSs), dtype=np.bool)\n",
    "\n",
    "#losses = np.empty((nSs,nUs,nYs,lams.shape[0]))\n",
    "\n",
    "print('Making data')\n",
    "Us = np.empty((nUs,N,D))\n",
    "for uu in range(nUs):\n",
    "    Us[uu] = expUtils.getUFast(N, D)\n",
    "    \n",
    "alphas = np.logspace(np.log10(1e-4),np.log10(10),nSs)\n",
    "Ss = np.empty((nSs,D))\n",
    "for ss in range(nSs):\n",
    "    Ss[ss] = np.exp(np.arange(D)*alphas[ss])\n",
    "    Ss[ss] /= Ss[ss].max()\n",
    "    \n",
    "noises = np.empty((nUs,N))\n",
    "for uu in range(nUs):\n",
    "    E = np.random.normal(scale=5.0, size=N)\n",
    "    #E -= Us[uu] @ Us[uu].T @ E\n",
    "    #E /= np.linalg.norm(E)\n",
    "    #noises[uu] = E*noiseNorm\n",
    "    noises[uu] = np.random.normal(scale=0.5, size=N)\n",
    "    \n",
    "    \n",
    "    \n",
    "\n",
    "print('Done making data')\n",
    "\n",
    "for uu in tqdm(range(nUs)):\n",
    "    res = ProcessingPool(nCores).map(oneTrial,\n",
    "                                     itertools.cycle([noises[uu]]),\n",
    "                                     itertools.cycle([Us[uu]]),\n",
    "                                     Ss,\n",
    "                                     itertools.cycle([thetaStar]),\n",
    "                                     itertools.cycle([lams]))\n",
    "    nonQVX[uu] = np.array([r[1] for r in res])\n",
    "    #losses[ss] = np.array([r[0] for r in res])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.scatter(np.abs((Ss - np.ones(D))).sum(1), nonQVX.mean(0))\n",
    "plt.xscale('log')\n",
    "plt.show()\n",
    "print(nonQVX.mean(0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.dirichlet(1e-3*np.ones(D))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
